mirror of
https://github.com/reactos/reactos.git
synced 2025-05-31 06:58:10 +00:00
PC speaker MIDI driver (see README.TXT for info)
Tested on Windows XP but should be compatible with ReactOS svn path=/trunk/; revision=25520
This commit is contained in:
parent
f8edfc8d99
commit
8a64d7ef61
4 changed files with 978 additions and 0 deletions
903
reactos/dll/win32/beepmidi/beepmidi.c
Normal file
903
reactos/dll/win32/beepmidi/beepmidi.c
Normal file
|
@ -0,0 +1,903 @@
|
||||||
|
/*
|
||||||
|
BeepMidi :: beep.sys MIDI player
|
||||||
|
|
||||||
|
(c) Andrew Greenwood, 2007.
|
||||||
|
|
||||||
|
Released as open-source software. You may copy, re-distribute and modify
|
||||||
|
this software, provided this copyright notice remains intact.
|
||||||
|
|
||||||
|
Please see the included README.TXT for more information
|
||||||
|
|
||||||
|
HISTORY :
|
||||||
|
16th January 2007 Started
|
||||||
|
17th January 2007 Polyphony support and threading added
|
||||||
|
18th January 2007 Made threading optional, added comments
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* The timeslice to allocate for all playing notes (in milliseconds) */
|
||||||
|
#define TIMESLICE_SIZE 60
|
||||||
|
|
||||||
|
/*
|
||||||
|
If this is defined, notes are added to the playing list, even if
|
||||||
|
they already exist. As a result, the note will sound twice during
|
||||||
|
each timeslice. Also each note on will require a corresponding note
|
||||||
|
off event.
|
||||||
|
*/
|
||||||
|
#define ALLOW_DUPLICATE_NOTES
|
||||||
|
|
||||||
|
/*
|
||||||
|
The maximum number of notes that may be playing at any one time.
|
||||||
|
Higher values result in a messier sound as all the frequencies get
|
||||||
|
mashed together. Do not set this below 2. Recommended = 4
|
||||||
|
*/
|
||||||
|
#define POLYPHONY 3
|
||||||
|
|
||||||
|
/*
|
||||||
|
Define CONTINUOUS_NOTES to perform note playback in a separate thread.
|
||||||
|
This was originally the intended behaviour, but after experimentation
|
||||||
|
doesn't sound as good for MIDI files which have a lot going on. If not
|
||||||
|
defined, all playing notes are output in sequence as a new note starts.
|
||||||
|
*/
|
||||||
|
#define CONTINUOUS_NOTES
|
||||||
|
|
||||||
|
#define WIN32_NO_STATUS
|
||||||
|
#define NTOS_MODE_USER
|
||||||
|
#include <windows.h>
|
||||||
|
#include <ndk/ntndk.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <ntddbeep.h>
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
|
#include <mmddk.h>
|
||||||
|
#include <mmsystem.h>
|
||||||
|
|
||||||
|
#define DPRINT printf
|
||||||
|
//#define DPRINT //
|
||||||
|
|
||||||
|
/* A few MIDI command categories */
|
||||||
|
#define MIDI_NOTE_OFF 0x80
|
||||||
|
#define MIDI_NOTE_ON 0x90
|
||||||
|
#define MIDI_CONTROL_CHANGE 0xB0
|
||||||
|
#define MIDI_PROGRAM 0xC0
|
||||||
|
#define MIDI_PITCH_BEND 0xE0
|
||||||
|
#define MIDI_SYSTEM 0xFF
|
||||||
|
|
||||||
|
/* Specific commands */
|
||||||
|
#define MIDI_RESET 0xFF
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct _NoteNode
|
||||||
|
{
|
||||||
|
struct _NoteNode* next;
|
||||||
|
struct _NoteNode* previous;
|
||||||
|
|
||||||
|
UCHAR note;
|
||||||
|
UCHAR velocity; /* 0 is note-off */
|
||||||
|
} NoteNode;
|
||||||
|
|
||||||
|
typedef struct _DeviceInfo
|
||||||
|
{
|
||||||
|
HDRVR mme_handle;
|
||||||
|
HANDLE kernel_device;
|
||||||
|
|
||||||
|
DWORD callback;
|
||||||
|
DWORD instance;
|
||||||
|
DWORD flags;
|
||||||
|
|
||||||
|
UCHAR running_status;
|
||||||
|
|
||||||
|
DWORD playing_notes_count;
|
||||||
|
NoteNode* note_list;
|
||||||
|
BOOL refresh_notes;
|
||||||
|
|
||||||
|
HANDLE thread_handle;
|
||||||
|
BOOL terminate_thread;
|
||||||
|
HANDLE thread_termination_complete;
|
||||||
|
} DeviceInfo;
|
||||||
|
|
||||||
|
DeviceInfo* the_device;
|
||||||
|
CRITICAL_SECTION device_lock;
|
||||||
|
|
||||||
|
void
|
||||||
|
FakePrintf(char* str, ...)
|
||||||
|
{
|
||||||
|
/* Just to shut the compiler up */
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
This is designed to be treated as a thread, however it behaves as a
|
||||||
|
normal function if CONTINUOUS_NOTES is not defined.
|
||||||
|
*/
|
||||||
|
|
||||||
|
DWORD WINAPI
|
||||||
|
ProcessPlayingNotes(
|
||||||
|
LPVOID parameter)
|
||||||
|
{
|
||||||
|
DeviceInfo* device_info = (DeviceInfo*) parameter;
|
||||||
|
NTSTATUS status;
|
||||||
|
IO_STATUS_BLOCK io_status_block;
|
||||||
|
DWORD arp_notes;
|
||||||
|
|
||||||
|
DPRINT("Note processing started\n");
|
||||||
|
|
||||||
|
/* We lock the note list only while accessing it */
|
||||||
|
|
||||||
|
#ifdef CONTINUOUS_NOTES
|
||||||
|
while ( ! device_info->terminate_thread )
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
NoteNode* node;
|
||||||
|
|
||||||
|
/* Number of notes being arpeggiated */
|
||||||
|
arp_notes = 1;
|
||||||
|
|
||||||
|
EnterCriticalSection(&device_lock);
|
||||||
|
|
||||||
|
/* Calculate how much time to allocate to each playing note */
|
||||||
|
|
||||||
|
DPRINT("%d notes active\n", (int) device_info->playing_notes_count);
|
||||||
|
|
||||||
|
node = device_info->note_list;
|
||||||
|
|
||||||
|
while ( ( node != NULL ) && ( arp_notes <= POLYPHONY ) )
|
||||||
|
{
|
||||||
|
DPRINT("playing..\n");
|
||||||
|
BEEP_SET_PARAMETERS beep_data;
|
||||||
|
DWORD actually_playing = 0;
|
||||||
|
|
||||||
|
double frequency = node->note;
|
||||||
|
frequency = frequency / 12;
|
||||||
|
frequency = pow(2, frequency);
|
||||||
|
frequency = 8.1758 * frequency;
|
||||||
|
|
||||||
|
if (device_info->playing_notes_count > POLYPHONY)
|
||||||
|
actually_playing = POLYPHONY;
|
||||||
|
else
|
||||||
|
actually_playing = device_info->playing_notes_count;
|
||||||
|
|
||||||
|
DPRINT("Frequency %f\n", frequency);
|
||||||
|
|
||||||
|
// TODO
|
||||||
|
beep_data.Frequency = (DWORD) frequency;
|
||||||
|
beep_data.Duration = TIMESLICE_SIZE / actually_playing; /* device_info->playing_notes_count; */
|
||||||
|
|
||||||
|
status = NtDeviceIoControlFile(device_info->kernel_device,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
&io_status_block,
|
||||||
|
IOCTL_BEEP_SET,
|
||||||
|
&beep_data,
|
||||||
|
sizeof(BEEP_SET_PARAMETERS),
|
||||||
|
NULL,
|
||||||
|
0);
|
||||||
|
|
||||||
|
if ( ! NT_SUCCESS(status) )
|
||||||
|
{
|
||||||
|
DPRINT("ERROR %d\n", (int) GetLastError());
|
||||||
|
}
|
||||||
|
|
||||||
|
SleepEx(beep_data.Duration, TRUE);
|
||||||
|
|
||||||
|
if ( device_info->refresh_notes )
|
||||||
|
{
|
||||||
|
device_info->refresh_notes = FALSE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
arp_notes ++;
|
||||||
|
node = node->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
LeaveCriticalSection(&device_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef CONTINUOUS_NOTES
|
||||||
|
SetEvent(device_info->thread_termination_complete);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Fills a MIDIOUTCAPS structure with information about our device.
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
GetDeviceCapabilities(
|
||||||
|
MIDIOUTCAPS* caps)
|
||||||
|
{
|
||||||
|
/* These are ignored for now */
|
||||||
|
caps->wMid = 0;
|
||||||
|
caps->wPid = 0;
|
||||||
|
|
||||||
|
caps->vDriverVersion = 0x0100;
|
||||||
|
|
||||||
|
memset(caps->szPname, 0, sizeof(caps->szPname));
|
||||||
|
memcpy(caps->szPname, L"PC speaker\0", strlen("PC speaker\0") * 2);
|
||||||
|
|
||||||
|
caps->wTechnology = MOD_SQSYNTH;
|
||||||
|
|
||||||
|
caps->wVoices = 1; /* We only have one voice */
|
||||||
|
caps->wNotes = POLYPHONY;
|
||||||
|
caps->wChannelMask = 0xFFBF; /* Ignore channel 10 */
|
||||||
|
|
||||||
|
caps->dwSupport = 0;
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Helper function that just simplifies calling the application making use
|
||||||
|
of us.
|
||||||
|
*/
|
||||||
|
|
||||||
|
BOOL
|
||||||
|
CallClient(
|
||||||
|
DeviceInfo* device_info,
|
||||||
|
DWORD message,
|
||||||
|
DWORD parameter1,
|
||||||
|
DWORD parameter2)
|
||||||
|
{
|
||||||
|
DPRINT("Calling client - callback 0x%x mmhandle 0x%x\n", (int) device_info->callback, (int) device_info->mme_handle);
|
||||||
|
return DriverCallback(device_info->callback,
|
||||||
|
HIWORD(device_info->flags),
|
||||||
|
device_info->mme_handle,
|
||||||
|
message,
|
||||||
|
device_info->instance,
|
||||||
|
parameter1,
|
||||||
|
parameter2);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Open the kernel-mode device and allocate resources. This opens the
|
||||||
|
BEEP.SYS kernel device.
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
OpenDevice(
|
||||||
|
DeviceInfo** private_data,
|
||||||
|
MIDIOPENDESC* open_desc,
|
||||||
|
DWORD flags)
|
||||||
|
{
|
||||||
|
NTSTATUS status;
|
||||||
|
HANDLE heap;
|
||||||
|
HANDLE kernel_device;
|
||||||
|
UNICODE_STRING beep_device_name;
|
||||||
|
OBJECT_ATTRIBUTES attribs;
|
||||||
|
IO_STATUS_BLOCK status_block;
|
||||||
|
|
||||||
|
/* One at a time.. */
|
||||||
|
if ( the_device )
|
||||||
|
{
|
||||||
|
DPRINT("Already allocated\n");
|
||||||
|
return MMSYSERR_ALLOCATED;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make the device name into a unicode string and open it */
|
||||||
|
|
||||||
|
RtlInitUnicodeString(&beep_device_name,
|
||||||
|
L"\\Device\\Beep");
|
||||||
|
|
||||||
|
InitializeObjectAttributes(&attribs,
|
||||||
|
&beep_device_name,
|
||||||
|
0,
|
||||||
|
NULL,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
status = NtCreateFile(&kernel_device,
|
||||||
|
FILE_READ_DATA | FILE_WRITE_DATA,
|
||||||
|
&attribs,
|
||||||
|
&status_block,
|
||||||
|
NULL,
|
||||||
|
0,
|
||||||
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
||||||
|
FILE_OPEN_IF,
|
||||||
|
0,
|
||||||
|
NULL,
|
||||||
|
0);
|
||||||
|
|
||||||
|
if ( ! NT_SUCCESS(status) )
|
||||||
|
{
|
||||||
|
DPRINT("Could not connect to BEEP device - %d\n", (int) GetLastError());
|
||||||
|
return MMSYSERR_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
DPRINT("Opened!\n");
|
||||||
|
|
||||||
|
/* Allocate and initialize the device info */
|
||||||
|
|
||||||
|
heap = GetProcessHeap();
|
||||||
|
|
||||||
|
the_device = HeapAlloc(heap, HEAP_ZERO_MEMORY, sizeof(DeviceInfo));
|
||||||
|
|
||||||
|
if ( ! the_device )
|
||||||
|
{
|
||||||
|
DPRINT("Out of memory\n");
|
||||||
|
return MMSYSERR_NOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Initialize */
|
||||||
|
the_device->kernel_device = kernel_device;
|
||||||
|
the_device->playing_notes_count = 0;
|
||||||
|
the_device->note_list = NULL;
|
||||||
|
the_device->thread_handle = 0;
|
||||||
|
the_device->terminate_thread = FALSE;
|
||||||
|
the_device->running_status = 0;
|
||||||
|
|
||||||
|
// TODO
|
||||||
|
the_device->mme_handle = (HDRVR) open_desc->hMidi;
|
||||||
|
the_device->callback = open_desc->dwCallback;
|
||||||
|
the_device->instance = open_desc->dwInstance;
|
||||||
|
the_device->flags = flags;
|
||||||
|
|
||||||
|
/* Store the pointer in the user data */
|
||||||
|
*private_data = the_device;
|
||||||
|
|
||||||
|
/* This is threading-related code */
|
||||||
|
#ifdef CONTINUOUS_NOTES
|
||||||
|
the_device->thread_termination_complete = CreateEvent(NULL, FALSE, FALSE, NULL);
|
||||||
|
|
||||||
|
if ( ! the_device->thread_termination_complete )
|
||||||
|
{
|
||||||
|
DPRINT("CreateEvent failed\n");
|
||||||
|
HeapFree(heap, 0, the_device);
|
||||||
|
return MMSYSERR_NOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
the_device->thread_handle = CreateThread(NULL,
|
||||||
|
0,
|
||||||
|
ProcessPlayingNotes,
|
||||||
|
(PVOID) the_device,
|
||||||
|
0,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
if ( ! the_device->thread_handle )
|
||||||
|
{
|
||||||
|
DPRINT("CreateThread failed\n");
|
||||||
|
CloseHandle(the_device->thread_termination_complete);
|
||||||
|
HeapFree(heap, 0, the_device);
|
||||||
|
return MMSYSERR_NOMEM;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Now we call the client application to say the device is open */
|
||||||
|
DPRINT("Sending MOM_OPEN\n");
|
||||||
|
DPRINT("Success? %d\n", (int) CallClient(the_device, MOM_OPEN, 0, 0));
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Close the kernel-mode device.
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
CloseDevice(DeviceInfo* device_info)
|
||||||
|
{
|
||||||
|
HANDLE heap = GetProcessHeap();
|
||||||
|
|
||||||
|
/* If we're working in threaded mode we need to wait for thread to die */
|
||||||
|
#ifdef CONTINUOUS_NOTES
|
||||||
|
the_device->terminate_thread = TRUE;
|
||||||
|
|
||||||
|
WaitForSingleObject(the_device->thread_termination_complete, INFINITE);
|
||||||
|
|
||||||
|
CloseHandle(the_device->thread_termination_complete);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Let the client application know the device is closing */
|
||||||
|
DPRINT("Sending MOM_CLOSE\n");
|
||||||
|
CallClient(device_info, MOM_CLOSE, 0, 0);
|
||||||
|
|
||||||
|
NtClose(device_info->kernel_device);
|
||||||
|
|
||||||
|
/* Free resources */
|
||||||
|
HeapFree(heap, 0, device_info);
|
||||||
|
|
||||||
|
the_device = NULL;
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Removes a note from the playing notes list. If the note is not playing,
|
||||||
|
we just pretend nothing happened.
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
StopNote(
|
||||||
|
DeviceInfo* device_info,
|
||||||
|
UCHAR note)
|
||||||
|
{
|
||||||
|
HANDLE heap = GetProcessHeap();
|
||||||
|
NoteNode* node;
|
||||||
|
NoteNode* prev_node = NULL;
|
||||||
|
|
||||||
|
DPRINT("StopNote\n");
|
||||||
|
|
||||||
|
EnterCriticalSection(&device_lock);
|
||||||
|
|
||||||
|
node = device_info->note_list;
|
||||||
|
|
||||||
|
while ( node != NULL )
|
||||||
|
{
|
||||||
|
if ( node->note == note )
|
||||||
|
{
|
||||||
|
/* Found the note - just remove the node from the list */
|
||||||
|
|
||||||
|
DPRINT("Stopping note %d\n", (int) node->note);
|
||||||
|
|
||||||
|
if ( prev_node != NULL )
|
||||||
|
prev_node->next = node->next;
|
||||||
|
else
|
||||||
|
device_info->note_list = node->next;
|
||||||
|
|
||||||
|
HeapFree(heap, 0, node);
|
||||||
|
|
||||||
|
device_info->playing_notes_count --;
|
||||||
|
|
||||||
|
DPRINT("Note stopped - now playing %d notes\n", (int) device_info->playing_notes_count);
|
||||||
|
|
||||||
|
LeaveCriticalSection(&device_lock);
|
||||||
|
device_info->refresh_notes = TRUE;
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
prev_node = node;
|
||||||
|
node = node->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
LeaveCriticalSection(&device_lock);
|
||||||
|
|
||||||
|
/* Hmm, a good idea? */
|
||||||
|
#ifndef CONTINUOUS_NOTES
|
||||||
|
ProcessPlayingNotes((PVOID) device_info);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Adds a note to the playing notes list. If the note is already playing,
|
||||||
|
the definition of ALLOW_DUPLICATE_NOTES determines if an existing note
|
||||||
|
may be duplicated. Otherwise, duplicate notes are ignored.
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
PlayNote(
|
||||||
|
DeviceInfo* device_info,
|
||||||
|
UCHAR note,
|
||||||
|
UCHAR velocity)
|
||||||
|
{
|
||||||
|
HANDLE heap = GetProcessHeap();
|
||||||
|
|
||||||
|
DPRINT("PlayNote\n");
|
||||||
|
|
||||||
|
NoteNode* node;
|
||||||
|
|
||||||
|
if ( velocity == 0 )
|
||||||
|
{
|
||||||
|
DPRINT("Zero velocity\n");
|
||||||
|
|
||||||
|
/* Velocity zero is effectively a "note off" */
|
||||||
|
StopNote(device_info, note);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Start playing the note */
|
||||||
|
NoteNode* new_node;
|
||||||
|
NoteNode* tail_node = NULL;
|
||||||
|
|
||||||
|
EnterCriticalSection(&device_lock);
|
||||||
|
|
||||||
|
node = device_info->note_list;
|
||||||
|
|
||||||
|
while ( node != NULL )
|
||||||
|
{
|
||||||
|
#ifndef ALLOW_DUPLICATE_NOTES
|
||||||
|
if ( ( node->note == note ) && ( velocity > 0 ) )
|
||||||
|
{
|
||||||
|
/* The note is already playing - do nothing */
|
||||||
|
DPRINT("Duplicate note playback request ignored\n");
|
||||||
|
LeaveCriticalSection(&device_lock);
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
tail_node = node;
|
||||||
|
node = node->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
new_node = HeapAlloc(heap, HEAP_ZERO_MEMORY, sizeof(NoteNode));
|
||||||
|
|
||||||
|
if ( ! new_node )
|
||||||
|
{
|
||||||
|
LeaveCriticalSection(&device_lock);
|
||||||
|
return MMSYSERR_NOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
new_node->note = note;
|
||||||
|
new_node->velocity = velocity;
|
||||||
|
|
||||||
|
/*
|
||||||
|
Prepend to the playing notes list. If exceeding polyphony,
|
||||||
|
remove the oldest note (which will be at the tail.)
|
||||||
|
*/
|
||||||
|
|
||||||
|
if ( device_info->note_list )
|
||||||
|
device_info->note_list->previous = new_node;
|
||||||
|
|
||||||
|
new_node->next = device_info->note_list;
|
||||||
|
new_node->previous = NULL;
|
||||||
|
|
||||||
|
device_info->note_list = new_node;
|
||||||
|
device_info->playing_notes_count ++;
|
||||||
|
|
||||||
|
/*
|
||||||
|
if ( device_info->playing_notes_count > POLYPHONY )
|
||||||
|
{
|
||||||
|
ASSERT(tail_node);
|
||||||
|
|
||||||
|
DPRINT("Polyphony exceeded\n");
|
||||||
|
|
||||||
|
tail_node->previous->next = NULL;
|
||||||
|
|
||||||
|
HeapFree(heap, 0, tail_node);
|
||||||
|
|
||||||
|
device_info->playing_notes_count --;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
LeaveCriticalSection(&device_lock);
|
||||||
|
|
||||||
|
DPRINT("Note started - now playing %d notes\n", (int) device_info->playing_notes_count);
|
||||||
|
device_info->refresh_notes = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef CONTINUOUS_NOTES
|
||||||
|
ProcessPlayingNotes((PVOID) device_info);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Decipher a short MIDI message (which is a MIDI message packed into a DWORD.)
|
||||||
|
This will set "running status", but does not take this into account when
|
||||||
|
processing messages (is this necessary?)
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
ProcessShortMidiMessage(
|
||||||
|
DeviceInfo* device_info,
|
||||||
|
DWORD message)
|
||||||
|
{
|
||||||
|
DWORD status;
|
||||||
|
|
||||||
|
DWORD category;
|
||||||
|
DWORD channel;
|
||||||
|
DWORD data1, data2;
|
||||||
|
|
||||||
|
status = message & 0x000000FF;
|
||||||
|
|
||||||
|
/* Deal with running status */
|
||||||
|
|
||||||
|
if ( status < MIDI_NOTE_OFF )
|
||||||
|
{
|
||||||
|
status = device_info->running_status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure the status is sane! */
|
||||||
|
|
||||||
|
if ( status < MIDI_NOTE_OFF )
|
||||||
|
{
|
||||||
|
/* It's garbage, ignore it */
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Figure out the message category and channel */
|
||||||
|
|
||||||
|
category = status & 0xF0;
|
||||||
|
channel = status & 0x0F; /* we don't use this */
|
||||||
|
|
||||||
|
data1 = (message & 0x0000FF00) >> 8;
|
||||||
|
data2 = (message & 0x00FF0000) >> 16;
|
||||||
|
|
||||||
|
DPRINT("0x%x, %d, %d\n", (int) status, (int) data1, (int) data2);
|
||||||
|
|
||||||
|
/* Filter drums (which are *usually* on channel 10) */
|
||||||
|
if ( channel == 10 )
|
||||||
|
{
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Pass to the appropriate message handler */
|
||||||
|
|
||||||
|
switch ( category )
|
||||||
|
{
|
||||||
|
case MIDI_NOTE_ON :
|
||||||
|
{
|
||||||
|
PlayNote(device_info, data1, data2);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case MIDI_NOTE_OFF :
|
||||||
|
{
|
||||||
|
StopNote(device_info, data1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#define PACK_MIDI(b1, b2, b3) \
|
||||||
|
((b3 * 65536) + (b2 * 256) + b1);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Processes a "long" MIDI message (ie, a MIDI message contained within a
|
||||||
|
buffer.) This is intended for supporting SysEx data, or blocks of MIDI
|
||||||
|
events. However in our case we're only interested in short MIDI messages,
|
||||||
|
so we scan the buffer, and each time we encounter a valid status byte
|
||||||
|
we start recording it as a new event. Once 3 bytes or a new status is
|
||||||
|
received, the event is passed to the short message handler.
|
||||||
|
*/
|
||||||
|
|
||||||
|
MMRESULT
|
||||||
|
ProcessLongMidiMessage(
|
||||||
|
DeviceInfo* device_info,
|
||||||
|
MIDIHDR* header)
|
||||||
|
{
|
||||||
|
int index = 0;
|
||||||
|
UCHAR* midi_bytes = (UCHAR*) header->lpData;
|
||||||
|
|
||||||
|
int msg_index = 0;
|
||||||
|
UCHAR msg[3];
|
||||||
|
|
||||||
|
/* Initialize the buffer */
|
||||||
|
msg[0] = msg[1] = msg[2] = 0;
|
||||||
|
|
||||||
|
if ( ! ( header->dwFlags & MHDR_PREPARED ) )
|
||||||
|
{
|
||||||
|
DPRINT("Not prepared!\n");
|
||||||
|
return MIDIERR_UNPREPARED;
|
||||||
|
}
|
||||||
|
|
||||||
|
DPRINT("Processing %d bytes of MIDI\n", (int) header->dwBufferLength);
|
||||||
|
|
||||||
|
while ( index < header->dwBufferLength )
|
||||||
|
{
|
||||||
|
/* New status byte? ( = new event) */
|
||||||
|
if ( midi_bytes[index] & 0x80 )
|
||||||
|
{
|
||||||
|
DWORD short_msg;
|
||||||
|
|
||||||
|
/* Deal with the existing event */
|
||||||
|
|
||||||
|
if ( msg[0] & 0x80 )
|
||||||
|
{
|
||||||
|
short_msg = PACK_MIDI(msg[0], msg[1], msg[2]);
|
||||||
|
|
||||||
|
DPRINT("Complete msg is 0x%x %d %d\n", (int) msg[0], (int) msg[1], (int) msg[2]);
|
||||||
|
ProcessShortMidiMessage(device_info, short_msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set new running status and start recording the event */
|
||||||
|
DPRINT("Set new running status\n");
|
||||||
|
device_info->running_status = midi_bytes[index];
|
||||||
|
msg[0] = midi_bytes[index];
|
||||||
|
msg_index = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Unexpected data byte? ( = re-use previous status) */
|
||||||
|
else if ( msg_index == 0 )
|
||||||
|
{
|
||||||
|
if ( device_info->running_status & 0x80 )
|
||||||
|
{
|
||||||
|
DPRINT("Retrieving running status\n");
|
||||||
|
msg[0] = device_info->running_status;
|
||||||
|
msg[1] = midi_bytes[index];
|
||||||
|
msg_index = 2;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
DPRINT("garbage\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Expected data ( = append to message until buffer full) */
|
||||||
|
else
|
||||||
|
{
|
||||||
|
DPRINT("Next byte...\n");
|
||||||
|
msg[msg_index] = midi_bytes[index];
|
||||||
|
msg_index ++;
|
||||||
|
|
||||||
|
if ( msg_index > 2 )
|
||||||
|
{
|
||||||
|
DWORD short_msg;
|
||||||
|
|
||||||
|
short_msg = PACK_MIDI(msg[0], msg[1], msg[2]);
|
||||||
|
|
||||||
|
DPRINT("Complete msg is 0x%x %d %d\n", (int) msg[0], (int) msg[1], (int) msg[2]);
|
||||||
|
ProcessShortMidiMessage(device_info, short_msg);
|
||||||
|
|
||||||
|
/* Reinit */
|
||||||
|
msg_index = 0;
|
||||||
|
msg[0] = msg[1] = msg[2] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
index ++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
We're meant to clear MHDR_DONE and set MHDR_INQUEUE but since we
|
||||||
|
deal with everything here and now we might as well just say so.
|
||||||
|
*/
|
||||||
|
header->dwFlags |= MHDR_DONE;
|
||||||
|
header->dwFlags &= ~ MHDR_INQUEUE;
|
||||||
|
|
||||||
|
DPRINT("Success? %d\n", (int) CallClient(the_device, MOM_DONE, (DWORD) header, 0));
|
||||||
|
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Exported function that receives messages from WINMM (the MME API.)
|
||||||
|
*/
|
||||||
|
|
||||||
|
FAR PASCAL
|
||||||
|
MMRESULT
|
||||||
|
modMessage(
|
||||||
|
UINT device_id,
|
||||||
|
UINT message,
|
||||||
|
DWORD private_data,
|
||||||
|
DWORD parameter1,
|
||||||
|
DWORD parameter2)
|
||||||
|
{
|
||||||
|
switch ( message )
|
||||||
|
{
|
||||||
|
case MODM_GETNUMDEVS :
|
||||||
|
{
|
||||||
|
/* Only one internal PC speaker device (and even that's too much) */
|
||||||
|
DPRINT("MODM_GETNUMDEVS\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_GETDEVCAPS :
|
||||||
|
{
|
||||||
|
DPRINT("MODM_GETDEVCAPS\n");
|
||||||
|
return GetDeviceCapabilities((MIDIOUTCAPS*) parameter1);
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_OPEN :
|
||||||
|
{
|
||||||
|
DPRINT("MODM_OPEN\n");
|
||||||
|
|
||||||
|
return OpenDevice((DeviceInfo**) private_data,
|
||||||
|
(MIDIOPENDESC*) parameter1,
|
||||||
|
parameter2);
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_CLOSE :
|
||||||
|
{
|
||||||
|
DPRINT("MODM_CLOSE\n");
|
||||||
|
return CloseDevice((DeviceInfo*) private_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_DATA :
|
||||||
|
{
|
||||||
|
return ProcessShortMidiMessage((DeviceInfo*) private_data, parameter1);
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_PREPARE :
|
||||||
|
{
|
||||||
|
/* We don't bother with this */
|
||||||
|
MIDIHDR* hdr = (MIDIHDR*) parameter1;
|
||||||
|
hdr->dwFlags |= MHDR_PREPARED;
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_UNPREPARE :
|
||||||
|
{
|
||||||
|
MIDIHDR* hdr = (MIDIHDR*) parameter1;
|
||||||
|
hdr->dwFlags &= ~MHDR_PREPARED;
|
||||||
|
return MMSYSERR_NOERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_LONGDATA :
|
||||||
|
{
|
||||||
|
DPRINT("LONGDATA\n");
|
||||||
|
return ProcessLongMidiMessage((DeviceInfo*) private_data, (MIDIHDR*) parameter1);
|
||||||
|
}
|
||||||
|
|
||||||
|
case MODM_RESET :
|
||||||
|
{
|
||||||
|
/* TODO */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DPRINT("Not supported %d\n", message);
|
||||||
|
|
||||||
|
return MMSYSERR_NOTSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Driver entrypoint.
|
||||||
|
*/
|
||||||
|
|
||||||
|
FAR PASCAL LONG
|
||||||
|
DriverProc(
|
||||||
|
DWORD driver_id,
|
||||||
|
HDRVR driver_handle,
|
||||||
|
UINT message,
|
||||||
|
LONG parameter1,
|
||||||
|
LONG parameter2)
|
||||||
|
{
|
||||||
|
switch ( message )
|
||||||
|
{
|
||||||
|
case DRV_LOAD :
|
||||||
|
DPRINT("DRV_LOAD\n");
|
||||||
|
the_device = NULL;
|
||||||
|
return 1L;
|
||||||
|
|
||||||
|
case DRV_FREE :
|
||||||
|
DPRINT("DRV_FREE\n");
|
||||||
|
return 1L;
|
||||||
|
|
||||||
|
case DRV_OPEN :
|
||||||
|
DPRINT("DRV_OPEN\n");
|
||||||
|
InitializeCriticalSection(&device_lock);
|
||||||
|
return 1L;
|
||||||
|
|
||||||
|
case DRV_CLOSE :
|
||||||
|
DPRINT("DRV_CLOSE\n");
|
||||||
|
return 1L;
|
||||||
|
|
||||||
|
case DRV_ENABLE :
|
||||||
|
DPRINT("DRV_ENABLE\n");
|
||||||
|
return 1L;
|
||||||
|
|
||||||
|
case DRV_DISABLE :
|
||||||
|
DPRINT("DRV_DISABLE\n");
|
||||||
|
return 1L;
|
||||||
|
|
||||||
|
/*
|
||||||
|
We don't provide configuration capabilities. This used to be
|
||||||
|
for things like I/O port, IRQ, DMA settings, etc.
|
||||||
|
*/
|
||||||
|
|
||||||
|
case DRV_QUERYCONFIGURE :
|
||||||
|
DPRINT("DRV_QUERYCONFIGURE\n");
|
||||||
|
return 0L;
|
||||||
|
|
||||||
|
case DRV_CONFIGURE :
|
||||||
|
DPRINT("DRV_CONFIGURE\n");
|
||||||
|
return 0L;
|
||||||
|
|
||||||
|
case DRV_INSTALL :
|
||||||
|
DPRINT("DRV_INSTALL\n");
|
||||||
|
return DRVCNF_RESTART;
|
||||||
|
};
|
||||||
|
|
||||||
|
DPRINT("???\n");
|
||||||
|
|
||||||
|
return DefDriverProc(driver_id,
|
||||||
|
driver_handle,
|
||||||
|
message,
|
||||||
|
parameter1,
|
||||||
|
parameter2);
|
||||||
|
}
|
12
reactos/dll/win32/beepmidi/beepmidi.def
Normal file
12
reactos/dll/win32/beepmidi/beepmidi.def
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
; beepmidi.def
|
||||||
|
;
|
||||||
|
; BeepMidi driver by Andrew Greenwood
|
||||||
|
;
|
||||||
|
; For ReactOS Operating System
|
||||||
|
;
|
||||||
|
|
||||||
|
LIBRARY beepmidi.dll
|
||||||
|
|
||||||
|
EXPORTS
|
||||||
|
DriverProc@20
|
||||||
|
modMessage@20
|
12
reactos/dll/win32/beepmidi/beepmidi.rbuild
Normal file
12
reactos/dll/win32/beepmidi/beepmidi.rbuild
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
<module name="beepmidi" type="win32dll" installbase="system32" installname="beepmidi.dll">
|
||||||
|
<importlibrary definition="beepmidi.def" />
|
||||||
|
<include base="beepmidi">.</include>
|
||||||
|
<define name="__USE_W32API" />
|
||||||
|
<define name="UNICODE" />
|
||||||
|
<define name="_UNICODE" />
|
||||||
|
<library>ntdll</library>
|
||||||
|
<library>kernel32</library>
|
||||||
|
<library>user32</library>
|
||||||
|
<library>winmm</library>
|
||||||
|
<file>beepmidi.c</file>
|
||||||
|
</module>
|
51
reactos/dll/win32/beepmidi/readme.txt
Normal file
51
reactos/dll/win32/beepmidi/readme.txt
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
BEEPMIDI :: BEEP.SYS MIDI DRIVER
|
||||||
|
(c) Andrew Greenwood, 2007.
|
||||||
|
|
||||||
|
http://www.silverblade.co.uk
|
||||||
|
|
||||||
|
Released as open-source software. You may copy, re-distribute and modify
|
||||||
|
this software, provided this copyright notice remains intact.
|
||||||
|
|
||||||
|
WHAT'S THIS ?
|
||||||
|
BeepMidi is a MME MIDI driver for NT-compatible operating systems,
|
||||||
|
which uses BEEP.SYS (the kernel-mode PC speaker driver) to play
|
||||||
|
MIDI data. It installs as a standard MIDI output device and can even
|
||||||
|
be selected as your default MIDI output device. The fundamental
|
||||||
|
code for interacting with BEEP.SYS was taken from ReactOS' kernel32
|
||||||
|
module.
|
||||||
|
|
||||||
|
WHY WAS THIS WRITTEN ?
|
||||||
|
Primarily for educational reasons - in the process, I've learned more
|
||||||
|
about the driver side of the MME API and how to interact with kernel
|
||||||
|
device drivers. It aids as a good starting point from which to
|
||||||
|
move on to bigger and better things :)
|
||||||
|
|
||||||
|
HOW TO INSTALL :
|
||||||
|
Copy the file to C:\WINDOWS\SYSTEM32\BEEPMIDI.DLL
|
||||||
|
|
||||||
|
Go into HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\
|
||||||
|
Drivers32 in RegEdit and look for the "midi" entries on the right hand
|
||||||
|
side. Find the highest numbered one (eg: midi1) and create a new STRING
|
||||||
|
value. Give it another midi name, but one above the current highest
|
||||||
|
entry present (eg: midi2.)
|
||||||
|
|
||||||
|
You'll now see a "PC Speaker" entry in Sound & Audio Devices.
|
||||||
|
|
||||||
|
TWEAKING:
|
||||||
|
See the comments toward the top of beepmidi.c for tweakable driver
|
||||||
|
parameters. These can only be adjusted in the source code at present.
|
||||||
|
|
||||||
|
FEATURES :
|
||||||
|
* Supports note-on and note-off messages on channels 1-9 and 11-16
|
||||||
|
(channel 10 is rhythm, which is not supported.)
|
||||||
|
* Fake polyphony (actually just arpeggiates playing notes!)
|
||||||
|
* Threaded design for continuous playback (optional.)
|
||||||
|
|
||||||
|
ROOM FOR IMPROVEMENT :
|
||||||
|
* Pitch bend is not supported
|
||||||
|
* Velocity could determine timeslice
|
||||||
|
* Should wait for timeslice to complete before adding/removing notes
|
||||||
|
* Would be nice to allow configuration of polyphony etc. via Control Panel
|
||||||
|
|
||||||
|
BUGS :
|
||||||
|
* Crashes when used with Windows Media Player (mplayer2 is fine though)
|
Loading…
Reference in a new issue