1 /** @file
2   Template for Timer Architecture Protocol driver of the ARM flavor
3 
4   Copyright (c) 2008 - 2009, Apple Inc. All rights reserved.<BR>
5   Copyright (c) 2011 - 2012, ARM Ltd. All rights reserved.<BR>
6 
7   This program and the accompanying materials
8   are licensed and made available under the terms and conditions of the BSD License
9   which accompanies this distribution.  The full text of the license may be found at
10   http://opensource.org/licenses/bsd-license.php
11 
12   THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
13   WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
14 
15 **/
16 
17 
18 #include <PiDxe.h>
19 
20 #include <Library/BaseLib.h>
21 #include <Library/DebugLib.h>
22 #include <Library/BaseMemoryLib.h>
23 #include <Library/UefiBootServicesTableLib.h>
24 #include <Library/UefiLib.h>
25 #include <Library/PcdLib.h>
26 #include <Library/IoLib.h>
27 
28 #include <Protocol/Timer.h>
29 #include <Protocol/HardwareInterrupt.h>
30 
31 #include <Drivers/SP804Timer.h>
32 
33 #define SP804_TIMER_PERIODIC_BASE     ((UINTN)PcdGet32 (PcdSP804TimerPeriodicBase))
34 #define SP804_TIMER_METRONOME_BASE    ((UINTN)PcdGet32 (PcdSP804TimerMetronomeBase))
35 #define SP804_TIMER_PERFORMANCE_BASE  ((UINTN)PcdGet32 (PcdSP804TimerPerformanceBase))
36 
37 // The notification function to call on every timer interrupt.
38 EFI_TIMER_NOTIFY      mTimerNotifyFunction     = (EFI_TIMER_NOTIFY)NULL;
39 EFI_EVENT             EfiExitBootServicesEvent = (EFI_EVENT)NULL;
40 
41 // The current period of the timer interrupt
42 UINT64 mTimerPeriod = 0;
43 
44 // Cached copy of the Hardware Interrupt protocol instance
45 EFI_HARDWARE_INTERRUPT_PROTOCOL *gInterrupt = NULL;
46 
47 // Cached interrupt vector
48 UINTN  gVector;
49 
50 
51 /**
52 
53   C Interrupt Handler called in the interrupt context when Source interrupt is active.
54 
55 
56   @param Source         Source of the interrupt. Hardware routing off a specific platform defines
57                         what source means.
58 
59   @param SystemContext  Pointer to system register context. Mostly used by debuggers and will
60                         update the system context after the return from the interrupt if
61                         modified. Don't change these values unless you know what you are doing
62 
63 **/
64 VOID
65 EFIAPI
TimerInterruptHandler(IN HARDWARE_INTERRUPT_SOURCE Source,IN EFI_SYSTEM_CONTEXT SystemContext)66 TimerInterruptHandler (
67   IN  HARDWARE_INTERRUPT_SOURCE   Source,
68   IN  EFI_SYSTEM_CONTEXT          SystemContext
69   )
70 {
71   EFI_TPL OriginalTPL;
72 
73   //
74   // DXE core uses this callback for the EFI timer tick. The DXE core uses locks
75   // that raise to TPL_HIGH and then restore back to current level. Thus we need
76   // to make sure TPL level is set to TPL_HIGH while we are handling the timer tick.
77   //
78   OriginalTPL = gBS->RaiseTPL (TPL_HIGH_LEVEL);
79 
80   // If the interrupt is shared then we must check if this interrupt source is the one associated to this Timer
81   if (MmioRead32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_MSK_INT_STS_REG) != 0) {
82     // Clear the periodic interrupt
83     MmioWrite32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_INT_CLR_REG, 0);
84 
85     // Signal end of interrupt early to help avoid losing subsequent ticks from long duration handlers
86     gInterrupt->EndOfInterrupt (gInterrupt, Source);
87 
88     if (mTimerNotifyFunction) {
89       mTimerNotifyFunction (mTimerPeriod);
90     }
91   }
92 
93   gBS->RestoreTPL (OriginalTPL);
94 }
95 
96 /**
97   This function registers the handler NotifyFunction so it is called every time
98   the timer interrupt fires.  It also passes the amount of time since the last
99   handler call to the NotifyFunction.  If NotifyFunction is NULL, then the
100   handler is unregistered.  If the handler is registered, then EFI_SUCCESS is
101   returned.  If the CPU does not support registering a timer interrupt handler,
102   then EFI_UNSUPPORTED is returned.  If an attempt is made to register a handler
103   when a handler is already registered, then EFI_ALREADY_STARTED is returned.
104   If an attempt is made to unregister a handler when a handler is not registered,
105   then EFI_INVALID_PARAMETER is returned.  If an error occurs attempting to
106   register the NotifyFunction with the timer interrupt, then EFI_DEVICE_ERROR
107   is returned.
108 
109   @param  This             The EFI_TIMER_ARCH_PROTOCOL instance.
110   @param  NotifyFunction   The function to call when a timer interrupt fires. This
111                            function executes at TPL_HIGH_LEVEL. The DXE Core will
112                            register a handler for the timer interrupt, so it can know
113                            how much time has passed. This information is used to
114                            signal timer based events. NULL will unregister the handler.
115   @retval EFI_SUCCESS           The timer handler was registered.
116   @retval EFI_UNSUPPORTED       The platform does not support timer interrupts.
117   @retval EFI_ALREADY_STARTED   NotifyFunction is not NULL, and a handler is already
118                                 registered.
119   @retval EFI_INVALID_PARAMETER NotifyFunction is NULL, and a handler was not
120                                 previously registered.
121   @retval EFI_DEVICE_ERROR      The timer handler could not be registered.
122 
123 **/
124 EFI_STATUS
125 EFIAPI
TimerDriverRegisterHandler(IN EFI_TIMER_ARCH_PROTOCOL * This,IN EFI_TIMER_NOTIFY NotifyFunction)126 TimerDriverRegisterHandler (
127   IN EFI_TIMER_ARCH_PROTOCOL  *This,
128   IN EFI_TIMER_NOTIFY         NotifyFunction
129   )
130 {
131   if ((NotifyFunction == NULL) && (mTimerNotifyFunction == NULL)) {
132     return EFI_INVALID_PARAMETER;
133   }
134 
135   if ((NotifyFunction != NULL) && (mTimerNotifyFunction != NULL)) {
136     return EFI_ALREADY_STARTED;
137   }
138 
139   mTimerNotifyFunction = NotifyFunction;
140 
141   return EFI_SUCCESS;
142 }
143 
144 /**
145     Make sure all Dual Timers are disabled
146 **/
147 VOID
148 EFIAPI
ExitBootServicesEvent(IN EFI_EVENT Event,IN VOID * Context)149 ExitBootServicesEvent (
150   IN EFI_EVENT  Event,
151   IN VOID       *Context
152   )
153 {
154   // Disable 'Periodic Operation' timer if enabled
155   if (MmioRead32(SP804_TIMER_PERIODIC_BASE + SP804_TIMER_CONTROL_REG) & SP804_TIMER_CTRL_ENABLE) {
156     MmioAnd32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_CONTROL_REG, 0);
157   }
158 
159   // Disable 'Metronome/Delay' timer if enabled
160   if (MmioRead32(SP804_TIMER_METRONOME_BASE + SP804_TIMER_CONTROL_REG) & SP804_TIMER_CTRL_ENABLE) {
161     MmioAnd32 (SP804_TIMER_METRONOME_BASE + SP804_TIMER_CONTROL_REG, 0);
162   }
163 
164   // Disable 'Performance' timer if enabled
165   if (MmioRead32(SP804_TIMER_PERFORMANCE_BASE + SP804_TIMER_CONTROL_REG) & SP804_TIMER_CTRL_ENABLE) {
166     MmioAnd32 (SP804_TIMER_PERFORMANCE_BASE + SP804_TIMER_CONTROL_REG, 0);
167   }
168 }
169 
170 /**
171 
172   This function adjusts the period of timer interrupts to the value specified
173   by TimerPeriod.  If the timer period is updated, then the selected timer
174   period is stored in EFI_TIMER.TimerPeriod, and EFI_SUCCESS is returned.  If
175   the timer hardware is not programmable, then EFI_UNSUPPORTED is returned.
176   If an error occurs while attempting to update the timer period, then the
177   timer hardware will be put back in its state prior to this call, and
178   EFI_DEVICE_ERROR is returned.  If TimerPeriod is 0, then the timer interrupt
179   is disabled.  This is not the same as disabling the CPU's interrupts.
180   Instead, it must either turn off the timer hardware, or it must adjust the
181   interrupt controller so that a CPU interrupt is not generated when the timer
182   interrupt fires.
183 
184   @param  This             The EFI_TIMER_ARCH_PROTOCOL instance.
185   @param  TimerPeriod      The rate to program the timer interrupt in 100 nS units. If
186                            the timer hardware is not programmable, then EFI_UNSUPPORTED is
187                            returned. If the timer is programmable, then the timer period
188                            will be rounded up to the nearest timer period that is supported
189                            by the timer hardware. If TimerPeriod is set to 0, then the
190                            timer interrupts will be disabled.
191 
192 
193   @retval EFI_SUCCESS           The timer period was changed.
194   @retval EFI_UNSUPPORTED       The platform cannot change the period of the timer interrupt.
195   @retval EFI_DEVICE_ERROR      The timer period could not be changed due to a device error.
196 
197 **/
198 EFI_STATUS
199 EFIAPI
TimerDriverSetTimerPeriod(IN EFI_TIMER_ARCH_PROTOCOL * This,IN UINT64 TimerPeriod)200 TimerDriverSetTimerPeriod (
201   IN EFI_TIMER_ARCH_PROTOCOL  *This,
202   IN UINT64                   TimerPeriod
203   )
204 {
205   EFI_STATUS  Status;
206   UINT64      TimerTicks;
207 
208   // always disable the timer
209   MmioAnd32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_CONTROL_REG, ~SP804_TIMER_CTRL_ENABLE);
210 
211   if (TimerPeriod == 0) {
212     // Leave timer disabled from above, and...
213 
214     // Disable timer 0/1 interrupt for a TimerPeriod of 0
215     Status = gInterrupt->DisableInterruptSource (gInterrupt, gVector);
216   } else {
217     // Convert TimerPeriod into 1MHz clock counts (us units = 100ns units * 10)
218     TimerTicks = DivU64x32 (TimerPeriod, 10);
219     TimerTicks = MultU64x32 (TimerTicks, PcdGet32(PcdSP804TimerFrequencyInMHz));
220 
221     // if it's larger than 32-bits, pin to highest value
222     if (TimerTicks > 0xffffffff) {
223       TimerTicks = 0xffffffff;
224     }
225 
226     // Program the SP804 timer with the new count value
227     MmioWrite32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_LOAD_REG, TimerTicks);
228 
229     // enable the timer
230     MmioOr32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_CONTROL_REG, SP804_TIMER_CTRL_ENABLE);
231 
232     // enable timer 0/1 interrupts
233     Status = gInterrupt->EnableInterruptSource (gInterrupt, gVector);
234   }
235 
236   // Save the new timer period
237   mTimerPeriod = TimerPeriod;
238   return Status;
239 }
240 
241 /**
242   This function retrieves the period of timer interrupts in 100 ns units,
243   returns that value in TimerPeriod, and returns EFI_SUCCESS.  If TimerPeriod
244   is NULL, then EFI_INVALID_PARAMETER is returned.  If a TimerPeriod of 0 is
245   returned, then the timer is currently disabled.
246 
247   @param  This             The EFI_TIMER_ARCH_PROTOCOL instance.
248   @param  TimerPeriod      A pointer to the timer period to retrieve in 100 ns units. If
249                            0 is returned, then the timer is currently disabled.
250 
251 
252   @retval EFI_SUCCESS           The timer period was returned in TimerPeriod.
253   @retval EFI_INVALID_PARAMETER TimerPeriod is NULL.
254 
255 **/
256 EFI_STATUS
257 EFIAPI
TimerDriverGetTimerPeriod(IN EFI_TIMER_ARCH_PROTOCOL * This,OUT UINT64 * TimerPeriod)258 TimerDriverGetTimerPeriod (
259   IN EFI_TIMER_ARCH_PROTOCOL   *This,
260   OUT UINT64                   *TimerPeriod
261   )
262 {
263   if (TimerPeriod == NULL) {
264     return EFI_INVALID_PARAMETER;
265   }
266 
267   *TimerPeriod = mTimerPeriod;
268   return EFI_SUCCESS;
269 }
270 
271 /**
272   This function generates a soft timer interrupt. If the platform does not support soft
273   timer interrupts, then EFI_UNSUPPORTED is returned. Otherwise, EFI_SUCCESS is returned.
274   If a handler has been registered through the EFI_TIMER_ARCH_PROTOCOL.RegisterHandler()
275   service, then a soft timer interrupt will be generated. If the timer interrupt is
276   enabled when this service is called, then the registered handler will be invoked. The
277   registered handler should not be able to distinguish a hardware-generated timer
278   interrupt from a software-generated timer interrupt.
279 
280   @param  This             The EFI_TIMER_ARCH_PROTOCOL instance.
281 
282   @retval EFI_SUCCESS           The soft timer interrupt was generated.
283   @retval EFI_UNSUPPORTED       The platform does not support the generation of soft timer interrupts.
284 
285 **/
286 EFI_STATUS
287 EFIAPI
TimerDriverGenerateSoftInterrupt(IN EFI_TIMER_ARCH_PROTOCOL * This)288 TimerDriverGenerateSoftInterrupt (
289   IN EFI_TIMER_ARCH_PROTOCOL  *This
290   )
291 {
292   return EFI_UNSUPPORTED;
293 }
294 
295 /**
296   Interface structure for the Timer Architectural Protocol.
297 
298   @par Protocol Description:
299   This protocol provides the services to initialize a periodic timer
300   interrupt, and to register a handler that is called each time the timer
301   interrupt fires.  It may also provide a service to adjust the rate of the
302   periodic timer interrupt.  When a timer interrupt occurs, the handler is
303   passed the amount of time that has passed since the previous timer
304   interrupt.
305 
306   @param RegisterHandler
307   Registers a handler that will be called each time the
308   timer interrupt fires.  TimerPeriod defines the minimum
309   time between timer interrupts, so TimerPeriod will also
310   be the minimum time between calls to the registered
311   handler.
312 
313   @param SetTimerPeriod
314   Sets the period of the timer interrupt in 100 nS units.
315   This function is optional, and may return EFI_UNSUPPORTED.
316   If this function is supported, then the timer period will
317   be rounded up to the nearest supported timer period.
318 
319 
320   @param GetTimerPeriod
321   Retrieves the period of the timer interrupt in 100 nS units.
322 
323   @param GenerateSoftInterrupt
324   Generates a soft timer interrupt that simulates the firing of
325   the timer interrupt. This service can be used to invoke the   registered handler if the timer interrupt has been masked for
326   a period of time.
327 
328 **/
329 EFI_TIMER_ARCH_PROTOCOL   gTimer = {
330   TimerDriverRegisterHandler,
331   TimerDriverSetTimerPeriod,
332   TimerDriverGetTimerPeriod,
333   TimerDriverGenerateSoftInterrupt
334 };
335 
336 
337 /**
338   Initialize the state information for the Timer Architectural Protocol and
339   the Timer Debug support protocol that allows the debugger to break into a
340   running program.
341 
342   @param  ImageHandle   of the loaded driver
343   @param  SystemTable   Pointer to the System Table
344 
345   @retval EFI_SUCCESS           Protocol registered
346   @retval EFI_OUT_OF_RESOURCES  Cannot allocate protocol data structure
347   @retval EFI_DEVICE_ERROR      Hardware problems
348 
349 **/
350 EFI_STATUS
351 EFIAPI
TimerInitialize(IN EFI_HANDLE ImageHandle,IN EFI_SYSTEM_TABLE * SystemTable)352 TimerInitialize (
353   IN EFI_HANDLE         ImageHandle,
354   IN EFI_SYSTEM_TABLE   *SystemTable
355   )
356 {
357   EFI_HANDLE  Handle = NULL;
358   EFI_STATUS  Status;
359 
360   // Set the interrupt timer number
361   gVector = PcdGet32(PcdSP804TimerPeriodicInterruptNum);
362 
363   // Find the interrupt controller protocol.  ASSERT if not found.
364   Status = gBS->LocateProtocol (&gHardwareInterruptProtocolGuid, NULL, (VOID **)&gInterrupt);
365   ASSERT_EFI_ERROR (Status);
366 
367   // Disable the timer
368   Status = TimerDriverSetTimerPeriod (&gTimer, 0);
369   ASSERT_EFI_ERROR (Status);
370 
371   // Install interrupt handler
372   Status = gInterrupt->RegisterInterruptSource (gInterrupt, gVector, TimerInterruptHandler);
373   ASSERT_EFI_ERROR (Status);
374 
375   // configure timer 0 for periodic operation, 32 bits, no prescaler, and interrupt enabled
376   MmioWrite32 (SP804_TIMER_PERIODIC_BASE + SP804_TIMER_CONTROL_REG, SP804_TIMER_CTRL_PERIODIC | SP804_TIMER_CTRL_32BIT | SP804_PRESCALE_DIV_1 | SP804_TIMER_CTRL_INT_ENABLE);
377 
378   // Set up default timer
379   Status = TimerDriverSetTimerPeriod (&gTimer, FixedPcdGet32(PcdTimerPeriod)); // TIMER_DEFAULT_PERIOD
380   ASSERT_EFI_ERROR (Status);
381 
382   // Install the Timer Architectural Protocol onto a new handle
383   Status = gBS->InstallMultipleProtocolInterfaces(
384                   &Handle,
385                   &gEfiTimerArchProtocolGuid,      &gTimer,
386                   NULL
387                   );
388   ASSERT_EFI_ERROR(Status);
389 
390   // Register for an ExitBootServicesEvent
391   Status = gBS->CreateEvent (EVT_SIGNAL_EXIT_BOOT_SERVICES, TPL_NOTIFY, ExitBootServicesEvent, NULL, &EfiExitBootServicesEvent);
392   ASSERT_EFI_ERROR (Status);
393 
394   return Status;
395 }
396