shithub: choc

Download patch

ref: af06c1fef012afcb4e3c5d8b889e92ab10b3705d
parent: 056cdff7cef7dd545fe045532188c77533c6b5f9
author: Alex Mayfield <[email protected]>
date: Fri Feb 17 19:40:02 EST 2017

Second attempt, this time with pipes

Unfortunately, as it turns out, SDL_net does not allow you to create a
TCP listening socket that only listens on localhost.  This has the
unfortunate side-effect of showing a firewall message, which is not
desirable.

Instead, I use pipes to communicate with the subprocess over stdin.
This approach actually works - the current state of the code is if you
launch Chocolate Doom, it will communicate with the subprocess.

This is only the beginning.  Although communication is occurring, I
don't appear to actually be able to hear anything, and Chocolate Doom
crashes as soon as there is an attempt to switch songs.

--- a/midiproc/main.c
+++ b/midiproc/main.c
@@ -1,5 +1,6 @@
 //
 // Copyright(C) 2012 James Haley
+// Copyright(C) 2017 Alex Mayfield
 //
 // This program is free software; you can redistribute it and/or
 // modify it under the terms of the GNU General Public License
@@ -23,73 +24,56 @@
 // Seriously, how did they screw up something so fundamental?
 //
 
+#define WIN32_LEAN_AND_MEAN
 #include <windows.h>
 #include <stdlib.h>
+
 #include "SDL.h"
 #include "SDL_mixer.h"
-#include "midiproc.h"
 
+#include "buffer.h"
+
+#include "config.h"
+#include "doomtype.h"
+#include "net_defs.h"
+
+static HANDLE    midi_process_in;  // Standard In.
+static HANDLE    midi_process_out; // Standard Out.
+static buffer_t *midi_buffer;      // Data from client.
+
 // Currently playing music track
 static Mix_Music *music = NULL;
-static SDL_RWops *rw    = NULL;
+static char *filename = NULL;
 
 static void UnregisterSong();
 
 //=============================================================================
 //
-// RPC Memory Management
-//
-
-void __RPC_FAR * __RPC_USER midl_user_allocate(size_t size)
-{
-   return malloc(size);
-}
-
-void __RPC_USER midl_user_free(void __RPC_FAR *p)
-{
-   free(p);
-}
-
-//=============================================================================
-//
 // SDL_mixer Interface
 //
 
 //
-// InitSDL
-//
-// Start up SDL and SDL_mixer.
-//
-static bool InitSDL()
-{
-   if(SDL_Init(SDL_INIT_AUDIO) == -1)
-      return false;
-
-   if(Mix_OpenAudio(44100, MIX_DEFAULT_FORMAT, 2, 2048) < 0)
-      return false;
-
-   return true;
-}
-
-//
 // RegisterSong
 //
-static void RegisterSong(void *data, size_t size)
+static void RegisterSong(char* filename)
 {
-   if(music)
-      UnregisterSong();
+    if (music)
+    {
+        UnregisterSong();
+    }
 
-   rw    = SDL_RWFromMem(data, size);
-   music = Mix_LoadMUS_RW(rw);
+    music = Mix_LoadMUS(filename);
 }
 
 //
 // StartSong
 //
-static void StartSong(bool loop)
+static void StartSong(boolean loop)
 {
-   if(music)
-      Mix_PlayMusic(music, loop ? -1 : 0);
+    if (music)
+    {
+        Mix_PlayMusic(music, loop ? -1 : 0);
+    }
 }
 
 //
@@ -97,7 +81,7 @@
 //
 static void SetVolume(int volume)
 {
-   Mix_VolumeMusic((volume * 128) / 15);
+    Mix_VolumeMusic((volume * 128) / 15);
 }
 
 static int paused_midi_volume;
@@ -107,8 +91,8 @@
 //
 static void PauseSong()
 {
-   paused_midi_volume = Mix_VolumeMusic(-1);
-   Mix_VolumeMusic(0);
+    paused_midi_volume = Mix_VolumeMusic(-1);
+    Mix_VolumeMusic(0);
 }
 
 //
@@ -116,7 +100,7 @@
 //
 static void ResumeSong()
 {
-   Mix_VolumeMusic(paused_midi_volume);
+    Mix_VolumeMusic(paused_midi_volume);
 }
 
 //
@@ -124,8 +108,10 @@
 //
 static void StopSong()
 {
-   if(music)
-      Mix_HaltMusic();
+    if (music)
+    {
+        Mix_HaltMusic();
+    }
 }
 
 //
@@ -133,13 +119,17 @@
 //
 static void UnregisterSong()
 {
-   if(!music)
-      return;
+    if (!music)
+    {
+        return;
+    }
 
-   StopSong();
-   Mix_FreeMusic(music);
-   rw    = NULL;
-   music = NULL;
+    StopSong();
+    Mix_FreeMusic(music);
+    free(filename);
+
+    filename = NULL;
+    music = NULL;
 }
 
 //
@@ -147,208 +137,277 @@
 //
 static void ShutdownSDL()
 {
-   UnregisterSong();
-   Mix_CloseAudio();
-   SDL_Quit();
+    UnregisterSong();
+    Mix_CloseAudio();
+    SDL_Quit();
 }
 
 //=============================================================================
 //
-// Song Buffer
-//
-// The MIDI program will be transmitted by the client across RPC in fixed-size
-// chunks until all data has been transmitted.
-//
-
-typedef unsigned char midibyte;
-
-class SongBuffer
-{
-protected:
-   midibyte *buffer;    // accumulated input
-   size_t    size;      // size of input
-   size_t    allocated; // amount of memory allocated (>= size)
-
-   static const int defaultSize = 128*1024; // 128 KB
-
-public:
-   // Constructor
-   // Start out with an empty 128 KB buffer.
-   SongBuffer()
-   {
-      buffer = static_cast<midibyte *>(calloc(1, defaultSize));
-      size = 0;
-      allocated = defaultSize;
-   }
-
-   // Destructor.
-   // Release the buffer.
-   ~SongBuffer()
-   {
-      if(buffer)
-      {
-         free(buffer);
-         buffer = NULL;
-         size = allocated = 0;
-      }
-   }
-
-   //
-   // addChunk
-   //
-   // Add a chunk of MIDI data to the buffer.
-   //
-   void addChunk(midibyte *data, size_t newsize)
-   {
-      if(size + newsize > allocated)
-      {
-         allocated += newsize * 2;
-         buffer = static_cast<midibyte *>(realloc(buffer, allocated));
-      }
-
-      memcpy(buffer + size, data, newsize);
-      size += newsize;
-   }
-
-   // Accessors
-
-   midibyte *getBuffer() const { return buffer; }
-   size_t    getSize()   const { return size;   }
-};
-
-static SongBuffer *song;
-
-//=============================================================================
-//
 // RPC Server Interface
 //
 
 //
-// MidiRPC_PrepareNewSong
+// MidiPipe_PrepareNewSong
 //
 // Prepare the engine to receive new song data from the RPC client.
 //
-void MidiRPC_PrepareNewSong()
+boolean MidiPipe_PrepareNewSong()
 {
-   // Stop anything currently playing and free it.
-   UnregisterSong();
+    // Stop anything currently playing and free it.
+    UnregisterSong();
 
-   // free any previous song buffer
-   delete song;
-
-   // prep new song buffer
-   song = new SongBuffer();
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_AddChunk
+// MidiPipe_AddChunk
 //
-// Add a chunk of data to the song.
+// Set the filename of the song.
 //
-void MidiRPC_AddChunk(unsigned int count, byte *pBuf)
+boolean MidiPipe_SetFilename(buffer_reader_t *reader)
 {
-   song->addChunk(pBuf, static_cast<size_t>(count));
+    free(filename);
+    filename = NULL;
+
+    char* file = Reader_ReadString(reader);
+    if (file == NULL)
+    {
+        return false;
+    }
+
+    int size = Reader_BytesRead(reader) - 2;
+    if (size <= 0)
+    {
+        return false;
+    }
+
+    filename = malloc(size);
+    if (filename == NULL)
+    {
+        return false;
+    }
+
+    memcpy(filename, file, size);
+
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_PlaySong
+// MidiPipe_PlaySong
 //
 // Start playing the song.
 //
-void MidiRPC_PlaySong(boolean looping)
+boolean MidiPipe_PlaySong(buffer_reader_t *reader)
 {
-   RegisterSong(song->getBuffer(), song->getSize());
-   StartSong(!!looping);
+    uint8_t looping;
+
+    if (!Reader_ReadInt8(reader, &looping))
+    {
+        return false;
+    }
+
+    RegisterSong(filename);
+    StartSong((boolean)looping);
+
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_StopSong
+// MidiPipe_StopSong
 //
 // Stop the song.
 //
-void MidiRPC_StopSong()
+boolean MidiPipe_StopSong()
 {
-   StopSong();
+    StopSong();
+
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_ChangeVolume
+// MidiPipe_ChangeVolume
 //
 // Set playback volume level.
 //
-void MidiRPC_ChangeVolume(int volume)
+boolean MidiPipe_ChangeVolume(buffer_reader_t *reader)
 {
-   SetVolume(volume);
+    int volume;
+
+    if (!Reader_ReadInt32(reader, &volume))
+    {
+        return false;
+    }
+
+    SetVolume(volume);
+
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_PauseSong
+// MidiPipe_PauseSong
 //
 // Pause the song.
 //
-void MidiRPC_PauseSong()
+boolean MidiPipe_PauseSong()
 {
-   PauseSong();
+    PauseSong();
+
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_ResumeSong
+// MidiPipe_ResumeSong
 //
 // Resume after pausing.
 //
-void MidiRPC_ResumeSong()
+boolean MidiPipe_ResumeSong()
 {
-   ResumeSong();
+    ResumeSong();
+
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
 //
-// MidiRPC_StopServer
+// MidiPipe_StopServer
 //
 // Stops the RPC server so the program can shutdown.
 //
-void MidiRPC_StopServer()
+boolean MidiPipe_StopServer()
 {
-   // Local shutdown tasks
-   ShutdownSDL();
-   delete song;
-   song = NULL;
+    // Local shutdown tasks
+    ShutdownSDL();
+    free(filename);
+    filename = NULL;
 
-   // Stop RPC server
-   RpcMgmtStopServerListening(NULL);
+    fprintf(stderr, "%s\n", __FUNCTION__);
+    return true;
 }
 
+//=============================================================================
 //
-// RPC Server Init
+// Server Implementation
 //
-static bool MidiRPC_InitServer()
+
+boolean ParseCommand(buffer_reader_t *reader, uint16_t command)
 {
-   RPC_STATUS status;
+    switch (command)
+    {
+    case NET_MIDISOCKET_PACKET_TYPE_PREPARE_NEW_SONG:
+        return MidiPipe_PrepareNewSong();
+    case NET_MIDISOCKET_PACKET_TYPE_SET_FILENAME:
+        return MidiPipe_SetFilename(reader);
+    case NET_MIDISOCKET_PACKET_TYPE_PLAY_SONG:
+        return MidiPipe_PlaySong(reader);
+    case NET_MIDISOCKET_PACKET_TYPE_STOP_SONG:
+        return MidiPipe_StopSong();
+    case NET_MIDISOCKET_PACKET_TYPE_CHANGE_VOLUME:
+        return MidiPipe_ChangeVolume(reader);
+    case NET_MIDISOCKET_PACKET_TYPE_PAUSE_SONG:
+        return MidiPipe_PauseSong();
+    case NET_MIDISOCKET_PACKET_TYPE_RESUME_SONG:
+        return MidiPipe_ResumeSong();
+    case NET_MIDISOCKET_PACKET_TYPE_STOP_SERVER:
+        return MidiPipe_StopServer();
+    default:
+        return false;
+    }
+}
 
-   // Initialize RPC protocol
-   status = 
-      RpcServerUseProtseqEp
-      (
-         (RPC_CSTR)("ncalrpc"),
-         RPC_C_PROTSEQ_MAX_REQS_DEFAULT,
-         (RPC_CSTR)("2d4dc2f9-ce90-4080-8a00-1cb819086970"),
-         NULL
-      );
+//
+// Server packet parser
+//
+boolean ParseMessage(buffer_t *buf)
+{
+    uint16_t command;
+    buffer_reader_t *reader = NewReader(buf);
 
-   if(status)
-      return false;
+    // Attempt to read a command out of the buffer.
+    if (!Reader_ReadInt16(reader, &command))
+    {
+        goto fail;
+    }
 
-   // Register server
-   status = RpcServerRegisterIf(MidiRPC_v1_0_s_ifspec, NULL, NULL);
+    // Attempt to parse a complete message.
+    if (!ParseCommand(reader, command))
+    {
+        goto fail;
+    }
 
-   if(status)
-      return false;
+    // We parsed a complete message!  We can now safely shift
+    // the prior message off the front of the buffer.
+    int bytes_read = Reader_BytesRead(reader);
+    DeleteReader(reader);
+    Buffer_Shift(buf, bytes_read);
 
-   // Start listening
-   status = RpcServerListen(1, RPC_C_LISTEN_MAX_CALLS_DEFAULT, FALSE);
+    return true;
 
-   return !status;
+fail:
+    // We did not read a complete packet.  Delete our reader and try again
+    // with more data.
+    DeleteReader(reader);
+    return false;
 }
 
+boolean ListenForever()
+{
+    BOOL wok = FALSE;
+    CHAR pipe_buffer[8192];
+    DWORD pipe_buffer_read = 0;
+
+    boolean ok = false;
+    buffer_t *buffer = NewBuffer();
+
+    fprintf(stderr, "%s\n", "In theory we should be reading...");
+    for (;;)
+    {
+        // Wait until we see some data on the pipe.
+        wok = PeekNamedPipe(midi_process_in, NULL, 0, NULL,
+            &pipe_buffer_read, NULL);
+        if (!wok)
+        {
+            return false;
+        }
+        else if (pipe_buffer_read == 0)
+        {
+            SDL_Delay(1);
+            continue;
+        }
+
+        // Read data off the pipe and add it to the buffer.
+        fprintf(stderr, "%s\n", "ReadFile");
+        wok = ReadFile(midi_process_in, pipe_buffer, sizeof(pipe_buffer),
+            &pipe_buffer_read, NULL);
+        if (!wok)
+        {
+            return false;
+        }
+
+        fprintf(stderr, "%s\n", "Buffer_Push");
+        ok = Buffer_Push(buffer, pipe_buffer, pipe_buffer_read);
+        if (!ok)
+        {
+            return false;
+        }
+
+        fprintf(stderr, "%s\n", "ParseMessage");
+        do
+        {
+            // Read messages off the buffer until we can't anymore.
+            ok = ParseMessage(buffer);
+        } while (ok);
+    }
+
+    return false;
+}
+
 //=============================================================================
 //
 // Main Program
@@ -355,23 +414,92 @@
 //
 
 //
-// WinMain
+// InitSDL
 //
-// Application entry point.
+// Start up SDL and SDL_mixer.
 //
-int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, 
-                     LPSTR lpCmdLine, int nCmdShow)
+boolean InitSDL()
 {
-   // Initialize SDL
-   if(!InitSDL())
-      return -1;
+    if (SDL_Init(SDL_INIT_AUDIO) == -1)
+    {
+        return false;
+    }
 
-   // Initialize RPC Server
-   if(!MidiRPC_InitServer())
-      return -1;
+    if (Mix_OpenAudio(44100, MIX_DEFAULT_FORMAT, 2, 2048) < 0)
+    {
+        return false;
+    }
 
-   return 0;
+    return true;
 }
 
-// EOF
+//
+// InitPipes
+//
+// Ensure that we can communicate.
+//
+boolean InitPipes()
+{
+    midi_process_in = GetStdHandle(STD_INPUT_HANDLE);
+    if (midi_process_in == INVALID_HANDLE_VALUE)
+    {
+        return false;
+    }
+
+    midi_process_out = GetStdHandle(STD_OUTPUT_HANDLE);
+    if (midi_process_out == INVALID_HANDLE_VALUE)
+    {
+        return false;
+    }
+
+    return true;
+}
+
+//
+// main
+//
+// Application entry point.
+//
+int main(int argc, char *argv[])
+{
+    // Make sure we're not launching this process by itself.
+    if (argc < 2)
+    {
+        MessageBox(NULL, TEXT("This program is tasked with playing Native ")
+            TEXT("MIDI music, and is intended to be launched by ")
+            TEXT(PACKAGE_NAME) TEXT("."),
+            TEXT(PACKAGE_STRING), MB_OK | MB_ICONASTERISK);
+
+        return EXIT_FAILURE;
+    }
+
+    // Make sure our Choccolate Doom and midiproc version are lined up.
+    if (strcmp(PACKAGE_STRING, argv[1]) != 0)
+    {
+        MessageBox(NULL, TEXT("It appears that the version of ")
+            TEXT(PACKAGE_NAME) TEXT(" and ") TEXT(PROGRAM_PREFIX)
+            TEXT("midiproc are out of sync.  Please reinstall ")
+            TEXT(PACKAGE_NAME) TEXT("."),
+            TEXT(PACKAGE_STRING), MB_OK | MB_ICONASTERISK);
+
+        return EXIT_FAILURE;
+    }
+
+    if (!InitPipes())
+    {
+        return EXIT_FAILURE;
+    }
+
+    if (!InitSDL())
+    {
+        return EXIT_FAILURE;
+    }
+
+    if (!ListenForever())
+    {
+        return EXIT_FAILURE;
+    }
+
+    return EXIT_SUCCESS;
+}
 
--- /dev/null
+++ b/src/i_midipipe.c
@@ -1,0 +1,420 @@
+//
+// Copyright(C) 2013 James Haley et al.
+// Copyright(C) 2017 Alex Mayfield
+//
+// This program is free software; you can redistribute it and/or
+// modify it under the terms of the GNU General Public License
+// as published by the Free Software Foundation; either version 2
+// of the License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+//
+// DESCRIPTION:
+//     Client Interface to RPC Midi Server
+//
+
+#if _WIN32
+
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+#include <SDL_net.h>
+
+#include "i_midipipe.h"
+
+#include "config.h"
+#include "m_misc.h"
+#include "net_packet.h"
+
+#if defined(_DEBUG)
+#define DEBUGOUT(s) puts(s)
+#else
+#define DEBUGOUT(s)
+#endif
+
+//=============================================================================
+//
+// Data
+//
+
+static HANDLE  midi_process_in_reader;  // Input stream for midi process.
+static HANDLE  midi_process_in_writer;
+static HANDLE  midi_process_out_reader; // Output stream for midi process.
+static HANDLE  midi_process_out_writer;
+
+static boolean server_init = false; // if true, server was started
+static boolean client_init = false; // if true, client was bound
+
+//=============================================================================
+//
+// RPC Wrappers
+//
+
+//
+// CHECK_RPC_STATUS
+//
+// If either server or client initialization failed, we don't try to make any
+// RPC calls.
+//
+#define CHECK_RPC_STATUS() \
+    if(!server_init) \
+        return false
+
+#define MIDIRPC_MAXTRIES 50 // This number * 10 is the amount of time you can try to wait for.
+
+static boolean I_MidiPipeWrite(void *data, int len)
+{
+    DWORD written;
+    if (WriteFile(midi_process_in_writer, data, len, &written, NULL))
+    {
+        return true;
+    }
+    else
+    {
+        return false;
+    }
+}
+
+static boolean I_MidiPipeWaitForServer()
+{
+    int tries = 0;
+    while(false) // TODO: Is there some way to tell if the server is listening?
+    {
+        I_Sleep(10);
+        if (++tries >= MIDIRPC_MAXTRIES)
+        {
+            return false;
+        }
+    }
+    return true;
+}
+
+//
+// I_MidiPipeRegisterSong
+//
+// Prepare the RPC MIDI engine to receive new song data, and transmit the song
+// filename to the server process.
+//
+boolean I_MidiPipeRegisterSong(const char *filename)
+{
+    BOOL wok;
+    net_packet_t *packet;
+
+    CHECK_RPC_STATUS();
+
+    packet = NET_NewPacket(64);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_PREPARE_NEW_SONG);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_SET_FILENAME);
+    NET_WriteString(packet, filename);
+    wok = WriteFile(midi_process_in_writer, packet->data, packet->len,
+        NULL, NULL);
+    NET_FreePacket(packet);
+
+    if (!wok)
+    {
+        DEBUGOUT("I_MidiPipeRegisterSong failed");
+        return false;
+    }
+
+    DEBUGOUT("I_MidiPipeRegisterSong succeeded");
+    return true;
+}
+
+//
+// I_MidiPipePlaySong
+//
+// Tell the RPC server to start playing a song.
+//
+boolean I_MidiPipePlaySong(boolean looping)
+{
+    BOOL wok;
+    net_packet_t *packet;
+
+    CHECK_RPC_STATUS();
+
+    packet = NET_NewPacket(3);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_PLAY_SONG);
+    NET_WriteInt8(packet, looping);
+    wok = WriteFile(midi_process_in_writer, packet->data, packet->len,
+        NULL, NULL);
+    NET_FreePacket(packet);
+
+    if (!wok)
+    {
+        DEBUGOUT("I_MidiPipePlaySong failed");
+        return false;
+    }
+
+    DEBUGOUT("I_MidiPipePlaySong succeeded");
+    return true;
+}
+
+// 
+// I_MidiPipeStopSong
+//
+// Tell the RPC server to stop any currently playing song.
+//
+boolean I_MidiPipeStopSong()
+{
+    BOOL wok;
+    net_packet_t *packet;
+
+    CHECK_RPC_STATUS();
+
+    packet = NET_NewPacket(2);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_STOP_SONG);
+    wok = WriteFile(midi_process_in_writer, packet->data, packet->len,
+        NULL, NULL);
+    NET_FreePacket(packet);
+
+    if (!wok)
+    {
+        DEBUGOUT("I_MidiPipeStopSong failed");
+        return false;
+    }
+
+    DEBUGOUT("I_MidiPipeStopSong succeeded");
+    return true;
+}
+
+//
+// I_MidiPipeSetVolume
+//
+// Change the volume level of music played by the RPC midi server.
+//
+boolean I_MidiPipeSetVolume(int volume)
+{
+    BOOL wok;
+    net_packet_t *packet;
+
+    CHECK_RPC_STATUS();
+
+    packet = NET_NewPacket(6);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_CHANGE_VOLUME);
+    NET_WriteInt32(packet, volume);
+    wok = WriteFile(midi_process_in_writer, packet->data, packet->len,
+        NULL, NULL);
+    NET_FreePacket(packet);
+
+    if (!wok)
+    {
+        DEBUGOUT("I_MidiPipeSetVolume failed");
+        return false;
+    }
+
+    DEBUGOUT("I_MidiPipeSetVolume succeeded");
+    return true;
+}
+
+//
+// I_MidiPipePauseSong
+//
+// Pause the music being played by the server. In actuality, due to SDL_mixer
+// limitations, this just temporarily sets the volume to zero.
+//
+boolean I_MidiPipePauseSong()
+{
+    BOOL wok;
+    net_packet_t *packet;
+
+    CHECK_RPC_STATUS();
+
+    packet = NET_NewPacket(2);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_PAUSE_SONG);
+    wok = WriteFile(midi_process_in_writer, packet->data, packet->len,
+        NULL, NULL);
+    NET_FreePacket(packet);
+
+    if (!wok)
+    {
+        DEBUGOUT("I_MidiPipePauseSong failed");
+        return false;
+    }
+
+    DEBUGOUT("I_MidiPipePauseSong succeeded");
+    return true;
+}
+
+//
+// I_MidiPipeResumeSong
+//
+// Resume a song after having paused it.
+//
+boolean I_MidiPipeResumeSong()
+{
+    BOOL wok;
+    net_packet_t *packet;
+
+    CHECK_RPC_STATUS();
+
+    packet = NET_NewPacket(2);
+    NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_RESUME_SONG);
+    wok = WriteFile(midi_process_in_writer, packet->data, packet->len,
+        NULL, NULL);
+    NET_FreePacket(packet);
+
+    if (!wok)
+    {
+        DEBUGOUT("I_MidiPipeResumeSong failed");
+        return false;
+    }
+
+    DEBUGOUT("I_MidiPipeResumeSong succeeded");
+   return true;
+}
+
+//=============================================================================
+//
+// Public Interface
+//
+
+//
+// I_MidiPipeInitServer
+//
+// Start up the MIDI server.
+//
+boolean I_MidiPipeInitServer()
+{
+    struct stat sbuf;
+    char filename[MAX_PATH+1];
+
+    memset(filename, 0, sizeof(filename));
+    size_t filename_len = GetModuleFileName(NULL, filename, MAX_PATH);
+
+    // Remove filespec
+    // TODO: Move this to m_misc
+    char *fp = &filename[filename_len];
+    while (filename <= fp && *fp != DIR_SEPARATOR)
+    {
+        fp--;
+    }
+    *(fp + 1) = '\0';
+    char* module = M_StringJoin(filename, PROGRAM_PREFIX "midiproc.exe", NULL);
+    char* cmdline = M_StringJoin(module, " \"" PACKAGE_STRING "\"", NULL);
+    DEBUGOUT(module);
+    DEBUGOUT(cmdline);
+
+    // Look for executable file
+    if(stat(module, &sbuf))
+    {
+        DEBUGOUT("Could not find midiproc");
+        return false;
+    }
+
+    // Set up pipes
+    SECURITY_ATTRIBUTES sec_attrs;
+    memset(&sec_attrs, 0, sizeof(SECURITY_ATTRIBUTES));
+    sec_attrs.nLength = sizeof(SECURITY_ATTRIBUTES);
+    sec_attrs.bInheritHandle = TRUE;
+    sec_attrs.lpSecurityDescriptor = NULL;
+
+    if (!CreatePipe(&midi_process_in_reader, &midi_process_in_writer, &sec_attrs, 0))
+    {
+        DEBUGOUT("Could not initialize midiproc stdin");
+        return false;
+    }
+
+    if (!SetHandleInformation(midi_process_in_writer, HANDLE_FLAG_INHERIT, 0))
+    {
+        DEBUGOUT("Could not disinherit midiproc stdin");
+        return false;
+    }
+
+    if (!CreatePipe(&midi_process_out_reader, &midi_process_out_writer, &sec_attrs, 0))
+    {
+        DEBUGOUT("Could not initialize midiproc stdout/stderr");
+        return false;
+    }
+
+    if (!SetHandleInformation(midi_process_out_reader, HANDLE_FLAG_INHERIT, 0))
+    {
+        DEBUGOUT("Could not disinherit midiproc stdin");
+        return false;
+    }
+
+    // Launch the subprocess
+    PROCESS_INFORMATION proc_info;
+    memset(&proc_info, 0, sizeof(proc_info));
+
+    STARTUPINFO startup_info;
+    memset(&startup_info, 0, sizeof(startup_info));
+    startup_info.cb = sizeof(startup_info);
+    startup_info.hStdInput = midi_process_in_reader;
+    startup_info.hStdOutput = midi_process_out_writer;
+    startup_info.dwFlags = STARTF_USESTDHANDLES;
+
+    boolean ok = CreateProcess(TEXT(module), TEXT(cmdline), NULL, NULL, TRUE,
+        CREATE_NEW_CONSOLE, NULL, NULL, &startup_info, &proc_info);
+
+    if (ok)
+    {
+        DEBUGOUT("midiproc started");
+        server_init = true;
+    }
+    else
+    {
+        DEBUGOUT("failed to start midiproc");
+    }
+
+    return ok;
+}
+
+//
+// I_MidiPipeInitClient
+//
+// Ensure that we can actually communicate with the subprocess.
+//
+boolean I_MidiPipeInitClient()
+{
+    client_init = true;
+    return true;
+}
+
+//
+// I_MidiPipeClientShutDown
+//
+// Shutdown the RPC Client
+//
+/* void I_MidiPipeClientShutDown()
+{
+    // stop the server
+    if(server_init)
+    {
+        net_packet_t *packet;
+        packet = NET_NewPacket(2);
+        NET_WriteInt16(packet, NET_MIDIPIPE_PACKET_TYPE_STOP_SERVER);
+        int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
+        NET_FreePacket(packet);
+        if (len < packet->len)
+        {
+            DEBUGOUT("Problem encountered when stopping RPC server");
+        }
+
+        server_init = false;
+    }
+
+    if (midi_socket)
+    {
+        SDLNet_TCP_Close(midi_socket);
+        midi_socket = NULL;
+    }
+
+    client_init = false;
+} */
+
+//
+// I_MidiPipeReady
+//
+// Returns true if both server and client initialized successfully.
+//
+boolean I_MidiPipeReady()
+{
+    CHECK_RPC_STATUS();
+
+    return true;
+}
+
+#endif
+
--- /dev/null
+++ b/src/i_midipipe.h
@@ -1,0 +1,41 @@
+//
+// Copyright(C) 2013 James Haley et al.
+// Copyright(C) 2017 Alex Mayfield
+//
+// This program is free software; you can redistribute it and/or
+// modify it under the terms of the GNU General Public License
+// as published by the Free Software Foundation; either version 2
+// of the License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+//
+// DESCRIPTION:
+//     Client Interface to RPC Midi Server
+//
+
+#ifndef __I_MIDISOCKET__
+#define __I_MIDISOCKET__
+
+#if _WIN32
+
+#include "doomtype.h"
+
+boolean I_MidiSocketInitServer();
+boolean I_MidiSocketInitClient();
+void I_MidiSocketClientShutDown();
+boolean I_MidiSocketReady();
+
+boolean I_MidiSocketRegisterSong(const char *filename);
+boolean I_MidiSocketPlaySong(boolean looping);
+boolean I_MidiSocketStopSong();
+boolean I_MidiSocketSetVolume(int volume);
+boolean I_MidiSocketPauseSong();
+boolean I_MidiSocketResumeSong();
+
+#endif
+
+#endif
+
--- a/src/i_midisocket.c
+++ /dev/null
@@ -1,391 +1,0 @@
-//
-// Copyright(C) 2013 James Haley et al.
-// Copyright(C) 2017 Alex Mayfield
-//
-// This program is free software; you can redistribute it and/or
-// modify it under the terms of the GNU General Public License
-// as published by the Free Software Foundation; either version 2
-// of the License, or (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-// GNU General Public License for more details.
-//
-// DESCRIPTION:
-//     Client Interface to RPC Midi Server
-//
-
-#if _WIN32
-
-#define WIN32_LEAN_AND_MEAN
-#include <windows.h>
-#include <SDL_net.h>
-
-#include "i_midisocket.h"
-
-#include "m_misc.h"
-#include "net_packet.h"
-
-#if defined(_DEBUG)
-#define DEBUGOUT(s) puts(s)
-#else
-#define DEBUGOUT(s)
-#endif
-
-typedef enum {
-	NET_MIDISOCKET_PACKET_TYPE_PREPARE_NEW_SONG,
-    NET_MIDISOCKET_PACKET_TYPE_REGISTER_SONG,
-	NET_MIDISOCKET_PACKET_TYPE_PLAY_SONG,
-	NET_MIDISOCKET_PACKET_TYPE_STOP_SONG,
-	NET_MIDISOCKET_PACKET_TYPE_CHANGE_VOLUME,
-	NET_MIDISOCKET_PACKET_TYPE_PAUSE_SONG,
-	NET_MIDISOCKET_PACKET_TYPE_RESUME_SONG,
-	NET_MIDISOCKET_PACKET_TYPE_STOP_SERVER
-} net_midisocket_packet_type_t;
-
-//=============================================================================
-//
-// Data
-//
-
-static TCPsocket midi_socket;       // Client socket
-static boolean server_init = false; // if true, server was started
-static boolean client_init = false; // if true, client was bound
-
-// server process information
-static STARTUPINFO         si;
-static PROCESS_INFORMATION pi;
-
-//=============================================================================
-//
-// RPC Wrappers
-//
-
-//
-// CHECK_RPC_STATUS
-//
-// If either server or client initialization failed, we don't try to make any
-// RPC calls.
-//
-#define CHECK_RPC_STATUS() \
-    if(!server_init || !client_init) \
-        return false
-
-#define MIDIRPC_MAXTRIES 50 // This number * 10 is the amount of time you can try to wait for.
-
-static boolean I_MidiSocketWaitForServer()
-{
-    int tries = 0;
-    while(false) // TODO: Is there some way to tell if the server is listening?
-    {
-        I_Sleep(10);
-        if (++tries >= MIDIRPC_MAXTRIES)
-        {
-            return false;
-        }
-    }
-    return true;
-}
-
-//
-// I_MidiSocketRegisterSong
-//
-// Prepare the RPC MIDI engine to receive new song data, and transmit the song
-// filename to the server process.
-//
-boolean I_MidiSocketRegisterSong(const char *filename)
-{
-    CHECK_RPC_STATUS();
-
-    net_packet_t *packet;
-    packet = NET_NewPacket(2);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_PREPARE_NEW_SONG);
-    int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    if (len < packet->len)
-    {
-        goto fail;
-    }
-    NET_FreePacket(packet);
-
-    packet = NET_NewPacket(64);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_REGISTER_SONG);
-    NET_WriteString(packet, filename);
-    len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    if (len < packet->len)
-    {
-        goto fail;
-    }
-    NET_FreePacket(packet);
-
-    DEBUGOUT("I_MidiSocketRegisterSong succeeded");
-    return true;
-
-fail:
-    NET_FreePacket(packet);
-
-    DEBUGOUT("I_MidiSocketRegisterSong failed");
-    return false;
-}
-
-//
-// I_MidiSocketPlaySong
-//
-// Tell the RPC server to start playing a song.
-//
-boolean I_MidiSocketPlaySong(boolean looping)
-{
-    CHECK_RPC_STATUS();
-
-    net_packet_t *packet;
-    packet = NET_NewPacket(3);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_PLAY_SONG);
-    NET_WriteInt8(packet, looping);
-    int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    NET_FreePacket(packet);
-    if (len < packet->len)
-    {
-        DEBUGOUT("I_MidiSocketPlaySong failed");
-        return false;
-    }
-
-    DEBUGOUT("I_MidiSocketPlaySong succeeded");
-    return true;
-}
-
-// 
-// I_MidiSocketStopSong
-//
-// Tell the RPC server to stop any currently playing song.
-//
-boolean I_MidiSocketStopSong()
-{
-    CHECK_RPC_STATUS();
-
-    net_packet_t *packet;
-    packet = NET_NewPacket(2);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_STOP_SONG);
-    int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    NET_FreePacket(packet);
-    if (len < packet->len)
-    {
-        DEBUGOUT("I_MidiSocketStopSong failed");
-        return false;
-    }
-
-    DEBUGOUT("I_MidiSocketStopSong succeeded");
-    return true;
-}
-
-//
-// I_MidiSocketSetVolume
-//
-// Change the volume level of music played by the RPC midi server.
-//
-boolean I_MidiSocketSetVolume(int volume)
-{
-    CHECK_RPC_STATUS();
-
-    net_packet_t *packet;
-    packet = NET_NewPacket(6);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_CHANGE_VOLUME);
-    NET_WriteInt32(packet, volume);
-    int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    NET_FreePacket(packet);
-    if (len < packet->len)
-    {
-        DEBUGOUT("I_MidiSocketSetVolume failed");
-        return false;
-    }
-
-    DEBUGOUT("I_MidiSocketSetVolume succeeded");
-    return true;
-}
-
-//
-// I_MidiSocketPauseSong
-//
-// Pause the music being played by the server. In actuality, due to SDL_mixer
-// limitations, this just temporarily sets the volume to zero.
-//
-boolean I_MidiSocketPauseSong()
-{
-    CHECK_RPC_STATUS();
-
-    net_packet_t *packet;
-    packet = NET_NewPacket(2);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_PAUSE_SONG);
-    int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    NET_FreePacket(packet);
-    if (len < packet->len)
-    {
-        DEBUGOUT("I_MidiSocketPauseSong failed");
-        return false;
-    }
-
-    DEBUGOUT("I_MidiSocketPauseSong succeeded");
-    return true;
-}
-
-//
-// I_MidiSocketResumeSong
-//
-// Resume a song after having paused it.
-//
-boolean I_MidiSocketResumeSong()
-{
-    CHECK_RPC_STATUS();
-
-    net_packet_t *packet;
-    packet = NET_NewPacket(2);
-    NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_RESUME_SONG);
-    int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-    NET_FreePacket(packet);
-    if (len < packet->len)
-    {
-        DEBUGOUT("I_MidiSocketResumeSong failed");
-        return false;
-    }
-
-    DEBUGOUT("I_MidiSocketResumeSong succeeded");
-   return true;
-}
-
-//=============================================================================
-//
-// Public Interface
-//
-
-//
-// I_MidiSocketInitServer
-//
-// Start up the RPC MIDI server.
-//
-boolean I_MidiSocketInitServer()
-{
-    struct stat sbuf;
-    char filename[MAX_PATH+1];
-
-    memset(filename, 0, sizeof(filename));
-    size_t filename_len = GetModuleFileName(NULL, filename, MAX_PATH);
-
-    // Remove filespec
-    // TODO: Move this to m_misc
-    char *fp = &filename[filename_len];
-    while (filename <= fp && *fp != DIR_SEPARATOR)
-    {
-        fp--;
-    }
-    *(fp + 1) = '\0';
-    char* module = M_StringJoin(filename, "midiproc.exe", NULL);
-    DEBUGOUT(module);
-
-    // Look for executable file
-    if(stat(module, &sbuf))
-    {
-        DEBUGOUT("Could not find midiproc");
-        return false;
-    }
-
-    si.cb = sizeof(si);
-
-    boolean result = CreateProcess(module, NULL, NULL, NULL, FALSE,
-                                   0, NULL, NULL, &si, &pi);
-
-    if (result)
-    {
-        DEBUGOUT("RPC server started");
-        server_init = true;
-    }
-    else
-    {
-        DEBUGOUT("CreateProcess failed to start midiproc");
-    }
-
-    return result;
-}
-
-//
-// I_MidiSocketInitClient
-//
-// Initialize client RPC bindings and bind to the server.
-//
-boolean I_MidiSocketInitClient()
-{
-    IPaddress ipaddress;
-
-    // If server didn't start, client cannot be bound.
-    if (!server_init)
-    {
-        goto fail;
-    }
-
-    // Resolve localhost to an IP address.
-    if (SDLNet_ResolveHost(&ipaddress, "localhost", 2993) == -1)
-    {
-        goto fail;
-    }
-
-    // Connect to the server.
-    midi_socket = SDLNet_TCP_Open(&ipaddress);
-    if (midi_socket == NULL)
-    {
-        goto fail;
-    }
-
-    DEBUGOUT("RPC client initialized");
-    client_init = true;
-
-    return I_MidiSocketWaitForServer();
-
-fail:
-    DEBUGOUT("RPC client binding failed");
-    return false;
-}
-
-//
-// I_MidiSocketClientShutDown
-//
-// Shutdown the RPC Client
-//
-void I_MidiSocketClientShutDown()
-{
-    // stop the server
-    if(server_init)
-    {
-        net_packet_t *packet;
-        packet = NET_NewPacket(2);
-        NET_WriteInt16(packet, NET_MIDISOCKET_PACKET_TYPE_STOP_SERVER);
-        int len = SDLNet_TCP_Send(midi_socket, packet->data, packet->len);
-        NET_FreePacket(packet);
-        if (len < packet->len)
-        {
-            DEBUGOUT("Problem encountered when stopping RPC server");
-        }
-
-        server_init = false;
-    }
-
-    if (midi_socket)
-    {
-        SDLNet_TCP_Close(midi_socket);
-        midi_socket = NULL;
-    }
-
-    client_init = false;
-}
-
-//
-// I_MidiSocketReady
-//
-// Returns true if both server and client initialized successfully.
-//
-boolean I_MidiSocketReady()
-{
-    CHECK_RPC_STATUS();
-
-    return true;
-}
-
-#endif
-
--- a/src/i_midisocket.h
+++ /dev/null
@@ -1,41 +1,0 @@
-//
-// Copyright(C) 2013 James Haley et al.
-// Copyright(C) 2017 Alex Mayfield
-//
-// This program is free software; you can redistribute it and/or
-// modify it under the terms of the GNU General Public License
-// as published by the Free Software Foundation; either version 2
-// of the License, or (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful,
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-// GNU General Public License for more details.
-//
-// DESCRIPTION:
-//     Client Interface to RPC Midi Server
-//
-
-#ifndef __I_MIDISOCKET__
-#define __I_MIDISOCKET__
-
-#if _WIN32
-
-#include "doomtype.h"
-
-boolean I_MidiSocketInitServer();
-boolean I_MidiSocketInitClient();
-void I_MidiSocketClientShutDown();
-boolean I_MidiSocketReady();
-
-boolean I_MidiSocketRegisterSong(const char *filename);
-boolean I_MidiSocketPlaySong(boolean looping);
-boolean I_MidiSocketStopSong();
-boolean I_MidiSocketSetVolume(int volume);
-boolean I_MidiSocketPauseSong();
-boolean I_MidiSocketResumeSong();
-
-#endif
-
-#endif
-
--- a/src/i_sdlmusic.c
+++ b/src/i_sdlmusic.c
@@ -25,7 +25,7 @@
 #include "SDL.h"
 #include "SDL_mixer.h"
 
-#include "i_midisocket.h"
+#include "i_midipipe.h"
 
 #include "config.h"
 #include "doomtype.h"
@@ -975,7 +975,7 @@
     }
 
 #if WIN32
-    I_MidiSocketInitServer();
+    I_MidiPipeInitServer();
 #endif
 
     return music_initialized;
@@ -1000,7 +1000,7 @@
     }
 
 #if WIN32
-    I_MidiSocketSetVolume(vol);
+    I_MidiPipeSetVolume(vol);
 #else
     Mix_VolumeMusic(vol);
 #endif
@@ -1055,7 +1055,7 @@
     }
 
 #if _WIN32
-    I_MidiSocketPlaySong(loops);
+    I_MidiPipePlaySong(loops);
 #else
     Mix_PlayMusic(current_track_music, loops);
 #endif
@@ -1093,7 +1093,7 @@
     }
 
 #if _WIN32
-    I_MidiSocketStopSong();
+    I_MidiPipeStopSong();
 #else
     Mix_HaltMusic();
 #endif
@@ -1209,7 +1209,7 @@
     // we have to generate a temporary file.
 
 #ifdef _WIN32
-    music = (Mix_Music*)I_MidiSocketRegisterSong(filename);
+    music = (Mix_Music*)I_MidiPipeRegisterSong(filename);
 #else
     music = Mix_LoadMUS(filename);
 #endif
--- a/src/net_defs.h
+++ b/src/net_defs.h
@@ -142,6 +142,17 @@
     NET_MASTER_PACKET_TYPE_SIGN_END_RESPONSE,
 } net_master_packet_type_t;
 
+typedef enum {
+    NET_MIDIPIPE_PACKET_TYPE_PREPARE_NEW_SONG,
+    NET_MIDIPIPE_PACKET_TYPE_SET_FILENAME,
+    NET_MIDIPIPE_PACKET_TYPE_PLAY_SONG,
+    NET_MIDIPIPE_PACKET_TYPE_STOP_SONG,
+    NET_MIDIPIPE_PACKET_TYPE_CHANGE_VOLUME,
+    NET_MIDIPIPE_PACKET_TYPE_PAUSE_SONG,
+    NET_MIDIPIPE_PACKET_TYPE_RESUME_SONG,
+    NET_MIDIPIPE_PACKET_TYPE_STOP_SERVER
+} net_midipipe_packet_type_t;
+
 // Settings specified when the client connects to the server.
 
 typedef struct