Logo Search packages:      
Sourcecode: scummvm version File versions  Download package

coremidi.cpp

/* ScummVM - Graphic Adventure Engine
 *
 * ScummVM is the legal property of its developers, whose names
 * are too numerous to list here. Please refer to the COPYRIGHT
 * file distributed with this source distribution.
 *
 * 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.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 *
 * $URL: https://scummvm.svn.sourceforge.net/svnroot/scummvm/scummvm/tags/release-1-2-1/backends/midi/coremidi.cpp $
 * $Id: coremidi.cpp 50128 2010-06-21 21:36:36Z fingolfin $
 */

#ifdef MACOSX

#include "common/config-manager.h"
#include "common/util.h"
#include "sound/musicplugin.h"
#include "sound/mpu401.h"

#include <CoreMIDI/CoreMIDI.h>



/*
For information on how to unify the CoreMidi and MusicDevice code:

http://lists.apple.com/archives/Coreaudio-api/2005/Jun/msg00194.html
http://lists.apple.com/archives/coreaudio-api/2003/Mar/msg00248.html
http://lists.apple.com/archives/coreaudio-api/2003/Jul/msg00137.html

*/


/* CoreMIDI MIDI driver
 * By Max Horn
 */
class MidiDriver_CoreMIDI : public MidiDriver_MPU401 {
public:
      MidiDriver_CoreMIDI();
      ~MidiDriver_CoreMIDI();
      int open();
      void close();
      void send(uint32 b);
      void sysEx(const byte *msg, uint16 length);

private:
      MIDIClientRef     mClient;
      MIDIPortRef       mOutPort;
      MIDIEndpointRef   mDest;
};

MidiDriver_CoreMIDI::MidiDriver_CoreMIDI()
      : mClient(0), mOutPort(0), mDest(0) {

      OSStatus err;
      err = MIDIClientCreate(CFSTR("ScummVM MIDI Driver for OS X"), NULL, NULL, &mClient);
}

MidiDriver_CoreMIDI::~MidiDriver_CoreMIDI() {
      if (mClient)
            MIDIClientDispose(mClient);
      mClient = 0;
}

int MidiDriver_CoreMIDI::open() {
      if (mDest)
            return MERR_ALREADY_OPEN;

      OSStatus err = noErr;

      mOutPort = 0;

      int dests = MIDIGetNumberOfDestinations();
      if (dests > 0 && mClient) {
            mDest = MIDIGetDestination(0);
            err = MIDIOutputPortCreate( mClient,
                                                      CFSTR("scummvm_output_port"),
                                                      &mOutPort);
      } else {
            return MERR_DEVICE_NOT_AVAILABLE;
      }

      if (err != noErr)
            return MERR_CANNOT_CONNECT;

      return 0;
}

void MidiDriver_CoreMIDI::close() {
      MidiDriver_MPU401::close();

      if (mOutPort && mDest) {
            MIDIPortDispose(mOutPort);
            mOutPort = 0;
            mDest = 0;
      }
}

void MidiDriver_CoreMIDI::send(uint32 b) {
      assert(mOutPort != 0);
      assert(mDest != 0);

      // Extract the MIDI data
      byte status_byte = (b & 0x000000FF);
      byte first_byte = (b & 0x0000FF00) >> 8;
      byte second_byte = (b & 0x00FF0000) >> 16;

      // Generate a single MIDI packet with that data
      MIDIPacketList packetList;
      MIDIPacket *packet = &packetList.packet[0];

      packetList.numPackets = 1;

      packet->timeStamp = 0;
      packet->data[0] = status_byte;
      packet->data[1] = first_byte;
      packet->data[2] = second_byte;

      // Compute the correct length of the MIDI command. This is important,
      // else things may screw up badly...
      switch (status_byte & 0xF0) {
      case 0x80:  // Note Off
      case 0x90:  // Note On
      case 0xA0:  // Polyphonic Aftertouch
      case 0xB0:  // Controller Change
      case 0xE0:  // Pitch Bending
            packet->length = 3;
            break;
      case 0xC0:  // Programm Change
      case 0xD0:  // Monophonic Aftertouch
            packet->length = 2;
            break;
      default:
            warning("CoreMIDI driver encountered unsupported status byte: 0x%02x", status_byte);
            packet->length = 3;
            break;
      }

      // Finally send it out to the synthesizer.
      MIDISend(mOutPort, mDest, &packetList);
}

void MidiDriver_CoreMIDI::sysEx(const byte *msg, uint16 length) {
      assert(mOutPort != 0);
      assert(mDest != 0);

      byte buf[384];
      MIDIPacketList *packetList = (MIDIPacketList *)buf;
      MIDIPacket *packet = packetList->packet;

      assert(sizeof(buf) >= sizeof(UInt32) + sizeof(MIDITimeStamp) + sizeof(UInt16) + length + 2);

      packetList->numPackets = 1;

      packet->timeStamp = 0;

      // Add SysEx frame
      packet->length = length + 2;
      packet->data[0] = 0xF0;
      memcpy(packet->data + 1, msg, length);
      packet->data[length + 1] = 0xF7;

      // Send it
      MIDISend(mOutPort, mDest, packetList);
}


// Plugin interface

class CoreMIDIMusicPlugin : public MusicPluginObject {
public:
      const char *getName() const {
            return "CoreMIDI";
      }

      const char *getId() const {
            return "coremidi";
      }

      MusicDevices getDevices() const;
      Common::Error createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle = 0) const;
};

MusicDevices CoreMIDIMusicPlugin::getDevices() const {
      MusicDevices devices;
      // TODO: Return a different music type depending on the configuration
      // TODO: List the available devices
      devices.push_back(MusicDevice(this, "", MT_GM));
      return devices;
}

Common::Error CoreMIDIMusicPlugin::createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle) const {
      *mididriver = new MidiDriver_CoreMIDI();

      return Common::kNoError;
}

//#if PLUGIN_ENABLED_DYNAMIC(COREMIDI)
      //REGISTER_PLUGIN_DYNAMIC(COREMIDI, PLUGIN_TYPE_MUSIC, CoreMIDIMusicPlugin);
//#else
      REGISTER_PLUGIN_STATIC(COREMIDI, PLUGIN_TYPE_MUSIC, CoreMIDIMusicPlugin);
//#endif

#endif // MACOSX

Generated by  Doxygen 1.6.0   Back to index