summaryrefslogtreecommitdiffstats
path: root/runtime/regs.c
diff options
context:
space:
mode:
authoraskeshav <askeshav>2005-10-28 22:49:28 +0000
committeraskeshav <askeshav>2005-10-28 22:49:28 +0000
commit1c9db4fdf66fe88a731319b99942872fa567d742 (patch)
tree33a86507a7fcf9a9c771cf9a1801c8f81c566cf8 /runtime/regs.c
parent063b69fe3f1109ec81b5e4c1f1333a442ccb3734 (diff)
downloadsystemtap-steved-1c9db4fdf66fe88a731319b99942872fa567d742.tar.gz
systemtap-steved-1c9db4fdf66fe88a731319b99942872fa567d742.tar.xz
systemtap-steved-1c9db4fdf66fe88a731319b99942872fa567d742.zip
IA64 Runtime support patches. With this in place
Systemtap should now be able to build on Ia64. Includes supports for - function probes, return probes, function parameter access and dumping stack backtrace. Added by Anil S Keshavamurthy <Anil.s.keshavamurthy@intel.com>
Diffstat (limited to 'runtime/regs.c')
-rw-r--r--runtime/regs.c80
1 files changed, 80 insertions, 0 deletions
diff --git a/runtime/regs.c b/runtime/regs.c
new file mode 100644
index 00000000..51c46db3
--- /dev/null
+++ b/runtime/regs.c
@@ -0,0 +1,80 @@
+/* register access functions
+ * Copyright (C) 2005 Intel Corporation.
+ *
+ * This file is part of systemtap, and is free software. You can
+ * redistribute it and/or modify it under the terms of the GNU General
+ * Public License (GPL); either version 2, or (at your option) any
+ * later version.
+ */
+
+#ifndef _REG_C_ /* -*- linux-c -*- */
+#define _REG_C_
+
+#if defined __ia64__
+
+struct ia64_stap_get_arbsp_param {
+ unsigned long ip;
+ unsigned long *address;
+};
+
+static void ia64_stap_get_arbsp(struct unw_frame_info *info, void *arg)
+{
+ unsigned long ip;
+ struct ia64_stap_get_arbsp_param *lp = arg;
+
+ do {
+ unw_get_ip(info, &ip);
+ if (ip == 0)
+ break;
+ if (ip == lp->ip) {
+ unw_get_bsp(info, (unsigned long*)&lp->address);
+ return;
+ }
+ } while (unw_unwind(info) >= 0);
+ lp->address = 0;
+}
+
+static long ia64_fetch_register(int regno, struct pt_regs *pt_regs)
+{
+ struct ia64_stap_get_arbsp_param pa;
+
+ if (regno < 32 || regno > 127)
+ return 0;
+
+ pa.ip = pt_regs->cr_iip;
+ unw_init_running(ia64_stap_get_arbsp, &pa);
+ if (pa.address == 0)
+ return 0;
+
+ return *ia64_rse_skip_regs(pa.address, regno-32);
+}
+
+static void ia64_store_register(int regno,
+ struct pt_regs *pt_regs,
+ unsigned long value)
+{
+ struct ia64_stap_get_arbsp_param pa;
+ unsigned long rsc_save = 0;
+
+ if (regno < 32 || regno > 127)
+ return;
+
+ pa.ip = pt_regs->cr_iip;
+ unw_init_running(ia64_stap_get_arbsp, &pa);
+ if (pa.address == 0)
+ return;
+ *ia64_rse_skip_regs(pa.address, regno-32) = value;
+ //Invalidate all stacked registers outside the current frame
+ asm volatile( "mov %0=ar.rsc;;\n\t"
+ "mov ar.rsc=0;;\n\t"
+ "{\n\tloadrs;;\n\t\n\t\n\t}\n\t"
+ "mov ar.rsc=%1\n\t"
+ :"=r" (rsc_save):"r" (rsc_save):"memory");
+
+ return;
+}
+
+#endif /* if defined __ia64__ */
+
+
+#endif /* _REG_C_ */