2 Copyright © 2009, The AROS Development Team. All rights reserved
11 * ---------- ------------------ -------------------------------------------------------------------
12 * 2005-03-06 T. Wiszkowski few corrections (thanks, Georg)
13 * 2005-03-05 T. Wiszkowski created file; initial benchmarked nanowait and timer-based micro/sec wait
16 #include <exec/types.h>
17 #include <devices/timer.h>
19 #include <proto/exec.h>
20 #include <aros/debug.h>
21 #include <proto/timer.h>
24 ULONG iters_per_100ns
= ~0;
26 static BOOL
ahci_Calibrate(struct IORequest
* tmr
)
28 struct Device
*TimerBase
= tmr
->io_Device
;
30 register ULONG scale
= 0x8000; // min iterations...
31 volatile register ULONG t
= 1;
32 struct timeval t1
, t2
;
34 D(bug("[AHCI ] Calibration started\n"));
36 while (scale
<= 0x80000000)
40 for (x
= 1; x
< scale
; x
++)
41 t
= (((t
+ x
) * t
) - x
) / x
; // add, mul, sub, div, trivial benchmark.
47 // ok, it's going to be totally insane, if secs > 1.
50 bug("[AHCI ] micro wait useless.\n");
55 * we expect at least 10000 times longer period, which should be 'achievable'
56 * unlikely we will cross the magic boundary here of 4 billion instructions in 10 millisecond (yielding 400'000MIPS?)
57 * on the other side, if we go as low as 1, then 4 iterations of add/sub/mul/div is perfectly fine yielding a bit more than 400ns...
60 if (t2
.tv_micro
>= 10000)
65 D(bug("[AHCI ] Executed %ld ops in %ldus\n", scale
, t2
.tv_micro
));
67 // always round up to the next value.. so 30.9 -> 31, 5.1 -> 6, etc
68 x
= (x
+ t2
.tv_micro
- 1) / t2
.tv_micro
;
71 bug("[AHCI ] Approximate number of iterations per 100 nanoseconds: %ld\n", x
);
76 struct IORequest
*ahci_OpenTimer()
78 struct MsgPort
*p
= CreateMsgPort();
81 struct IORequest
*io
= CreateIORequest(p
, sizeof(struct timerequest
));
86 * We only need this timer for relatively long delays
88 if (0 == OpenDevice(TIMERNAME
, UNIT_MICROHZ
, io
, 0))
90 if (iters_per_100ns
== ~0)
98 bug("[AHCI ] Failed to open timer.device, unit MICROHZ\n");
104 bug("[AHCI ] Failed to create timerequest\n");
110 bug("[AHCI ] Failed to create timer port\n");
116 void ahci_CloseTimer(struct IORequest
*tmr
)
120 struct MsgPort
*p
= tmr
->io_Message
.mn_ReplyPort
;
122 DeleteIORequest(tmr
);
127 void ahci_WaitNano(register ULONG ns
)
129 volatile register ULONG t
= 1;
130 ns
= (ns
+ 99) / 100;
131 ns
*= iters_per_100ns
;
134 t
= (((t
+ ns
) * t
) - ns
) / ns
; // add, mul, sub, div, trivial benchmark.
139 ULONG
ahci_WaitTO(struct IORequest
* tmr
, ULONG secs
, ULONG micro
, ULONG sigs
)
141 ULONG iosig
= 1 << tmr
->io_Message
.mn_ReplyPort
->mp_SigBit
;
143 //D(bug("[AHCI--] Timed wait %lds %ldu\n", secs, micro));
145 tmr
->io_Command
= TR_ADDREQUEST
;
147 ((struct timerequest
*)tmr
)->tr_time
.tv_secs
= secs
;
148 ((struct timerequest
*)tmr
)->tr_time
.tv_micro
= micro
;
151 sigs
= Wait(sigs
| iosig
);
152 if (0 == (sigs
& iosig
)) {
160 return sigs
&~ iosig
;