[RFC Patch v1 38/55] ARC: Low level event capture/logging

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



From: Vineet Gupta <vgupta@xxxxxxxxxxxx>

Poorman's version of LTT

Signed-off-by: Vineet Gupta <vgupta@xxxxxxxxxxxx>
---
 arch/arc/Kconfig                     |    7 +
 arch/arc/Makefile                    |    3 +
 arch/arc/include/asm/event-log-asm.h |  185 +++++++++++++++++++++
 arch/arc/include/asm/event-log.h     |  102 ++++++++++++
 arch/arc/kernel/Makefile             |    1 +
 arch/arc/kernel/asm-offsets.c        |   15 ++
 arch/arc/kernel/entry.S              |   31 ++++
 arch/arc/kernel/event-log.c          |  304 ++++++++++++++++++++++++++++++++++
 arch/arc/kernel/signal.c             |    3 +
 arch/arc/mm/tlb.c                    |    3 +
 arch/arc/mm/tlbex.S                  |    7 +
 11 files changed, 661 insertions(+), 0 deletions(-)
 create mode 100644 arch/arc/include/asm/event-log-asm.h
 create mode 100644 arch/arc/include/asm/event-log.h
 create mode 100644 arch/arc/kernel/event-log.c

diff --git a/arch/arc/Kconfig b/arch/arc/Kconfig
index e096545..15d740c 100644
--- a/arch/arc/Kconfig
+++ b/arch/arc/Kconfig
@@ -323,6 +323,13 @@ config ARC_DBG_TLB_PARANOIA
 	depends on ARC_DBG
 	default n
 
+config ARC_DBG_EVENT_TIMELINE
+	bool "Low level event capture"
+	depends on ARC_DBG
+	default n
+	help
+	  Capture low level events: IRQ/Trap/Exception
+
 config ARC_DBG_TLB_MISS_COUNT
 	bool "Profile TLB Misses"
 	default n
diff --git a/arch/arc/Makefile b/arch/arc/Makefile
index 7ef6767..a533546 100644
--- a/arch/arc/Makefile
+++ b/arch/arc/Makefile
@@ -47,6 +47,9 @@ endif
 disable_small_data := y
 cflags-$(disable_small_data)		+= -mno-sdata -fcall-used-gp
 
+# Low level event tracing with Metaware debugger assist needs symbol info
+cflags-$(CONFIG_ARC_DBG_EVENT_TIMELINE) += -g
+
 cflags-$(CONFIG_CPU_BIG_ENDIAN)		+= -mbig-endian
 ldflags-$(CONFIG_CPU_BIG_ENDIAN)	+= -EB
 
diff --git a/arch/arc/include/asm/event-log-asm.h b/arch/arc/include/asm/event-log-asm.h
new file mode 100644
index 0000000..bc29e7d
--- /dev/null
+++ b/arch/arc/include/asm/event-log-asm.h
@@ -0,0 +1,185 @@
+/*
+ *  Low level Event Capture API callable from Assembly Code
+ *  vineetg: Feb 2008
+ *
+ *  TBD: SMP Safe
+ *
+ * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com)
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ */
+
+#ifndef __ASM_ARC_EVENT_LOG_ASM_H
+#define __ASM_ARC_EVENT_LOG_ASM_H
+
+#include <asm/event-log.h>
+
+#ifdef __ASSEMBLY__
+
+#ifndef CONFIG_ARC_DBG_EVENT_TIMELINE
+
+.macro TAKE_SNAP_ASM reg_scratch, reg_ptr, type
+.endm
+
+.macro TAKE_SNAP_C_FROM_ASM type
+.endm
+
+#else
+
+#include <asm/asm-offsets.h>
+
+/*
+ * Macro to invoke the ASM event logger routine from assmebly code
+ * This is generated in-place in caller.
+ *
+ * @reg_scratch and @reg_ptr:
+ *	Registers provided by caller for coding the macro itself.
+ *	At this point if call, say Low level ISR, the Reg-File might not have
+ *	been saved, so only use reg safe.
+ * @type:
+ *	The low level event, defined in event-log.h
+ */
+.macro TAKE_SNAP_ASM reg_scratch, reg_ptr, type
+
+	/*
+	 * Earlier we used to save only reg_scratch and clobber reg_ptr and rely
+	 * on caller to understand this. Too much trouble.
+	 * Now we save both
+	 */
+	st \reg_scratch, [tmp_save_reg]
+	st \reg_ptr, [tmp_save_reg2]
+
+	ld \reg_ptr, [timeline_ctr]
+
+	/* HACK to detect if the circular log buffer is being overflowed */
+	brne \reg_ptr, MAX_SNAPS, 1f
+	flag 1
+	nop
+1:
+#ifdef CONFIG_ARC_HAS_HW_MPY
+	mpyu \reg_ptr, \reg_ptr, EVLOG_RECORD_SZ
+#else
+#error "even logger broken for !CONFIG_ARC_HAS_HW_MPY
+#endif
+
+	add \reg_ptr, timeline_log, \reg_ptr
+
+	/*############ Common data ########## */
+
+	/* TIMER1 count in timeline_log[timeline_ctr].time */
+	lr \reg_scratch, [ARC_REG_TIMER1_CNT]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_TIME]
+
+	/* current task ptr in timeline_log[timeline_ctr].task */
+	ld \reg_scratch, [_current_task]
+	ld \reg_scratch, [\reg_scratch, TASK_TGID]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_TASK]
+
+	/* Type of event (Intr/Excp/Trap etc) */
+	mov \reg_scratch, \type
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EVENT_ID]
+
+	/* save SP at time of exception */
+	st sp, [\reg_ptr, EVLOG_FIELD_SP]
+
+	st 0, [\reg_ptr, EVLOG_FIELD_EXTRA]
+	st 0, [\reg_ptr, EVLOG_FIELD_CAUSE]
+	st 0, [\reg_ptr, EVLOG_FIELD_EXTRA3]
+
+	lr \reg_scratch, [status32]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EXTRA2]
+
+	/* ############ Event specific data ########## */
+	mov \reg_scratch, \type
+	and.f 0, \reg_scratch, EVENT_CLASS_EXIT
+	bz 1f
+
+	/* Stuff to do for all kernel exit events */
+	ld \reg_scratch, [sp, PT_status32]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EXTRA]
+
+	/* preempt count in log->sp */
+	and \reg_scratch, sp, ~(0x2000 - 1)
+	ld \reg_scratch, [\reg_scratch, THREAD_INFO_PREEMPT_COUNT]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_SP]
+
+	ld \reg_scratch, [sp, PT_ret]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EFA]
+
+	mov \reg_scratch, \type
+1:
+2:
+	/* for Trap, Syscall number  */
+	cmp \reg_scratch, SNAP_TRAP_IN
+	bnz 3f
+	st r8, [\reg_ptr, EVLOG_FIELD_CAUSE]
+	lr \reg_scratch, [erstatus]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EXTRA]
+	j 99f
+3:
+5:
+	/* For Exceptions (TLB/ProtV etc) */
+	cmp \reg_scratch, SNAP_EXCP_IN
+	bnz 6f
+
+	lr \reg_scratch, [ecr]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_CAUSE]
+	lr \reg_scratch, [eret]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EFA]
+	lr \reg_scratch, [erstatus]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EXTRA]
+	lr \reg_scratch, [efa]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EXTRA3]
+	j 99f
+
+6:	/* for Interrupts, IRQ */
+	cmp \reg_scratch, SNAP_INTR_OUT
+	bnz 7f
+	lr \reg_scratch, [icause1]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_CAUSE]
+	j 99f
+
+7:
+	cmp \reg_scratch, SNAP_INTR_OUT2
+	bnz 8f
+	lr \reg_scratch, [icause2]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_CAUSE]
+	j 99f
+
+8:
+	cmp \reg_scratch, SNAP_EXCP_OUT_FAST
+	bnz 9f
+	lr \reg_scratch, [erstatus]
+	st \reg_scratch, [\reg_ptr, EVLOG_FIELD_EXTRA]
+	j 99f
+
+	/* place holder for next */
+9:
+
+99:
+	/* increment timeline_ctr  with mode on max */
+	ld \reg_scratch, [timeline_ctr]
+	add \reg_scratch, \reg_scratch, 1
+	and \reg_scratch, \reg_scratch, MAX_SNAPS_MASK
+	st \reg_scratch, [timeline_ctr]
+
+	/* Restore back orig scratch reg */
+	ld \reg_scratch, [tmp_save_reg]
+	ld \reg_ptr, [tmp_save_reg2]
+.endm
+
+/*
+ * Macro to invoke the "C" event logger routine from assmebly code
+ */
+.macro TAKE_SNAP_C_FROM_ASM type
+	mov r0, \type
+	bl take_snap2
+.endm
+
+#endif	/* CONFIG_ARC_DBG_EVENT_TIMELINE */
+
+#endif	/* __ASSEMBLY__ */
+
+#endif
diff --git a/arch/arc/include/asm/event-log.h b/arch/arc/include/asm/event-log.h
new file mode 100644
index 0000000..423f549
--- /dev/null
+++ b/arch/arc/include/asm/event-log.h
@@ -0,0 +1,102 @@
+/*
+ * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com)
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ *  vineetg: Dec 2009
+ *      Reworked the numbering scheme into Event Classes for making it easier to
+ *      do class specific things in the snapshot routines
+ *
+ *  vineetg: Feb 2008
+ *      System Event Logging APIs
+ */
+
+#ifndef __ASM_ARC_EVENT_LOG_H
+#define __ASM_ARC_EVENT_LOG_H
+
+/*######################################################################
+ *
+ *    Event Logging API
+ *
+ *#####################################################################*/
+
+/* Size of the log buffer */
+#define MAX_SNAPS	1024
+#define MAX_SNAPS_MASK	(MAX_SNAPS-1)
+
+/* Helpers to setup Event IDs:
+ * 8 classes of events possible
+ * 23 unique events for each Class
+ * Right now we have only 3 classes:
+ * Entry into kernel, exit from kernel and everything else is custom event
+ *
+ * Need for this fancy numbering scheme so that in event logger, class specific
+ * things, common for all events in class, could be easily done
+ */
+#define EVENT_ID(x)             (0x100 << x)
+#define EVENT_CLASS_ENTER       0x01	/* Need to start from 1, not 0 */
+#define EVENT_CLASS_EXIT        0x02
+#define EVENT_CLASS_CUSTOM      0x80
+
+#define KERNEL_ENTER_EVENT(x)   (EVENT_ID(x)|EVENT_CLASS_ENTER)
+#define KERNEL_EXIT_EVENT(x)    (EVENT_ID(x)|EVENT_CLASS_EXIT)
+#define CUSTOM_EVENT(x)         (EVENT_ID(x)|EVENT_CLASS_CUSTOM)
+
+/* Actual Event IDs used in kernel code */
+#define SNAP_INTR_IN            KERNEL_ENTER_EVENT(0)
+#define SNAP_EXCP_IN            KERNEL_ENTER_EVENT(1)
+#define SNAP_TRAP_IN            KERNEL_ENTER_EVENT(2)
+#define SNAP_INTR_IN2           KERNEL_ENTER_EVENT(3)
+
+#define SNAP_INTR_OUT           KERNEL_EXIT_EVENT(0)
+#define SNAP_EXCP_OUT           KERNEL_EXIT_EVENT(1)
+#define SNAP_TRAP_OUT           KERNEL_EXIT_EVENT(2)
+#define SNAP_INTR_OUT2          KERNEL_EXIT_EVENT(3)
+#define SNAP_EXCP_OUT_FAST      KERNEL_EXIT_EVENT(4)
+
+#define SNAP_PRE_CTXSW_2_U      CUSTOM_EVENT(0)
+#define SNAP_PRE_CTXSW_2_K      CUSTOM_EVENT(1)
+#define SNAP_DO_PF_ENTER        CUSTOM_EVENT(2)
+#define SNAP_DO_PF_EXIT         CUSTOM_EVENT(3)
+#define SNAP_TLB_FLUSH_ALL      CUSTOM_EVENT(4)
+#define SNAP_PREEMPT_SCH_IRQ    CUSTOM_EVENT(5)
+#define SNAP_PREEMPT_SCH        CUSTOM_EVENT(6)
+#define SNAP_SIGRETURN          CUSTOM_EVENT(7)
+#define SNAP_BEFORE_SIG         CUSTOM_EVENT(8)
+
+#define SNAP_SENTINEL           CUSTOM_EVENT(22)
+
+#ifndef CONFIG_ARC_DBG_EVENT_TIMELINE
+
+#define take_snap(type, extra, ptreg)
+#define sort_snaps(halt_after_sort)
+
+#else
+
+#ifndef __ASSEMBLY__
+
+typedef struct {
+
+	/*  0 */ char nm[16];
+	/* 16 */ unsigned int extra; /* Traps: Syscall num, Intr: IRQ, Excep */
+	/* 20 */ unsigned int fault_addr;
+	/* 24 */ unsigned int cause;
+	/* 28 */ unsigned int task;
+	/* 32 */ unsigned long time;
+	/* 36 */ unsigned int event;
+	/* 40 */ unsigned int sp;
+	/* 44 */ unsigned int extra2;
+	/* 40 */ unsigned int extra3;
+
+} timeline_log_t;
+
+void take_snap(int type, unsigned int extra, unsigned int extra2);
+void sort_snaps(int halt_after_sort);
+
+#endif /* __ASSEMBLY__ */
+
+#endif /* CONFIG_ARC_DBG_EVENT_TIMELINE */
+
+#endif /* __ASM_ARC_EVENT_PROFILE_H */
diff --git a/arch/arc/kernel/Makefile b/arch/arc/kernel/Makefile
index 9151bbe..ec1f130 100644
--- a/arch/arc/kernel/Makefile
+++ b/arch/arc/kernel/Makefile
@@ -12,6 +12,7 @@ obj-y	:= arcksyms.o setup.o irq.o time.o reset.o ptrace.o entry.o process.o \
 	   signal.o traps.o sys.o troubleshoot.o stacktrace.o
 
 obj-$(CONFIG_MODULES)			+= arcksyms.o module.o
+obj-$(CONFIG_ARC_DBG_EVENT_TIMELINE)	+= event-log.o
 obj-$(CONFIG_ARC_FPU_SAVE_RESTORE)	+= fpu.o
 CFLAGS_fpu.o   += -mdpfp
 
diff --git a/arch/arc/kernel/asm-offsets.c b/arch/arc/kernel/asm-offsets.c
index b0e7254..0c06b7a 100644
--- a/arch/arc/kernel/asm-offsets.c
+++ b/arch/arc/kernel/asm-offsets.c
@@ -13,6 +13,7 @@
 #include <linux/thread_info.h>
 #include <asm/page.h>
 #include <linux/kbuild.h>
+#include <asm/event-log.h>
 
 int main(void)
 {
@@ -45,5 +46,19 @@ int main(void)
 
 	DEFINE(MM_CTXT_ASID, offsetof(mm_context_t, asid));
 
+#ifdef CONFIG_ARC_DBG_EVENT_TIMELINE
+	BLANK();
+	DEFINE(EVLOG_FIELD_EXTRA, offsetof(timeline_log_t, extra));
+	DEFINE(EVLOG_FIELD_EFA, offsetof(timeline_log_t, fault_addr));
+	DEFINE(EVLOG_FIELD_CAUSE, offsetof(timeline_log_t, cause));
+	DEFINE(EVLOG_FIELD_TASK, offsetof(timeline_log_t, task));
+	DEFINE(EVLOG_FIELD_TIME, offsetof(timeline_log_t, time));
+	DEFINE(EVLOG_FIELD_EVENT_ID, offsetof(timeline_log_t, event));
+	DEFINE(EVLOG_FIELD_SP, offsetof(timeline_log_t, sp));
+	DEFINE(EVLOG_RECORD_SZ, sizeof(timeline_log_t));
+	DEFINE(EVLOG_FIELD_EXTRA2, offsetof(timeline_log_t, extra2));
+	DEFINE(EVLOG_FIELD_EXTRA3, offsetof(timeline_log_t, extra3));
+#endif
+
 	return 0;
 }
diff --git a/arch/arc/kernel/entry.S b/arch/arc/kernel/entry.S
index 13a0052..5babb8a 100644
--- a/arch/arc/kernel/entry.S
+++ b/arch/arc/kernel/entry.S
@@ -148,6 +148,7 @@ VECTOR   reserved                ; Reserved Exceptions
 #include <asm/errno.h>
 #include <asm/arcregs.h>
 #include <asm/irqflags.h>
+#include <asm/event-log-asm.h>
 
 ;##################### Scratch Mem for IRQ stack switching #############
 
@@ -213,6 +214,8 @@ ARC_ENTRY handle_interrupt_level2
 	st      r9, [r10, THREAD_INFO_PREEMPT_COUNT]
 
 1:
+	TAKE_SNAP_C_FROM_ASM  SNAP_INTR_IN2
+
 	;------------------------------------------------------
 	; setup params for Linux common ISR and invoke it
 	;------------------------------------------------------
@@ -245,6 +248,9 @@ ARC_ENTRY handle_interrupt_level1
 	SWITCH_TO_KERNEL_STK
 	SAVE_ALL_INT1
 
+	; snapshot routine takes care of disabling nested intr
+	TAKE_SNAP_C_FROM_ASM  SNAP_INTR_IN
+
 	lr  r0, [icause1]
 	and r0, r0, 0x1f
 
@@ -267,6 +273,8 @@ ARC_ENTRY instr_service
 
 	EXCPN_PROLOG_FREEUP_REG r9
 
+	TAKE_SNAP_ASM r8, r9, SNAP_EXCP_IN
+
 	lr  r9, [erstatus]
 
 	SWITCH_TO_KERNEL_STK
@@ -291,6 +299,8 @@ ARC_ENTRY mem_service
 
 	EXCPN_PROLOG_FREEUP_REG r9
 
+	TAKE_SNAP_ASM r8, r9, SNAP_EXCP_IN
+
 	lr  r9, [erstatus]
 
 	SWITCH_TO_KERNEL_STK
@@ -342,6 +352,8 @@ ARC_ENTRY EV_TLBProtV
 
 	EXCPN_PROLOG_FREEUP_REG r9
 
+	TAKE_SNAP_ASM r8, r9, SNAP_EXCP_IN
+
 	;Which mode (user/kernel) was the system in when Exception occured
 	lr  r9, [erstatus]
 
@@ -400,6 +412,8 @@ ARC_ENTRY EV_PrivilegeV
 
 	EXCPN_PROLOG_FREEUP_REG r9
 
+	TAKE_SNAP_ASM r8, r9, SNAP_EXCP_IN
+
 	lr  r9, [erstatus]
 
 	SWITCH_TO_KERNEL_STK
@@ -526,6 +540,8 @@ ARC_ENTRY EV_Trap
 	; Need at least 1 reg to code the early exception prolog
 	EXCPN_PROLOG_FREEUP_REG r9
 
+	TAKE_SNAP_ASM r10, r9,  SNAP_TRAP_IN
+
 	;Which mode (user/kernel) was the system in when intr occured
 	lr  r9, [erstatus]
 
@@ -733,6 +749,7 @@ not_exception:
 	st      r9, [r10, THREAD_INFO_PREEMPT_COUNT]
 
 149:
+	TAKE_SNAP_C_FROM_ASM SNAP_INTR_OUT2
 	;return from level 2
 	RESTORE_ALL_INT2
 debug_marker_l2:
@@ -745,6 +762,7 @@ not_level2_interrupt:
 	bbit0  r10, STATUS_A1_BIT, not_level1_interrupt
 
 	;return from level 1
+	TAKE_SNAP_C_FROM_ASM SNAP_INTR_OUT
 
 	RESTORE_ALL_INT1
 debug_marker_l1:
@@ -754,6 +772,19 @@ not_level1_interrupt:
 
 	;this case is for syscalls or Exceptions (with fake rtie)
 
+#ifdef CONFIG_ARC_DBG_EVENT_TIMELINE
+	ld r8, [sp, PT_orig_r8]
+	cmp r8, NR_syscalls+1
+	jeq  149f
+	TAKE_SNAP_ASM r9, r10, SNAP_TRAP_OUT
+	j 150f
+
+149:
+	TAKE_SNAP_ASM r9, r10, SNAP_EXCP_OUT
+150:
+
+#endif
+
 	RESTORE_ALL_SYS
 debug_marker_syscall:
 	rtie
diff --git a/arch/arc/kernel/event-log.c b/arch/arc/kernel/event-log.c
new file mode 100644
index 0000000..f8ab807
--- /dev/null
+++ b/arch/arc/kernel/event-log.c
@@ -0,0 +1,304 @@
+/*
+ * event-log.c : Poorman's version of LTT for low level event capturing
+ *
+ * captures IRQ/Exceptions/Sys-Calls/arbitrary function call
+ * XXX: Not SMP Safe
+ *
+ * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com)
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ *
+ *  vineetg Jan 2009
+ *      -Converted strcpy to strncpy
+ *
+ *  vineetg: Feb 2008: Event capturing Framework
+ *      -Captures the event-id and related info in circular log buffer
+ *
+ *      -USAGE:
+ *          Events are defined in API file, include/asm-arc/event-log.h
+ *          To log the event, "C" code calls API
+ *              take_snap(event-id, event-specific-info)
+ *          To log the event, ASM caller calls a "asm" macro
+ *              TAKE_SNAP_ASM reg-x, reg-y, event-id
+ *          To stop the capture and sort the log buffer,
+ *              sort_snaps(halt-after-sort)
+ *
+ *      -The reason for 2 APIs is that in low level handlers
+ *          which we are interested in capturing, often don't have
+ *          stack switched, thus a "C" API wont work. Also there
+ *          is a very strict requirement of which registers are usable
+ *          hence the 2 regs
+ *
+ *      -Done primarily to chase the Random Segmentation Faults
+ *          when we switched from gcc 3.4 to 4.2
+ */
+
+#include <linux/sort.h>
+#include <linux/sched.h>
+#include <linux/interrupt.h>
+#include <asm/current.h>
+#include <asm/event-log.h>
+#include <linux/module.h>
+#include <linux/reboot.h>
+
+#ifdef CONFIG_ARC_CURR_IN_REG
+/*
+ * current on ARC is a register variable "r25" setup on entry to kernel and
+ * restored back to user value on return.
+ * However if the event snap shotting return is called very late from
+ * ISR/Exception return code, r25 might already have been restored to user
+ * value, hence would no longer point to current task. This can cause weird
+ * de-referencing crashes. Safest option is to undef it and instead define
+ * it in terms of current_thread_info() which is derived from SP
+ */
+#undef current
+#define current (current_thread_info()->task)
+#endif
+
+/*
+ * Log buffer which stores the event info
+ *
+ *  There is race condition when the counter goes 1 more than
+ *  max-value (if IRQ sneaks in in the logging routine. Since
+ *  we don't want to do fancy intr-enable-disable etc,
+ *  we keep 1 extra element in log buffer
+ */
+timeline_log_t timeline_log[MAX_SNAPS + 1];
+
+/* counter in log bugger for next entry */
+int timeline_ctr;
+
+/* Used in the low level asm handler to free up a reg */
+int tmp_save_reg, tmp_save_reg2;
+int l2_ctr;
+
+/* Event capture API */
+void take_snap(int event, unsigned int arg1, unsigned int arg2)
+{
+	timeline_log_t *entry = &timeline_log[timeline_ctr];
+
+	entry->time = read_aux_reg(ARC_REG_TIMER1_CNT);
+	entry->task = current->pid;
+	entry->event = event;
+	entry->extra2 = read_aux_reg(0xa);	/* status32 */
+
+	entry->cause = read_aux_reg(0x403);	/* ecr */
+	entry->fault_addr = read_aux_reg(0x404);	/* efa */
+
+	entry->extra = arg1;
+	entry->sp = arg2;
+
+	entry->extra3 =
+	    (unsigned int)__builtin_return_address(0);
+
+	if (timeline_ctr == (MAX_SNAPS - 1))
+		timeline_ctr = 0;
+	else
+		timeline_ctr++;
+
+}
+EXPORT_SYMBOL(take_snap);
+
+void take_snap2(int event)
+{
+	unsigned long x, flags = 0, stat32;
+	timeline_log_t *entry = &timeline_log[timeline_ctr];
+
+	stat32 = read_aux_reg(0xa);	/* status32 */
+
+	/* In case this is for Level 1 ISR, disable further Interrupts
+	 * so that timeline_ctr is not clobbered
+	 */
+	if (event == SNAP_INTR_IN)
+		local_irq_save(flags);
+
+	entry->time = read_aux_reg(ARC_REG_TIMER1_CNT);
+	entry->task = current->pid;
+	entry->event = event;
+	entry->extra2 = stat32;
+
+	entry->sp = current_thread_info()->preempt_count;
+
+	if (event == SNAP_INTR_IN2) {
+		entry->cause = read_aux_reg(0x40B);	/* icause2 */
+		entry->extra = read_aux_reg(0x0C);	/* statsu32_l2 */
+		__asm__ __volatile__("mov %0, ilink2   \r\n" : "=r"(x));
+		entry->fault_addr = x;
+	} else if (event == SNAP_INTR_IN) {
+		entry->cause = read_aux_reg(0x40A);	/* icause1 */
+		entry->extra = read_aux_reg(0x0B);	/* statsu32_l1 */
+		__asm__ __volatile__("mov %0, ilink1   \r\n" : "=r"(x));
+		entry->fault_addr = x;
+	}
+
+	if (timeline_ctr == (MAX_SNAPS - 1))
+		timeline_ctr = 0;
+	else
+		timeline_ctr++;
+
+	if (current_thread_info()->preempt_count == 0xFFFFFFFF)
+		sort_snaps(1);
+
+	if (event == SNAP_INTR_IN)
+		local_irq_restore(flags);
+}
+EXPORT_SYMBOL(take_snap2);
+
+/* CMP routine called by event sort
+ * When comparing the click time entries of @a to @b:
+ *  gt: returns 1
+ *  lt:  -1
+ *  eq: returns 0
+ */
+static int snap_cmp(const void *a, const void *b)
+{
+	timeline_log_t *click_a, *click_b;
+
+	click_a = (timeline_log_t *) a;
+	click_b = (timeline_log_t *) b;
+
+	if (click_a->time == click_b->time)
+		return 0;
+	else if (click_a->time < click_b->time)
+		return -1;
+
+	return 1;
+}
+
+/* Event Sort API, so that counter Rollover is not visibel to user */
+void sort_snaps(int halt_after_sort)
+{
+	int i;
+	unsigned int flags, tmp;
+
+	/* TODO SMP */
+	local_irq_save(flags);
+
+	take_snap(SNAP_SENTINEL, 0, 0);
+
+	sort(timeline_log, MAX_SNAPS, sizeof(timeline_log_t), snap_cmp, NULL);
+
+	for (i = 0; i < MAX_SNAPS; i++) {
+		memset(timeline_log[i].nm, 0, 16);
+
+		switch (timeline_log[i].event) {
+		case SNAP_TLB_FLUSH_ALL:
+			strcpy(timeline_log[i].nm, "TLB FLUSH ALL");
+			break;
+		case 85:
+			strcpy(timeline_log[i].nm, "FORK");
+			break;
+		case 86:
+			strcpy(timeline_log[i].nm, "EXEC");
+			break;
+		case 99:
+			strcpy(timeline_log[i].nm, "Slow-TLB-Write");
+			break;
+		case SNAP_EXCP_IN:
+			switch (timeline_log[i].cause >> 16) {
+			case 0x21:
+				strcpy(timeline_log[i].nm, "I-TLB");
+				break;
+			case 0x22:
+				strcpy(timeline_log[i].nm, "D-TLB");
+				break;
+			case 0x23:
+				strcpy(timeline_log[i].nm, "PROT-V-TLB");
+				break;
+			default:
+				strcpy(timeline_log[i].nm, "?#?");
+				break;
+			}
+			break;
+		case SNAP_EXCP_OUT_FAST:
+			strcpy(timeline_log[i].nm, "TLB Refill");
+			break;
+		case SNAP_EXCP_OUT:
+			strcpy(timeline_log[i].nm, "Excp-RET");
+			break;
+		case SNAP_TRAP_IN:
+			strcpy(timeline_log[i].nm, "SyCall :");
+			switch (timeline_log[i].cause) {
+			case 1:
+				strcat(timeline_log[i].nm, "Exit");
+				break;
+			case 2:
+				strcat(timeline_log[i].nm, "fork");
+				break;
+			case 114:
+				strcat(timeline_log[i].nm, "wait4");
+				break;
+			default:
+				strcat(timeline_log[i].nm, "???");
+			}
+			break;
+		case SNAP_TRAP_OUT:
+			strcpy(timeline_log[i].nm, "SyCall-RET");
+			break;
+		case SNAP_PRE_CTXSW_2_U:
+			strcpy(timeline_log[i].nm, "2-U-Ctx-sw");
+			break;
+		case SNAP_SENTINEL:
+			memset(&timeline_log[i], 0, sizeof(timeline_log[i]));
+			strcpy(timeline_log[i].nm, "----------");
+			break;
+		case SNAP_PRE_CTXSW_2_K:
+			strcpy(timeline_log[i].nm, "2-K-Ctx-sw");
+			break;
+		case SNAP_INTR_OUT:
+			strcpy(timeline_log[i].nm, "IRQ-OUT");
+			break;
+		case SNAP_INTR_OUT2:
+			strcpy(timeline_log[i].nm, "IRQ(2)-OUT");
+			break;
+		case SNAP_INTR_IN:
+			strcpy(timeline_log[i].nm, "IRQ-in");
+			break;
+		case SNAP_INTR_IN2:
+			strcpy(timeline_log[i].nm, "IRQ(2)-in");
+			break;
+		case SNAP_DO_PF_EXIT:
+			strcpy(timeline_log[i].nm, "PF-RET");
+			break;
+		case SNAP_PREEMPT_SCH_IRQ:
+			strcpy(timeline_log[i].nm, "Prem-Sch IRQ");
+			break;
+		case SNAP_PREEMPT_SCH:
+			strcpy(timeline_log[i].nm, "Prem-Sch");
+			break;
+		case SNAP_DO_PF_ENTER:
+			tmp = timeline_log[i].cause >> 16;
+			switch (tmp) {
+			case 0x21:
+				strcpy(timeline_log[i].nm, "PF-in:I-TLB");
+				break;
+			case 0x22:
+				strcpy(timeline_log[i].nm, "PF-in:D-TLB");
+				break;
+			case 0x23:
+				strcpy(timeline_log[i].nm, "PF-in:PROTV");
+				break;
+			default:
+				strcpy(timeline_log[i].nm, "PF-in:???");
+				break;
+			}
+			break;
+		case SNAP_SIGRETURN:
+			strcpy(timeline_log[i].nm, "sigreturn");
+			break;
+		case SNAP_BEFORE_SIG:
+			strcpy(timeline_log[i].nm, "before sig");
+			break;
+		}
+
+	}
+
+	if (halt_after_sort)
+		__asm__("flag 1");
+	else
+		local_irq_restore(flags);
+
+}
+EXPORT_SYMBOL(sort_snaps);
diff --git a/arch/arc/kernel/signal.c b/arch/arc/kernel/signal.c
index bc834da..2ec908b 100644
--- a/arch/arc/kernel/signal.c
+++ b/arch/arc/kernel/signal.c
@@ -54,6 +54,7 @@
 #include <linux/syscalls.h>
 #include <linux/tracehook.h>
 #include <asm/ucontext.h>
+#include <asm/event-log.h>
 
 #define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))
 
@@ -137,6 +138,8 @@ SYSCALL_DEFINE0(rt_sigreturn)
 		if (do_sigaltstack(&sf->uc.uc_stack, NULL, regs->sp) == -EFAULT)
 			goto badframe;
 
+	take_snap(SNAP_SIGRETURN, 0, 0);
+
 	return regs->r0;
 
 badframe:
diff --git a/arch/arc/mm/tlb.c b/arch/arc/mm/tlb.c
index c10111d..0e5cb9f 100644
--- a/arch/arc/mm/tlb.c
+++ b/arch/arc/mm/tlb.c
@@ -56,6 +56,7 @@
 #include <asm/setup.h>
 #include <asm/mmu_context.h>
 #include <asm/tlb.h>
+#include <asm/event-log.h>
 
 /*			Need for ARC MMU v2
  *
@@ -191,6 +192,8 @@ noinline void local_flush_tlb_all(void)
 	unsigned int entry;
 	struct cpuinfo_arc_mmu *mmu = &cpuinfo_arc700[smp_processor_id()].mmu;
 
+	take_snap(SNAP_TLB_FLUSH_ALL, 0, 0);
+
 	local_irq_save(flags);
 
 	/* Load PD0 and PD1 with template for a Blank Entry */
diff --git a/arch/arc/mm/tlbex.S b/arch/arc/mm/tlbex.S
index fc5b971..86804d3 100644
--- a/arch/arc/mm/tlbex.S
+++ b/arch/arc/mm/tlbex.S
@@ -41,6 +41,7 @@
 #include <asm/entry.h>
 #include <asm/tlb.h>
 #include <asm/pgtable.h>
+#include <asm/event-log-asm.h>
 #include <asm/arcregs.h>
 #include <asm/cache.h>
 #include <asm/processor.h>
@@ -194,6 +195,9 @@ ex_saved_reg1:
 	st_s  r2, [r0, 8]
 	st_s  r3, [r0, 12]
 
+	; take a snapshot of upon entering FAST Path TLB Hdlr
+	TAKE_SNAP_ASM r0, r1, SNAP_EXCP_IN
+
 	; VERIFY if the ASID in MMU-PID Reg is same as
 	; one in Linux data structures
 
@@ -317,6 +321,9 @@ do_slow_path_pf:
 	; That requires freeing up r9
 	EXCPN_PROLOG_FREEUP_REG r9
 
+	; take a snapshot of upon entering SLOW Path TLB Hdlr
+	TAKE_SNAP_ASM r8, r9, SNAP_DO_PF_ENTER
+
 	lr  r9, [erstatus]
 
 	SWITCH_TO_KERNEL_STK
-- 
1.7.4.1

--
To unsubscribe from this list: send the line "unsubscribe linux-arch" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Linux Kernel]     [Kernel Newbies]     [x86 Platform Driver]     [Netdev]     [Linux Wireless]     [Netfilter]     [Bugtraq]     [Linux Filesystems]     [Yosemite Discussion]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Device Mapper]

  Powered by Linux