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