view src/adplug/core/mkj.cxx @ 3198:83b1a4e5f453

alsa-ng: Keep mixer open even when playback stopped.
author John Lindgren <john.lindgren@tds.net>
date Wed, 22 Jul 2009 16:42:16 -0400
parents 4709ce4e209e
children
line wrap: on
line source

/*
 * Adplug - Replayer for many OPL2/OPL3 audio file formats.
 * Copyright (C) 1999 - 2004 Simon Peter, <dn.tlp@gmx.net>, et al.
 * 
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 * 
 * This library 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
 * Lesser General Public License for more details.
 * 
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * mkj.cpp - MKJamz Player, by Simon Peter <dn.tlp@gmx.net>
 */

#include <assert.h>

#include "mkj.h"
#include "debug.h"

CPlayer *
CmkjPlayer::factory (Copl * newopl)
{
  return new CmkjPlayer (newopl);
}

bool
CmkjPlayer::load (VFSFile * fd, const CFileProvider & fp)
{
  binistream *f = fp.open (fd);
  if (!f)
    return false;
  char id[6];
  float ver;
  int i, j;
  short inst[8];

  // file validation
  f->readString (id, 6);
  if (strncmp (id, "MKJamz", 6))
  {
    fp.close (f);
    return false;
  }
  ver = f->readFloat (binio::Single);
  if (ver > 1.12)
  {
    fp.close (f);
    return false;
  }

  // load
  maxchannel = f->readInt (2);
  opl->init ();
  opl->write (1, 32);
  for (i = 0; i < maxchannel; i++)
  {
    for (j = 0; j < 8; j++)
      inst[j] = f->readInt (2);
    opl->write (0x20 + op_table[i], inst[4]);
    opl->write (0x23 + op_table[i], inst[0]);
    opl->write (0x40 + op_table[i], inst[5]);
    opl->write (0x43 + op_table[i], inst[1]);
    opl->write (0x60 + op_table[i], inst[6]);
    opl->write (0x63 + op_table[i], inst[2]);
    opl->write (0x80 + op_table[i], inst[7]);
    opl->write (0x83 + op_table[i], inst[3]);
  }
  maxnotes = f->readInt (2);
  songbuf = new short[(maxchannel + 1) * maxnotes];
  for (i = 0; i < maxchannel; i++)
    channel[i].defined = f->readInt (2);
  for (i = 0; i < (maxchannel + 1) * maxnotes; i++)
    songbuf[i] = f->readInt (2);

  AdPlug_LogWrite
    ("CmkjPlayer::load(\"%s\"): loaded file ver %.2f, %d channels,"
     " %d notes/channel.\n", fd->uri, ver, maxchannel, maxnotes);
  fp.close (f);
  rewind (0);
  return true;
}

bool
CmkjPlayer::update ()
{
  int c, i;
  short note;

  for (c = 0; c < maxchannel; c++)
  {
    if (!channel[c].defined)    // skip if channel is disabled
      continue;

    if (channel[c].pstat)
    {
      channel[c].pstat--;
      continue;
    }

    opl->write (0xb0 + c, 0);   // key off
    do
    {
      assert (channel[c].songptr < (maxchannel + 1) * maxnotes);
      note = songbuf[channel[c].songptr];
      if (channel[c].songptr - c > maxchannel)
        if (note && note < 250)
          channel[c].pstat = channel[c].speed;
      switch (note)
      {
        // normal notes
      case 68:
        opl->write (0xa0 + c, 0x81);
        opl->write (0xb0 + c, 0x21 + 4 * channel[c].octave);
        break;
      case 69:
        opl->write (0xa0 + c, 0xb0);
        opl->write (0xb0 + c, 0x21 + 4 * channel[c].octave);
        break;
      case 70:
        opl->write (0xa0 + c, 0xca);
        opl->write (0xb0 + c, 0x21 + 4 * channel[c].octave);
        break;
      case 71:
        opl->write (0xa0 + c, 0x2);
        opl->write (0xb0 + c, 0x22 + 4 * channel[c].octave);
        break;
      case 65:
        opl->write (0xa0 + c, 0x41);
        opl->write (0xb0 + c, 0x22 + 4 * channel[c].octave);
        break;
      case 66:
        opl->write (0xa0 + c, 0x87);
        opl->write (0xb0 + c, 0x22 + 4 * channel[c].octave);
        break;
      case 67:
        opl->write (0xa0 + c, 0xae);
        opl->write (0xb0 + c, 0x22 + 4 * channel[c].octave);
        break;
      case 17:
        opl->write (0xa0 + c, 0x6b);
        opl->write (0xb0 + c, 0x21 + 4 * channel[c].octave);
        break;
      case 18:
        opl->write (0xa0 + c, 0x98);
        opl->write (0xb0 + c, 0x21 + 4 * channel[c].octave);
        break;
      case 20:
        opl->write (0xa0 + c, 0xe5);
        opl->write (0xb0 + c, 0x21 + 4 * channel[c].octave);
        break;
      case 21:
        opl->write (0xa0 + c, 0x20);
        opl->write (0xb0 + c, 0x22 + 4 * channel[c].octave);
        break;
      case 15:
        opl->write (0xa0 + c, 0x63);
        opl->write (0xb0 + c, 0x22 + 4 * channel[c].octave);
        break;
      case 255:                // delay
        channel[c].songptr += maxchannel;
        channel[c].pstat = songbuf[channel[c].songptr];
        break;
      case 254:                // set octave
        channel[c].songptr += maxchannel;
        channel[c].octave = songbuf[channel[c].songptr];
        break;
      case 253:                // set speed
        channel[c].songptr += maxchannel;
        channel[c].speed = songbuf[channel[c].songptr];
        break;
      case 252:                // set waveform
        channel[c].songptr += maxchannel;
        channel[c].waveform = songbuf[channel[c].songptr] - 300;
        if (c > 2)
          opl->write (0xe0 + c + (c + 6), channel[c].waveform);
        else
          opl->write (0xe0 + c, channel[c].waveform);
        break;
      case 251:                // song end
        for (i = 0; i < maxchannel; i++)
          channel[i].songptr = i;
        songend = true;
        return false;
      }

      if (channel[c].songptr - c < maxnotes)
        channel[c].songptr += maxchannel;
      else
        channel[c].songptr = c;
    } while (!channel[c].pstat);
  }

  return !songend;
}

void
CmkjPlayer::rewind (int subsong)
{
  int i;

  for (i = 0; i < maxchannel; i++)
  {
    channel[i].pstat = 0;
    channel[i].speed = 0;
    channel[i].waveform = 0;
    channel[i].songptr = i;
    channel[i].octave = 4;
  }

  songend = false;
}

float
CmkjPlayer::getrefresh ()
{
  return 100.0f;
}