[PATCH] 7/8 LTT for 2.5.34: SuperH trace support

From: Karim Yaghmour (karim@opersys.com)
Date: Tue Sep 10 2002 - 13:17:10 EST


This patch adds SuperH trace support. Here are the files modified:
arch/sh/config.in
arch/sh/kernel/entry.S
arch/sh/kernel/irq.c
arch/sh/kernel/process.c
arch/sh/kernel/sys_sh.c
arch/sh/kernel/traps.c
arch/sh/mm/fault.c
include/asm-sh/trace.h

diff -urN linux-2.5.34/arch/sh/config.in linux-2.5.34-ltt/arch/sh/config.in
--- linux-2.5.34/arch/sh/config.in Mon Sep 9 13:35:12 2002
+++ linux-2.5.34-ltt/arch/sh/config.in Mon Sep 9 19:06:34 2002
@@ -358,6 +358,8 @@
 fi
 endmenu
 
+source drivers/trace/Config.in
+
 mainmenu_option next_comment
 comment 'Kernel hacking'
 
@@ -366,6 +368,7 @@
 if [ "$CONFIG_SH_STANDARD_BIOS" = "y" ]; then
    bool 'Early printk support' CONFIG_SH_EARLY_PRINTK
 fi
+
 endmenu
 
 source security/Config.in
diff -urN linux-2.5.34/arch/sh/kernel/entry.S linux-2.5.34-ltt/arch/sh/kernel/entry.S
--- linux-2.5.34/arch/sh/kernel/entry.S Mon Sep 9 13:35:08 2002
+++ linux-2.5.34-ltt/arch/sh/kernel/entry.S Mon Sep 9 19:06:34 2002
@@ -370,6 +370,20 @@
         mov.l r10, @r14 ! set syscall_nr
         STI()
         !
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ ! TODO: for i386 this code only happens when not ptrace'd
+ mov r15, r4 ! pass pt_regs* as first arg
+ mov.l __trsen, r11 ! Call trace_real_syscall_entry()
+ jsr @r11 ! (will chomp R[0-7])
+ nop
+ ! Reload R4-R7 from kernel stack
+ mov.l @(OFF_R4,r15), r4 ! arg0
+ mov.l @(OFF_R5,r15), r5
+ mov.l @(OFF_R6,r15), r6
+ mov.l @(OFF_R7,r15), r7 ! arg3
+ mov.l @(OFF_R3,r15), r3 ! syscall_nr
+#endif
+
         stc k_current, r11
 #error mov.l @(tsk_ptrace,r11), r10 ! Is current PTRACE_SYSCALL'd?
 #error mov #PT_TRACESYS, r11
@@ -421,6 +435,14 @@
         ! In case of trace
 syscall_ret_trace:
         mov.l r0, @(OFF_R0,r15) ! save the return value
+
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ ! TODO: for i386 this code only happens when not ptrace'd
+ mov.l __trsex, r1 ! Call trace_real_syscall_exit()
+ jsr @r1
+ nop
+#endif
+
         mov.l __syscall_trace, r1
         mova ret_from_syscall, r0
         jmp @r1 ! Call syscall_trace() which notifies superior
@@ -504,6 +526,14 @@
         .long syscall_ret_trace
 __syscall_ret:
         .long syscall_ret
+
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+__trsen:
+ .long trace_real_syscall_entry
+__trsex:
+ .long trace_real_syscall_exit
+#endif
+
 __INV_IMASK:
         .long 0xffffff0f ! ~(IMASK)
 
@@ -536,6 +566,14 @@
 #endif
 syscall_ret:
         mov.l r0, @(OFF_R0,r15) ! save the return value
+
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ ! TODO: for i386 this code only happens when not ptrace'd
+ mov.l __trsex2, r1 ! Call trace_real_syscall_exit()
+ jsr @r1
+ nop
+#endif
+
         /* fall through */
 
 ENTRY(ret_from_syscall)
@@ -563,6 +601,11 @@
 #error .long do_signal
 __irq_stat:
         .long irq_stat
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+__trsex2:
+ .long trace_real_syscall_exit
+#endif
+
 
         .align 2
 restore_all:
diff -urN linux-2.5.34/arch/sh/kernel/irq.c linux-2.5.34-ltt/arch/sh/kernel/irq.c
--- linux-2.5.34/arch/sh/kernel/irq.c Mon Sep 9 13:35:08 2002
+++ linux-2.5.34-ltt/arch/sh/kernel/irq.c Mon Sep 9 19:06:34 2002
@@ -30,6 +30,8 @@
 #include <linux/init.h>
 #include <linux/seq_file.h>
 
+#include <linux/trace.h>
+
 #include <asm/system.h>
 #include <asm/io.h>
 #include <asm/bitops.h>
@@ -127,6 +129,12 @@
 
         irq_enter(cpu, irq);
 
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ if (irq != TIMER_IRQ) { /* avoid double-reporting the timer IRQ */
+ TRACE_IRQ_ENTRY(irq, !(user_mode(regs)));
+ }
+#endif
+
         status = 1; /* Force the "do bottom halves" bit */
 
         if (!(action->flags & SA_INTERRUPT))
@@ -142,6 +150,12 @@
         local_irq_disable();
 
         irq_exit(cpu, irq);
+
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ if (irq != TIMER_IRQ) { /* avoid double-reporting the timer IRQ */
+ TRACE_IRQ_EXIT();
+ }
+#endif
 
         return status;
 }
diff -urN linux-2.5.34/arch/sh/kernel/process.c linux-2.5.34-ltt/arch/sh/kernel/process.c
--- linux-2.5.34/arch/sh/kernel/process.c Mon Sep 9 13:35:14 2002
+++ linux-2.5.34-ltt/arch/sh/kernel/process.c Mon Sep 9 19:06:34 2002
@@ -16,6 +16,8 @@
 #include <linux/slab.h>
 #include <linux/a.out.h>
 
+#include <linux/trace.h>
+
 #include <asm/io.h>
 #include <asm/uaccess.h>
 #include <asm/mmu_context.h>
@@ -138,7 +140,16 @@
                 : "i" (__NR_exit), "r" (__sc3), "r" (__sc4), "r" (__sc5),
                   "r" (__sc8), "r" (__sc9)
                 : "memory", "t");
- return __sc0;
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ {
+ volatile unsigned long retval = __sc0;
+ if (retval > 0)
+ TRACE_PROCESS(TRACE_EV_PROCESS_KTHREAD, retval, (int) fn);
+ return retval;
+ }
+#else
+ return __sc0;
+#endif
 }
 
 /*
diff -urN linux-2.5.34/arch/sh/kernel/sys_sh.c linux-2.5.34-ltt/arch/sh/kernel/sys_sh.c
--- linux-2.5.34/arch/sh/kernel/sys_sh.c Mon Sep 9 13:35:12 2002
+++ linux-2.5.34-ltt/arch/sh/kernel/sys_sh.c Mon Sep 9 19:06:34 2002
@@ -21,6 +21,8 @@
 #include <linux/file.h>
 #include <linux/utsname.h>
 
+#include <linux/trace.h>
+
 #include <asm/uaccess.h>
 #include <asm/ipc.h>
 
@@ -138,6 +140,8 @@
 
         version = call >> 16; /* hack for backward compatibility */
         call &= 0xffff;
+
+ TRACE_IPC(TRACE_EV_IPC_CALL, call, first);
 
         if (call <= SEMCTL)
                 switch (call) {
diff -urN linux-2.5.34/arch/sh/kernel/traps.c linux-2.5.34-ltt/arch/sh/kernel/traps.c
--- linux-2.5.34/arch/sh/kernel/traps.c Mon Sep 9 13:35:01 2002
+++ linux-2.5.34-ltt/arch/sh/kernel/traps.c Mon Sep 9 23:52:26 2002
@@ -25,6 +25,8 @@
 #include <linux/delay.h>
 #include <linux/spinlock.h>
 
+#include <linux/trace.h>
+
 #include <asm/system.h>
 #include <asm/uaccess.h>
 #include <asm/io.h>
@@ -42,7 +44,9 @@
         sti(); \
         tsk->thread.error_code = error_code; \
         tsk->thread.trap_no = trapnr; \
+ TRACE_TRAP_ENTRY(trapnr, regs.pc); \
         force_sig(signr, tsk); \
+ TRACE_TRAP_EXIT(); \
         die_if_no_fixup(str,&regs,error_code); \
 }
 
@@ -464,6 +468,8 @@
 
         asm volatile("stc r2_bank,%0": "=r" (error_code));
 
+ TRACE_TRAP_ENTRY(error_code >> 5, regs->pc);
+
         oldfs = get_fs();
 
         if (user_mode(regs)) {
@@ -487,8 +493,10 @@
                 tmp = handle_unaligned_access(instruction, regs);
                 set_fs(oldfs);
 
- if (tmp==0)
- return; /* sorted */
+ if (tmp==0) {
+ TRACE_TRAP_EXIT();
+ return; /* sorted */
+ }
 
         uspace_segv:
                 printk(KERN_NOTICE "Killing process \"%s\" due to unaligned access\n", current->comm);
@@ -509,6 +517,7 @@
                 handle_unaligned_access(instruction, regs);
                 set_fs(oldfs);
         }
+ TRACE_TRAP_EXIT();
 }
 
 DO_ERROR(12, SIGILL, "reserved instruction", reserved_inst, current)
@@ -586,3 +595,72 @@
 {
         printk("Backtrace not yet implemented for SH.\n");
 }
+
+/* Trace related code */
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+asmlinkage void trace_real_syscall_entry(struct pt_regs *regs)
+{
+ int use_depth;
+ int use_bounds;
+ int depth = 0;
+ int seek_depth;
+ unsigned long lower_bound;
+ unsigned long upper_bound;
+ unsigned long addr;
+ unsigned long *stack;
+ trace_syscall_entry trace_syscall_event;
+
+ /* Set the syscall ID */
+ trace_syscall_event.syscall_id = (uint8_t) regs->regs[REG_REG0 + 3];
+
+ /* Set the address in any case */
+ trace_syscall_event.address = regs->pc;
+
+ /* Are we in the kernel (This is a kernel thread)? */
+ if (!user_mode(regs))
+ /* Don't go digining anywhere */
+ goto trace_syscall_end;
+
+ /* Get the trace configuration */
+ if (trace_get_config(&use_depth, &use_bounds, &seek_depth,
+ (void *) &lower_bound, (void *) &upper_bound) < 0)
+ goto trace_syscall_end;
+
+ /* Do we have to search for an eip address range */
+ if ((use_depth == 1) || (use_bounds == 1)) {
+ /* Start at the top of the stack (bottom address since stacks grow downward) */
+ stack = (unsigned long *) regs->regs[REG_REG15];
+
+ /* Keep on going until we reach the end of the process' stack limit (wherever it may be) */
+ while (!get_user(addr, stack)) {
+ /* Does this LOOK LIKE an address in the program */
+ /* TODO: does this work with shared libraries?? - Greg Banks */
+ if ((addr > current->mm->start_code) && (addr < current->mm->end_code)) {
+ /* Does this address fit the description */
+ if (((use_depth == 1) && (depth == seek_depth))
+ || ((use_bounds == 1) && (addr > lower_bound)
+ && (addr < upper_bound))) {
+ /* Set the address */
+ trace_syscall_event.address = addr;
+
+ /* We're done */
+ goto trace_syscall_end;
+ } else
+ /* We're one depth more */
+ depth++;
+ }
+ /* Go on to the next address */
+ stack++;
+ }
+ }
+trace_syscall_end:
+ /* Trace the event */
+ trace_event(TRACE_EV_SYSCALL_ENTRY, &trace_syscall_event);
+}
+
+asmlinkage void trace_real_syscall_exit(void)
+{
+ trace_event(TRACE_EV_SYSCALL_EXIT, NULL);
+}
+
+#endif /* (CONFIG_TRACE || CONFIG_TRACE_MODULE) */
diff -urN linux-2.5.34/arch/sh/mm/fault.c linux-2.5.34-ltt/arch/sh/mm/fault.c
--- linux-2.5.34/arch/sh/mm/fault.c Mon Sep 9 13:35:06 2002
+++ linux-2.5.34-ltt/arch/sh/mm/fault.c Mon Sep 9 19:06:34 2002
@@ -20,6 +20,8 @@
 #include <linux/smp_lock.h>
 #include <linux/interrupt.h>
 
+#include <linux/trace.h>
+
 #include <asm/system.h>
 #include <asm/io.h>
 #include <asm/uaccess.h>
@@ -98,6 +100,14 @@
         tsk = current;
         mm = tsk->mm;
 
+#if (CONFIG_TRACE || CONFIG_TRACE_MODULE)
+ {
+ unsigned long trapnr;
+ asm volatile("stc r2_bank,%0": "=r" (trapnr));
+ TRACE_TRAP_ENTRY(trapnr >> 5, regs->pc); /* trap 4,5 or 6 */
+ }
+#endif
+
         /*
          * If we're in an interrupt or have no user
          * context, we must not take the fault..
@@ -149,6 +159,7 @@
         }
 
         up_read(&mm->mmap_sem);
+ TRACE_TRAP_EXIT();
         return;
 
 /*
@@ -162,6 +173,7 @@
                 tsk->thread.address = address;
                 tsk->thread.error_code = writeaccess;
                 force_sig(SIGSEGV, tsk);
+ TRACE_TRAP_EXIT();
                 return;
         }
 
@@ -170,6 +182,7 @@
         fixup = search_exception_table(regs->pc);
         if (fixup != 0) {
                 regs->pc = fixup;
+ TRACE_TRAP_EXIT();
                 return;
         }
 
@@ -231,6 +244,8 @@
         /* Kernel mode? Handle exceptions or die */
         if (!user_mode(regs))
                 goto no_context;
+
+ TRACE_TRAP_EXIT();
 }
 
 /*
diff -urN linux-2.5.34/include/asm-sh/trace.h linux-2.5.34-ltt/include/asm-sh/trace.h
--- linux-2.5.34/include/asm-sh/trace.h Wed Dec 31 19:00:00 1969
+++ linux-2.5.34-ltt/include/asm-sh/trace.h Mon Sep 9 19:06:35 2002
@@ -0,0 +1,15 @@
+/*
+ * linux/include/asm-sh/trace.h
+ *
+ * Copyright (C) 2002, Karim Yaghmour
+ *
+ * SuperH definitions for tracing system
+ */
+
+#include <linux/trace.h>
+
+/* Current arch type */
+#define TRACE_ARCH_TYPE TRACE_ARCH_TYPE_SH
+
+/* Current variant type */
+#define TRACE_ARCH_VARIANT TRACE_ARCH_VARIANT_NONE
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/



This archive was generated by hypermail 2b29 : Sun Sep 15 2002 - 22:00:22 EST