Made int9 handler add keystroke to BIOS keyboard buffer.
[wine/multimedia.git] / misc / system.c
blob93026714daa6ed601c2b918aace8a10f0fc48eaa
1 /*
2 * SYSTEM DLL routines
4 * Copyright 1996 Alexandre Julliard
5 */
7 #include "wine/winbase16.h"
8 #include "wine/winuser16.h"
9 #include "services.h"
10 #include "callback.h"
11 #include "stackframe.h"
12 #include "builtin16.h"
13 #include "debugtools.h"
15 DEFAULT_DEBUG_CHANNEL(system)
17 typedef struct
19 SYSTEMTIMERPROC callback; /* NULL if not in use */
20 INT rate;
21 INT ticks;
22 } SYSTEM_TIMER;
24 #define NB_SYS_TIMERS 8
25 #define SYS_TIMER_RATE 54925
27 static SYSTEM_TIMER SYS_Timers[NB_SYS_TIMERS];
28 static int SYS_NbTimers = 0;
29 static HANDLE SYS_Service = INVALID_HANDLE_VALUE;
32 /***********************************************************************
33 * SYSTEM_TimerTick
35 static void CALLBACK SYSTEM_TimerTick( ULONG_PTR arg )
37 int i;
39 for (i = 0; i < NB_SYS_TIMERS; i++)
41 if (!SYS_Timers[i].callback) continue;
42 if ((SYS_Timers[i].ticks -= SYS_TIMER_RATE) <= 0)
44 SYS_Timers[i].ticks += SYS_Timers[i].rate;
45 SYS_Timers[i].callback( i+1 );
50 /**********************************************************************
51 * SYSTEM_StartTicks
53 * Start the system tick timer.
55 static void SYSTEM_StartTicks(void)
57 if ( SYS_Service == INVALID_HANDLE_VALUE )
58 SYS_Service = SERVICE_AddTimer( SYS_TIMER_RATE, SYSTEM_TimerTick, 0L );
62 /**********************************************************************
63 * SYSTEM_StopTicks
65 * Stop the system tick timer.
67 static void SYSTEM_StopTicks(void)
69 if ( SYS_Service != INVALID_HANDLE_VALUE )
71 SERVICE_Delete( SYS_Service );
72 SYS_Service = INVALID_HANDLE_VALUE;
77 /***********************************************************************
78 * InquireSystem (SYSTEM.1)
80 * Note: the function always takes 2 WORD arguments, contrary to what
81 * "Undocumented Windows" says.
83 DWORD WINAPI InquireSystem16( WORD code, WORD arg )
85 WORD drivetype;
87 switch(code)
89 case 0: /* Get timer resolution */
90 return SYS_TIMER_RATE;
92 case 1: /* Get drive type */
93 drivetype = GetDriveType16( arg );
94 return MAKELONG( drivetype, drivetype );
96 case 2: /* Enable one-drive logic */
97 FIXME("Case %d: set single-drive %d not supported\n", code, arg );
98 return 0;
100 WARN("Unknown code %d\n", code );
101 return 0;
105 /***********************************************************************
106 * CreateSystemTimer (SYSTEM.2)
108 WORD WINAPI CreateSystemTimer( WORD rate, SYSTEMTIMERPROC callback )
110 int i;
111 for (i = 0; i < NB_SYS_TIMERS; i++)
112 if (!SYS_Timers[i].callback) /* Found one */
114 SYS_Timers[i].rate = (UINT)rate * 1000;
115 if (SYS_Timers[i].rate < SYS_TIMER_RATE)
116 SYS_Timers[i].rate = SYS_TIMER_RATE;
117 SYS_Timers[i].ticks = SYS_Timers[i].rate;
118 SYS_Timers[i].callback = callback;
119 if (++SYS_NbTimers == 1) SYSTEM_StartTicks();
120 return i + 1; /* 0 means error */
122 return 0;
125 static void SYSTEM_CallSystemTimerProc( FARPROC16 proc, WORD timer )
127 CONTEXT86 context;
128 memset( &context, '\0', sizeof(context) );
130 CS_reg( &context ) = SELECTOROF( proc );
131 EIP_reg( &context ) = OFFSETOF( proc );
132 EBP_reg( &context ) = OFFSETOF( NtCurrentTeb()->cur_stack )
133 + (WORD)&((STACK16FRAME*)0)->bp;
135 AX_reg( &context ) = timer;
137 CallTo16RegisterShort( &context, 0 );
140 WORD WINAPI WIN16_CreateSystemTimer( WORD rate, FARPROC16 proc )
142 FARPROC thunk = THUNK_Alloc( proc, (RELAY)SYSTEM_CallSystemTimerProc );
143 WORD timer = CreateSystemTimer( rate, (SYSTEMTIMERPROC)thunk );
144 if (!timer) THUNK_Free( thunk );
145 return timer;
149 /***********************************************************************
150 * KillSystemTimer (SYSTEM.3)
152 * Note: do not confuse this function with USER.182
154 WORD WINAPI SYSTEM_KillSystemTimer( WORD timer )
156 if ( !timer || timer > NB_SYS_TIMERS || !SYS_Timers[timer-1].callback )
157 return timer; /* Error */
159 THUNK_Free( (FARPROC)SYS_Timers[timer-1].callback );
160 SYS_Timers[timer-1].callback = NULL;
162 if (!--SYS_NbTimers) SYSTEM_StopTicks();
163 return 0;
167 /***********************************************************************
168 * EnableSystemTimers (SYSTEM.4)
170 void WINAPI EnableSystemTimers16(void)
172 if ( SYS_Service != INVALID_HANDLE_VALUE )
173 SERVICE_Enable( SYS_Service );
177 /***********************************************************************
178 * DisableSystemTimers (SYSTEM.5)
180 void WINAPI DisableSystemTimers16(void)
182 if ( SYS_Service != INVALID_HANDLE_VALUE )
183 SERVICE_Disable( SYS_Service );
187 /***********************************************************************
188 * Get80x87SaveSize16 (SYSTEM.7)
190 WORD WINAPI Get80x87SaveSize16(void)
192 return 94;
196 /***********************************************************************
197 * Save80x87State16 (SYSTEM.8)
199 void WINAPI Save80x87State16( char *ptr )
201 #ifdef __i386__
202 __asm__(".byte 0x66; fsave %0; fwait" : "=m" (ptr) );
203 #endif
207 /***********************************************************************
208 * Restore80x87State16 (SYSTEM.9)
210 void WINAPI Restore80x87State16( const char *ptr )
212 #ifdef __i386__
213 __asm__(".byte 0x66; frstor %0" : : "m" (ptr) );
214 #endif