mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 03:54:02 +00:00
[NTVDM]
Part 2 of PIT + sound fix. - Move port 61h management from speaker.c to the emulator.c module; - Add PIT OUT callbacks support; - Add (unimplemented) PitSetGate function (will be used later on). Still WIP. svn path=/branches/ntvdm/; revision=61866
This commit is contained in:
parent
5dafb169e0
commit
03d8479401
6 changed files with 173 additions and 87 deletions
|
@ -31,6 +31,7 @@ LPVOID BaseAddress = NULL;
|
||||||
BOOLEAN VdmRunning = TRUE;
|
BOOLEAN VdmRunning = TRUE;
|
||||||
|
|
||||||
static BOOLEAN A20Line = FALSE;
|
static BOOLEAN A20Line = FALSE;
|
||||||
|
static BYTE Port61hState = 0x00;
|
||||||
|
|
||||||
LPCWSTR ExceptionName[] =
|
LPCWSTR ExceptionName[] =
|
||||||
{
|
{
|
||||||
|
@ -125,6 +126,79 @@ VOID WINAPI EmulatorDebugBreak(LPWORD Stack)
|
||||||
DebugBreak();
|
DebugBreak();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static BYTE WINAPI Port61hRead(ULONG Port)
|
||||||
|
{
|
||||||
|
return Port61hState;
|
||||||
|
}
|
||||||
|
|
||||||
|
static VOID WINAPI Port61hWrite(ULONG Port, BYTE Data)
|
||||||
|
{
|
||||||
|
BYTE OldPort61hState = Port61hState;
|
||||||
|
|
||||||
|
/* Only the four lowest bytes can be written */
|
||||||
|
Port61hState = (Port61hState & 0xF0) | (Data & 0x0F);
|
||||||
|
|
||||||
|
if ((OldPort61hState ^ Port61hState) & 0x01)
|
||||||
|
{
|
||||||
|
DPRINT1("PIT 2 Gate %s\n", Port61hState & 0x01 ? "on" : "off");
|
||||||
|
}
|
||||||
|
|
||||||
|
PitSetGate(2, !!(Port61hState & 0x01));
|
||||||
|
|
||||||
|
if ((OldPort61hState ^ Port61hState) & 0x02)
|
||||||
|
{
|
||||||
|
/* There were some change for the speaker... */
|
||||||
|
DPRINT1("Speaker %s\n", Port61hState & 0x02 ? "on" : "off");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static VOID WINAPI PitChan0Out(LPVOID Param, BOOLEAN State)
|
||||||
|
{
|
||||||
|
if (State)
|
||||||
|
{
|
||||||
|
DPRINT("PicInterruptRequest\n");
|
||||||
|
PicInterruptRequest(0); // Raise IRQ 0
|
||||||
|
}
|
||||||
|
// else < Lower IRQ 0 >
|
||||||
|
}
|
||||||
|
|
||||||
|
static VOID WINAPI PitChan1Out(LPVOID Param, BOOLEAN State)
|
||||||
|
{
|
||||||
|
#if 0
|
||||||
|
if (State)
|
||||||
|
{
|
||||||
|
/* Set bit 4 of Port 61h */
|
||||||
|
Port61hState |= 1 << 4;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Clear bit 4 of Port 61h */
|
||||||
|
Port61hState &= ~(1 << 4);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
Port61hState = (Port61hState & 0xEF) | (State << 4);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static VOID WINAPI PitChan2Out(LPVOID Param, BOOLEAN State)
|
||||||
|
{
|
||||||
|
#if 0
|
||||||
|
if (State)
|
||||||
|
{
|
||||||
|
/* Set bit 5 of Port 61h */
|
||||||
|
Port61hState |= 1 << 5;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Clear bit 5 of Port 61h */
|
||||||
|
Port61hState &= ~(1 << 5);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
Port61hState = (Port61hState & 0xDF) | (State << 5);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/* PUBLIC FUNCTIONS ***********************************************************/
|
/* PUBLIC FUNCTIONS ***********************************************************/
|
||||||
|
|
||||||
BOOLEAN EmulatorInitialize(HANDLE ConsoleInput, HANDLE ConsoleOutput)
|
BOOLEAN EmulatorInitialize(HANDLE ConsoleInput, HANDLE ConsoleOutput)
|
||||||
|
@ -160,6 +234,14 @@ BOOLEAN EmulatorInitialize(HANDLE ConsoleInput, HANDLE ConsoleOutput)
|
||||||
CmosInitialize();
|
CmosInitialize();
|
||||||
SpeakerInitialize();
|
SpeakerInitialize();
|
||||||
|
|
||||||
|
/* Set output functions */
|
||||||
|
PitSetOutFunction(0, NULL, PitChan0Out);
|
||||||
|
PitSetOutFunction(1, NULL, PitChan1Out);
|
||||||
|
PitSetOutFunction(2, NULL, PitChan2Out);
|
||||||
|
|
||||||
|
/* Register the I/O Ports */
|
||||||
|
RegisterIoPort(CONTROL_SYSTEM_PORT61H, Port61hRead, Port61hWrite);
|
||||||
|
|
||||||
/* Initialize the PS2 port */
|
/* Initialize the PS2 port */
|
||||||
PS2Initialize(ConsoleInput);
|
PS2Initialize(ConsoleInput);
|
||||||
|
|
||||||
|
|
|
@ -61,6 +61,10 @@
|
||||||
#define BCD_TO_BINARY(x) (((x) >> 12) * 1000 + ((x) >> 8) * 100 + ((x) >> 4) * 10 + ((x) & 0x0F))
|
#define BCD_TO_BINARY(x) (((x) >> 12) * 1000 + ((x) >> 8) * 100 + ((x) >> 4) * 10 + ((x) & 0x0F))
|
||||||
|
|
||||||
|
|
||||||
|
/* System I/O ports */
|
||||||
|
#define CONTROL_SYSTEM_PORT61H 0x61
|
||||||
|
|
||||||
|
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
EMULATOR_EXCEPTION_DIVISION_BY_ZERO,
|
EMULATOR_EXCEPTION_DIVISION_BY_ZERO,
|
||||||
|
|
|
@ -25,37 +25,19 @@
|
||||||
|
|
||||||
/* PRIVATE VARIABLES **********************************************************/
|
/* PRIVATE VARIABLES **********************************************************/
|
||||||
|
|
||||||
static BYTE Port61hState = 0x00;
|
static HANDLE hBeep = NULL;
|
||||||
HANDLE hBeep = NULL;
|
|
||||||
|
|
||||||
/* PRIVATE FUNCTIONS **********************************************************/
|
/* PRIVATE FUNCTIONS **********************************************************/
|
||||||
|
|
||||||
static BYTE SpeakerReadStatus(VOID)
|
/* PUBLIC FUNCTIONS ***********************************************************/
|
||||||
{
|
|
||||||
return Port61hState;
|
|
||||||
}
|
|
||||||
|
|
||||||
static VOID SpeakerWriteCommand(BYTE Value)
|
VOID SpeakerPool(VOID)
|
||||||
{
|
{
|
||||||
BOOLEAN IsConnectedToPITChannel2;
|
BYTE Port61hState = IOReadB(CONTROL_SYSTEM_PORT61H);
|
||||||
UCHAR SpeakerData;
|
BOOLEAN IsConnectedToPITChannel2 = !!(Port61hState & 0x01);
|
||||||
|
BOOLEAN SpeakerDataOn = !!(Port61hState & 0x02);
|
||||||
|
|
||||||
Port61hState = Value;
|
if (PitChannel2 && IsConnectedToPITChannel2 && SpeakerDataOn)
|
||||||
IsConnectedToPITChannel2 = ((Port61hState & 0x01) != 0);
|
|
||||||
SpeakerData = (Port61hState & 0x02);
|
|
||||||
|
|
||||||
if (PitChannel2 && IsConnectedToPITChannel2)
|
|
||||||
{
|
|
||||||
/* Set bit 5 of Port 61h */
|
|
||||||
Port61hState |= 1 << 5;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* Clear bit 5 of Port 61h */
|
|
||||||
Port61hState &= ~(1 << 5);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (PitChannel2 && IsConnectedToPITChannel2 && (SpeakerData != 0))
|
|
||||||
{
|
{
|
||||||
/* Start beeping - Adapted from kernel32:Beep() */
|
/* Start beeping - Adapted from kernel32:Beep() */
|
||||||
NTSTATUS Status;
|
NTSTATUS Status;
|
||||||
|
@ -121,18 +103,6 @@ static VOID SpeakerWriteCommand(BYTE Value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static BYTE WINAPI SpeakerReadPort(ULONG Port)
|
|
||||||
{
|
|
||||||
return SpeakerReadStatus();
|
|
||||||
}
|
|
||||||
|
|
||||||
static VOID WINAPI SpeakerWritePort(ULONG Port, BYTE Data)
|
|
||||||
{
|
|
||||||
SpeakerWriteCommand(Data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* PUBLIC FUNCTIONS ***********************************************************/
|
|
||||||
|
|
||||||
VOID SpeakerInitialize(VOID)
|
VOID SpeakerInitialize(VOID)
|
||||||
{
|
{
|
||||||
NTSTATUS Status;
|
NTSTATUS Status;
|
||||||
|
@ -165,9 +135,6 @@ VOID SpeakerInitialize(VOID)
|
||||||
{
|
{
|
||||||
DPRINT1("Failed to open Beep driver, Status 0x%08lx\n", Status);
|
DPRINT1("Failed to open Beep driver, Status 0x%08lx\n", Status);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Register the I/O Ports */
|
|
||||||
RegisterIoPort(SPEAKER_CONTROL_PORT, SpeakerReadPort, SpeakerWritePort);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
VOID SpeakerCleanup(VOID)
|
VOID SpeakerCleanup(VOID)
|
||||||
|
|
|
@ -15,10 +15,10 @@
|
||||||
|
|
||||||
/* DEFINES ********************************************************************/
|
/* DEFINES ********************************************************************/
|
||||||
|
|
||||||
#define SPEAKER_CONTROL_PORT 0x61
|
|
||||||
|
|
||||||
/* FUNCTIONS ******************************************************************/
|
/* FUNCTIONS ******************************************************************/
|
||||||
|
|
||||||
|
VOID SpeakerPool(VOID);
|
||||||
|
|
||||||
VOID SpeakerInitialize(VOID);
|
VOID SpeakerInitialize(VOID);
|
||||||
VOID SpeakerCleanup(VOID);
|
VOID SpeakerCleanup(VOID);
|
||||||
|
|
||||||
|
|
|
@ -37,13 +37,14 @@ static VOID PitLatchChannelStatus(BYTE Channel)
|
||||||
if (PitChannels[Channel].LatchStatusSet == FALSE)
|
if (PitChannels[Channel].LatchStatusSet == FALSE)
|
||||||
{
|
{
|
||||||
BYTE StatusLatch = 0;
|
BYTE StatusLatch = 0;
|
||||||
/* HACK!! */BYTE NullCount = 0;/* HACK!! */
|
/** HACK!! **/BYTE NullCount = 0;/** HACK!! **/
|
||||||
|
|
||||||
StatusLatch = PitChannels[Channel].Out << 7 | NullCount << 6;
|
StatusLatch = PitChannels[Channel].Out << 7 | NullCount << 6;
|
||||||
StatusLatch |= (PitChannels[Channel].ReadWriteMode & 0x03) << 4;
|
StatusLatch |= (PitChannels[Channel].ReadWriteMode & 0x03) << 4;
|
||||||
StatusLatch |= (PitChannels[Channel].Mode & 0x07) << 1;
|
StatusLatch |= (PitChannels[Channel].Mode & 0x07) << 1;
|
||||||
StatusLatch |= (PitChannels[Channel].Bcd & 0x01);
|
StatusLatch |= (PitChannels[Channel].Bcd & 0x01);
|
||||||
|
|
||||||
|
/* Latch the counter's status */
|
||||||
PitChannels[Channel].LatchStatusSet = TRUE;
|
PitChannels[Channel].LatchStatusSet = TRUE;
|
||||||
PitChannels[Channel].StatusLatch = StatusLatch;
|
PitChannels[Channel].StatusLatch = StatusLatch;
|
||||||
}
|
}
|
||||||
|
@ -61,23 +62,24 @@ static VOID PitLatchChannelCount(BYTE Channel)
|
||||||
*/
|
*/
|
||||||
if (PitChannels[Channel].ReadStatus == 0x00)
|
if (PitChannels[Channel].ReadStatus == 0x00)
|
||||||
{
|
{
|
||||||
|
/* Latch the counter's value */
|
||||||
PitChannels[Channel].ReadStatus = PitChannels[Channel].ReadWriteMode;
|
PitChannels[Channel].ReadStatus = PitChannels[Channel].ReadWriteMode;
|
||||||
|
|
||||||
/* Convert the current value to BCD if needed */
|
/* Convert the current value to BCD if needed */
|
||||||
PitChannels[Channel].OutputLatch = READ_PIT_VALUE(PitChannels[Channel],
|
PitChannels[Channel].OutputLatch =
|
||||||
PitChannels[Channel].CurrentValue);
|
READ_PIT_VALUE(PitChannels[Channel], PitChannels[Channel].CurrentValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static VOID PitSetOut(PPIT_CHANNEL Channel, BOOLEAN State)
|
static VOID PitSetOut(PPIT_CHANNEL Channel, BOOLEAN State)
|
||||||
{
|
{
|
||||||
if (State == Channel->Out) return;
|
/** HACK!! **\ if (State == Channel->Out) return; \** HACK!! **/
|
||||||
|
|
||||||
/* Set the new state of the OUT pin */
|
/* Set the new state of the OUT pin */
|
||||||
Channel->Out = State;
|
Channel->Out = State;
|
||||||
|
|
||||||
// /* Call the callback */
|
/* Call the callback */
|
||||||
// if (Channel->OutFunction) Channel->OutFunction(Channel->OutParam, State);
|
if (Channel->OutFunction) Channel->OutFunction(Channel->OutParam, State);
|
||||||
}
|
}
|
||||||
|
|
||||||
static VOID PitInitCounter(PPIT_CHANNEL Channel)
|
static VOID PitInitCounter(PPIT_CHANNEL Channel)
|
||||||
|
@ -138,20 +140,16 @@ static VOID PitWriteCommand(BYTE Value)
|
||||||
|
|
||||||
/* ... otherwise, set the modes and reset flip-flops */
|
/* ... otherwise, set the modes and reset flip-flops */
|
||||||
PitChannels[Channel].ReadWriteMode = ReadWriteMode;
|
PitChannels[Channel].ReadWriteMode = ReadWriteMode;
|
||||||
|
PitChannels[Channel].ReadStatus = 0x00;
|
||||||
|
PitChannels[Channel].WriteStatus = 0x00;
|
||||||
|
|
||||||
PitChannels[Channel].LatchStatusSet = FALSE;
|
PitChannels[Channel].LatchStatusSet = FALSE;
|
||||||
PitChannels[Channel].StatusLatch = 0x00;
|
PitChannels[Channel].StatusLatch = 0x00;
|
||||||
|
|
||||||
PitChannels[Channel].ReadStatus = 0x00;
|
|
||||||
PitChannels[Channel].WriteStatus = 0x00;
|
|
||||||
|
|
||||||
PitChannels[Channel].CountRegister = 0x00;
|
PitChannels[Channel].CountRegister = 0x00;
|
||||||
PitChannels[Channel].OutputLatch = 0x00;
|
PitChannels[Channel].OutputLatch = 0x00;
|
||||||
|
|
||||||
PitChannels[Channel].Pulsed = FALSE;
|
/** HACK!! **/PitChannels[Channel].FlipFlop = FALSE;/** HACK!! **/
|
||||||
|
|
||||||
|
|
||||||
// PitChannels[Channel].Out = FALSE; // <-- unneeded, see the PitInitCounter call below.
|
|
||||||
|
|
||||||
/* Fix the current value if we switch to BCD counting */
|
/* Fix the current value if we switch to BCD counting */
|
||||||
PitChannels[Channel].Bcd = IsBcd;
|
PitChannels[Channel].Bcd = IsBcd;
|
||||||
|
@ -192,9 +190,8 @@ static BYTE PitReadData(BYTE Channel)
|
||||||
LPWORD CurrentValue = NULL;
|
LPWORD CurrentValue = NULL;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the status was latched, the first read operation
|
* If the status was latched, the first read operation will return the
|
||||||
* will return the latched status, whichever the count
|
* latched status, whichever value (count or status) was latched first.
|
||||||
* value or the status was latched first.
|
|
||||||
*/
|
*/
|
||||||
if (PitChannels[Channel].LatchStatusSet)
|
if (PitChannels[Channel].LatchStatusSet)
|
||||||
{
|
{
|
||||||
|
@ -239,6 +236,8 @@ static VOID PitWriteData(BYTE Channel, BYTE Value)
|
||||||
PitChannels[Channel].WriteStatus = PitChannels[Channel].ReadWriteMode;
|
PitChannels[Channel].WriteStatus = PitChannels[Channel].ReadWriteMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ASSERT(PitChannels[Channel].WriteStatus != 0);
|
||||||
|
|
||||||
ReadWriteMode = &PitChannels[Channel].WriteStatus;
|
ReadWriteMode = &PitChannels[Channel].WriteStatus;
|
||||||
|
|
||||||
if (*ReadWriteMode & 1)
|
if (*ReadWriteMode & 1)
|
||||||
|
@ -261,6 +260,7 @@ static VOID PitWriteData(BYTE Channel, BYTE Value)
|
||||||
{
|
{
|
||||||
if (PitChannels[Channel].CountRegister == 0x0000)
|
if (PitChannels[Channel].CountRegister == 0x0000)
|
||||||
{
|
{
|
||||||
|
/* Wrap around to the highest count */
|
||||||
if (PitChannels[Channel].Bcd)
|
if (PitChannels[Channel].Bcd)
|
||||||
PitChannels[Channel].CountRegister = 9999;
|
PitChannels[Channel].CountRegister = 9999;
|
||||||
else
|
else
|
||||||
|
@ -268,8 +268,8 @@ static VOID PitWriteData(BYTE Channel, BYTE Value)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Convert the current value from BCD if needed */
|
/* Convert the current value from BCD if needed */
|
||||||
PitChannels[Channel].CountRegister = WRITE_PIT_VALUE(PitChannels[Channel],
|
PitChannels[Channel].CountRegister =
|
||||||
PitChannels[Channel].CountRegister);
|
WRITE_PIT_VALUE(PitChannels[Channel], PitChannels[Channel].CountRegister);
|
||||||
PitChannels[Channel].ReloadValue = PitChannels[Channel].CountRegister;
|
PitChannels[Channel].ReloadValue = PitChannels[Channel].CountRegister;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -311,6 +311,8 @@ static VOID WINAPI PitWritePort(ULONG Port, BYTE Data)
|
||||||
|
|
||||||
static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
{
|
{
|
||||||
|
if (Count == 0) return;
|
||||||
|
|
||||||
switch (Channel->Mode)
|
switch (Channel->Mode)
|
||||||
{
|
{
|
||||||
case PIT_MODE_INT_ON_TERMINAL_COUNT:
|
case PIT_MODE_INT_ON_TERMINAL_COUNT:
|
||||||
|
@ -324,11 +326,10 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
else Channel->CurrentValue -= Count;
|
else Channel->CurrentValue -= Count;
|
||||||
|
|
||||||
/* Did it fall to the terminal count? */
|
/* Did it fall to the terminal count? */
|
||||||
if (Channel->CurrentValue == 0 && !Channel->Pulsed)
|
if (Channel->CurrentValue == 0 && !Channel->Out)
|
||||||
{
|
{
|
||||||
/* Yes, raise the output line */
|
/* Yes, raise the output line */
|
||||||
if (Channel == &PitChannels[0]) PicInterruptRequest(0);
|
PitSetOut(Channel, TRUE);
|
||||||
Channel->Pulsed = TRUE;
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -342,7 +343,7 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
if ((Count > Channel->CurrentValue)
|
if ((Count > Channel->CurrentValue)
|
||||||
&& (Channel->CurrentValue != 0))
|
&& (Channel->CurrentValue != 0))
|
||||||
{
|
{
|
||||||
/* Decrease the count */
|
/* Decrement the count */
|
||||||
Count -= Channel->CurrentValue;
|
Count -= Channel->CurrentValue;
|
||||||
|
|
||||||
/* Reload the value */
|
/* Reload the value */
|
||||||
|
@ -353,7 +354,7 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* Decrease the value */
|
/* Decrement the value */
|
||||||
Channel->CurrentValue -= Count;
|
Channel->CurrentValue -= Count;
|
||||||
|
|
||||||
/* Clear the count */
|
/* Clear the count */
|
||||||
|
@ -368,8 +369,8 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If there was a reload on channel 0, raise IRQ 0 */
|
/* If there was a reload, raise the output line */
|
||||||
if ((Channel == &PitChannels[0]) && Reloaded) PicInterruptRequest(0);
|
if (Reloaded) PitSetOut(Channel, TRUE);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -387,7 +388,7 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
if (((Count * 2) > Channel->CurrentValue)
|
if (((Count * 2) > Channel->CurrentValue)
|
||||||
&& (Channel->CurrentValue != 0))
|
&& (Channel->CurrentValue != 0))
|
||||||
{
|
{
|
||||||
/* Decrease the count */
|
/* Decrement the count */
|
||||||
Count -= Channel->CurrentValue / 2;
|
Count -= Channel->CurrentValue / 2;
|
||||||
|
|
||||||
/* Reload the value */
|
/* Reload the value */
|
||||||
|
@ -398,7 +399,7 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* Decrease the value */
|
/* Decrement the value */
|
||||||
Channel->CurrentValue -= Count * 2;
|
Channel->CurrentValue -= Count * 2;
|
||||||
|
|
||||||
/* Clear the count */
|
/* Clear the count */
|
||||||
|
@ -421,16 +422,15 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
/* Toggle the flip-flop if the number of reloads was odd */
|
/* Toggle the flip-flop if the number of reloads was odd */
|
||||||
if (ReloadCount & 1)
|
if (ReloadCount & 1)
|
||||||
{
|
{
|
||||||
Channel->Out = !Channel->Out;
|
Channel->FlipFlop = !Channel->FlipFlop;
|
||||||
|
// PitSetOut(Channel, !Channel->Out);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Was there any rising edge on channel 0 ? */
|
/* Was there any rising edge? */
|
||||||
if (((Channel->Out && (ReloadCount == 1))
|
if ((Channel->FlipFlop && (ReloadCount == 1)) || (ReloadCount > 1))
|
||||||
|| (ReloadCount > 1))
|
|
||||||
&& (Channel == &PitChannels[0]))
|
|
||||||
{
|
{
|
||||||
/* Yes, IRQ 0 */
|
/* Yes, raise the output line */
|
||||||
PicInterruptRequest(0);
|
PitSetOut(Channel, TRUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
@ -453,6 +453,23 @@ static VOID PitDecrementCount(PPIT_CHANNEL Channel, DWORD Count)
|
||||||
|
|
||||||
/* PUBLIC FUNCTIONS ***********************************************************/
|
/* PUBLIC FUNCTIONS ***********************************************************/
|
||||||
|
|
||||||
|
VOID PitSetOutFunction(BYTE Channel, LPVOID Param, PIT_OUT_FUNCTION OutFunction)
|
||||||
|
{
|
||||||
|
if (Channel >= PIT_CHANNELS) return;
|
||||||
|
|
||||||
|
PitChannels[Channel].OutParam = Param;
|
||||||
|
PitChannels[Channel].OutFunction = OutFunction;
|
||||||
|
}
|
||||||
|
|
||||||
|
VOID PitSetGate(BYTE Channel, BOOLEAN State)
|
||||||
|
{
|
||||||
|
if (Channel >= PIT_CHANNELS) return;
|
||||||
|
if (State == PitChannels[Channel].Gate) return;
|
||||||
|
|
||||||
|
/* UNIMPLEMENTED */
|
||||||
|
PitChannels[Channel].Gate = State;
|
||||||
|
}
|
||||||
|
|
||||||
VOID PitClock(DWORD Count)
|
VOID PitClock(DWORD Count)
|
||||||
{
|
{
|
||||||
UINT i;
|
UINT i;
|
||||||
|
@ -461,7 +478,7 @@ VOID PitClock(DWORD Count)
|
||||||
|
|
||||||
for (i = 0; i < PIT_CHANNELS; i++)
|
for (i = 0; i < PIT_CHANNELS; i++)
|
||||||
{
|
{
|
||||||
// if (!PitChannels[i].Couting) continue;
|
// if (!PitChannels[i].Counting) continue;
|
||||||
PitDecrementCount(&PitChannels[i], Count);
|
PitDecrementCount(&PitChannels[i], Count);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -487,6 +504,14 @@ DWORD PitGetResolution(VOID)
|
||||||
|
|
||||||
VOID PitInitialize(VOID)
|
VOID PitInitialize(VOID)
|
||||||
{
|
{
|
||||||
|
/* Set up the timers to their default value */
|
||||||
|
PitSetOutFunction(0, NULL, NULL);
|
||||||
|
PitSetGate(0, TRUE);
|
||||||
|
PitSetOutFunction(1, NULL, NULL);
|
||||||
|
PitSetGate(1, TRUE);
|
||||||
|
PitSetOutFunction(2, NULL, NULL);
|
||||||
|
PitSetGate(2, FALSE);
|
||||||
|
|
||||||
/* Register the I/O Ports */
|
/* Register the I/O Ports */
|
||||||
RegisterIoPort(PIT_COMMAND_PORT, NULL , PitWritePort);
|
RegisterIoPort(PIT_COMMAND_PORT, NULL , PitWritePort);
|
||||||
RegisterIoPort(PIT_DATA_PORT(0), PitReadPort, PitWritePort);
|
RegisterIoPort(PIT_DATA_PORT(0), PitReadPort, PitWritePort);
|
||||||
|
|
|
@ -38,23 +38,25 @@ typedef enum _PIT_MODE
|
||||||
PIT_MODE_HARDWARE_STROBE
|
PIT_MODE_HARDWARE_STROBE
|
||||||
} PIT_MODE, *PPIT_MODE;
|
} PIT_MODE, *PPIT_MODE;
|
||||||
|
|
||||||
|
typedef VOID (WINAPI *PIT_OUT_FUNCTION)(LPVOID Param, BOOLEAN State);
|
||||||
|
|
||||||
typedef struct _PIT_CHANNEL
|
typedef struct _PIT_CHANNEL
|
||||||
{
|
{
|
||||||
BOOLEAN Pulsed;
|
/* PIT Status fields */
|
||||||
|
|
||||||
|
|
||||||
/* PIT Status members */
|
|
||||||
PIT_MODE Mode;
|
PIT_MODE Mode;
|
||||||
BOOLEAN Bcd;
|
BOOLEAN Bcd;
|
||||||
BYTE ReadWriteMode; // 0 --> Counter Latch ; 1 --> LSB R/W ; 2 --> MSB R/W ; 3 --> LSB then MSB R/W
|
BYTE ReadWriteMode; // 0 --> Counter Latch ; 1 --> LSB R/W ; 2 --> MSB R/W ; 3 --> LSB then MSB R/W
|
||||||
|
|
||||||
/* Reading the PIT status byte */
|
/* For interleaved reading and writing in 2-byte RW mode */
|
||||||
|
BYTE ReadStatus; // Same convention as ReadWriteMode
|
||||||
|
BYTE WriteStatus; // Same convention as ReadWriteMode
|
||||||
|
|
||||||
|
/* For reading the PIT status byte */
|
||||||
BOOLEAN LatchStatusSet;
|
BOOLEAN LatchStatusSet;
|
||||||
BYTE StatusLatch;
|
BYTE StatusLatch;
|
||||||
|
|
||||||
/* For interleaving reading and writing in 2-byte RW mode */
|
/* Counting */
|
||||||
BYTE ReadStatus; // Same convention as ReadWriteMode
|
BOOLEAN Gate;
|
||||||
BYTE WriteStatus; // Same convention as ReadWriteMode
|
|
||||||
|
|
||||||
/**/WORD CountRegister;/**/ // Our ReloadValue ???
|
/**/WORD CountRegister;/**/ // Our ReloadValue ???
|
||||||
WORD OutputLatch;
|
WORD OutputLatch;
|
||||||
|
@ -65,6 +67,9 @@ typedef struct _PIT_CHANNEL
|
||||||
|
|
||||||
/* PIT Output */
|
/* PIT Output */
|
||||||
BOOLEAN Out; // 0: Low ; 1: High
|
BOOLEAN Out; // 0: Low ; 1: High
|
||||||
|
/** HACK!! **/BOOLEAN FlipFlop;/** HACK!! **/
|
||||||
|
LPVOID OutParam;
|
||||||
|
PIT_OUT_FUNCTION OutFunction;
|
||||||
|
|
||||||
} PIT_CHANNEL, *PPIT_CHANNEL;
|
} PIT_CHANNEL, *PPIT_CHANNEL;
|
||||||
|
|
||||||
|
@ -72,6 +77,9 @@ extern PPIT_CHANNEL PitChannel2; // Needed for PC Speaker
|
||||||
|
|
||||||
/* FUNCTIONS ******************************************************************/
|
/* FUNCTIONS ******************************************************************/
|
||||||
|
|
||||||
|
VOID PitSetOutFunction(BYTE Channel, LPVOID Param, PIT_OUT_FUNCTION OutFunction);
|
||||||
|
VOID PitSetGate(BYTE Channel, BOOLEAN State);
|
||||||
|
|
||||||
VOID PitClock(DWORD Count);
|
VOID PitClock(DWORD Count);
|
||||||
DWORD PitGetResolution(VOID);
|
DWORD PitGetResolution(VOID);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue