Linux Kernel  3.7.1
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
op_model_ev6.c
Go to the documentation of this file.
1 
10 #include <linux/oprofile.h>
11 #include <linux/init.h>
12 #include <linux/smp.h>
13 #include <asm/ptrace.h>
14 
15 #include "op_impl.h"
16 
17 
18 /* Compute all of the registers in preparation for enabling profiling. */
19 
20 static void
21 ev6_reg_setup(struct op_register_config *reg,
22  struct op_counter_config *ctr,
23  struct op_system_config *sys)
24 {
25  unsigned long ctl, reset, need_reset, i;
26 
27  /* Select desired events. We've mapped the event numbers
28  such that they fit directly into the event selection fields. */
29  ctl = 0;
30  if (ctr[0].enabled && ctr[0].event)
31  ctl |= (ctr[0].event & 1) << 4;
32  if (ctr[1].enabled)
33  ctl |= (ctr[1].event - 2) & 15;
34  reg->mux_select = ctl;
35 
36  /* Select logging options. */
37  /* ??? Need to come up with some mechanism to trace only
38  selected processes. EV6 does not have a mechanism to
39  select kernel or user mode only. For now, enable always. */
40  reg->proc_mode = 0;
41 
42  /* EV6 cannot change the width of the counters as with the
43  other implementations. But fortunately, we can write to
44  the counters and set the value such that it will overflow
45  at the right time. */
46  reset = need_reset = 0;
47  for (i = 0; i < 2; ++i) {
48  unsigned long count = ctr[i].count;
49  if (!ctr[i].enabled)
50  continue;
51 
52  if (count > 0x100000)
53  count = 0x100000;
54  ctr[i].count = count;
55  reset |= (0x100000 - count) << (i ? 6 : 28);
56  if (count != 0x100000)
57  need_reset |= 1 << i;
58  }
59  reg->reset_values = reset;
60  reg->need_reset = need_reset;
61 }
62 
63 /* Program all of the registers in preparation for enabling profiling. */
64 
65 static void
66 ev6_cpu_setup (void *x)
67 {
68  struct op_register_config *reg = x;
69 
70  wrperfmon(2, reg->mux_select);
71  wrperfmon(3, reg->proc_mode);
72  wrperfmon(6, reg->reset_values | 3);
73 }
74 
75 /* CTR is a counter for which the user has requested an interrupt count
76  in between one of the widths selectable in hardware. Reset the count
77  for CTR to the value stored in REG->RESET_VALUES. */
78 
79 static void
80 ev6_reset_ctr(struct op_register_config *reg, unsigned long ctr)
81 {
82  wrperfmon(6, reg->reset_values | (1 << ctr));
83 }
84 
85 static void
86 ev6_handle_interrupt(unsigned long which, struct pt_regs *regs,
87  struct op_counter_config *ctr)
88 {
89  /* Record the sample. */
90  oprofile_add_sample(regs, which);
91 }
92 
93 
95  .reg_setup = ev6_reg_setup,
96  .cpu_setup = ev6_cpu_setup,
97  .reset_ctr = ev6_reset_ctr,
98  .handle_interrupt = ev6_handle_interrupt,
99  .cpu_type = "alpha/ev6",
100  .num_counters = 2,
101  .can_set_proc_mode = 0,
102 };