mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			337 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			337 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			Java
		
	
	
	
| /* ShortMessage.java -- A MIDI message no longer than 3 bytes
 | |
|    Copyright (C) 2005 Free Software Foundation, Inc.
 | |
| 
 | |
| This file is part of GNU Classpath.
 | |
| 
 | |
| GNU Classpath 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, or (at your option)
 | |
| any later version.
 | |
| 
 | |
| GNU Classpath 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 GNU Classpath; see the file COPYING.  If not, write to the
 | |
| Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | |
| 02110-1301 USA.
 | |
| 
 | |
| Linking this library statically or dynamically with other modules is
 | |
| making a combined work based on this library.  Thus, the terms and
 | |
| conditions of the GNU General Public License cover the whole
 | |
| combination.
 | |
| 
 | |
| As a special exception, the copyright holders of this library give you
 | |
| permission to link this library with independent modules to produce an
 | |
| executable, regardless of the license terms of these independent
 | |
| modules, and to copy and distribute the resulting executable under
 | |
| terms of your choice, provided that you also meet, for each linked
 | |
| independent module, the terms and conditions of the license of that
 | |
| module.  An independent module is a module which is not derived from
 | |
| or based on this library.  If you modify this library, you may extend
 | |
| this exception to your version of the library, but you are not
 | |
| obligated to do so.  If you do not wish to do so, delete this
 | |
| exception statement from your version. */
 | |
| 
 | |
| 
 | |
| package javax.sound.midi;
 | |
| 
 | |
| /**
 | |
|  * A short MIDI message that is no longer than 3 bytes long.
 | |
|  *
 | |
|  * @author Anthony Green (green@redhat.com)
 | |
|  * @since 1.3
 | |
|  *
 | |
|  */
 | |
| public class ShortMessage extends MidiMessage
 | |
| {
 | |
|   /**
 | |
|    * Status byte for Time Code message.
 | |
|    */
 | |
|   public static final int MIDI_TIME_CODE = 0xF1;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Song Position Pointer message.
 | |
|    */
 | |
|   public static final int SONG_POSITION_POINTER = 0xF2;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Song Select message.
 | |
|    */
 | |
|   public static final int SONG_SELECT = 0xF3;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Tune Request message.
 | |
|    */
 | |
|   public static final int TUNE_REQUEST = 0xF6;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for End Of Exclusive message.
 | |
|    */
 | |
|   public static final int END_OF_EXCLUSIVE = 0xF7;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Timing Clock message.
 | |
|    */
 | |
|   public static final int TIMING_CLOCK = 0xF8;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Start message.
 | |
|    */
 | |
|   public static final int START = 0xFA;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Continue message.
 | |
|    */
 | |
|   public static final int CONTINUE = 0xFB;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Stop message.
 | |
|    */
 | |
|   public static final int STOP = 0xFC;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for Active Sensing message.
 | |
|    */
 | |
|   public static final int ACTIVE_SENSING = 0xFE;
 | |
| 
 | |
|   /**
 | |
|    * Status byte for System Reset message.
 | |
|    */
 | |
|   public static final int SYSTEM_RESET = 0xFF;
 | |
| 
 | |
|   /**
 | |
|    * Status nibble for Note Off message.
 | |
|    */
 | |
|   public static final int NOTE_OFF = 0x80;
 | |
| 
 | |
|   /**
 | |
|    * Status nibble for Note On message.
 | |
|    */
 | |
|   public static final int NOTE_ON = 0x90;
 | |
| 
 | |
|   /**
 | |
|    * Status nibble for Poly Pressure message.
 | |
|    */
 | |
|   public static final int POLY_PRESSURE = 0xA0;
 | |
| 
 | |
|   /**
 | |
|    * Status nibble for Control Change message.
 | |
|    */
 | |
|   public static final int CONTROL_CHANGE = 0xB0;
 | |
| 
 | |
|   /**
 | |
|    * Status nibble for Program Change message.
 | |
|    */
 | |
|   public static final int PROGRAM_CHANGE = 0xC0;
 | |
| 
 | |
|   /**
 | |
|    * Statue nibble for Channel Pressure message.
 | |
|    */
 | |
|   public static final int CHANNEL_PRESSURE = 0xD0;
 | |
| 
 | |
|   /**
 | |
|    * Status nibble for Pitch Bend message.
 | |
|    */
 | |
|   public static final int PITCH_BEND = 0xE0;
 | |
| 
 | |
|   // Create and initialize a default, arbitrary message.
 | |
|   private static byte[] defaultMessage;
 | |
|   static
 | |
|   {
 | |
|     defaultMessage = new byte[1];
 | |
|     defaultMessage[0] = (byte) STOP;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Create a short MIDI message.
 | |
|    *
 | |
|    * The spec requires that this represent a valid MIDI message, but doesn't
 | |
|    * specify what it should be.  We've chosen the STOP message for our
 | |
|    * implementation.
 | |
|    */
 | |
|   public ShortMessage()
 | |
|   {
 | |
|     this(defaultMessage);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Create a short MIDI message.
 | |
|    *
 | |
|    * The data argument should be a valid MIDI message.  Unfortunately the spec
 | |
|    * does not allow us to throw an InvalidMidiDataException if data is invalid.
 | |
|    *
 | |
|    * @param data the message data
 | |
|    */
 | |
|   protected ShortMessage(byte[] data)
 | |
|   {
 | |
|     super(data);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Set the MIDI message.
 | |
|    *
 | |
|    * @param status the status byte for this message
 | |
|    * @param data1 the first data byte for this message
 | |
|    * @param data2 the second data byte for this message
 | |
|    * @throws InvalidMidiDataException if status is bad, or data is out of range
 | |
|    */
 | |
|   public void setMessage(int status, int data1, int data2)
 | |
|     throws InvalidMidiDataException
 | |
|   {
 | |
|     length = getDataLength(status);
 | |
|     length++;
 | |
|     if (data == null || data.length < length)
 | |
|       data = new byte[length];
 | |
|     data[0] = (byte) status;
 | |
|     if (length > 1)
 | |
|     {
 | |
|       if (data1 < 0 || data1 > 127)
 | |
|         throw new InvalidMidiDataException("data1 (" + data1
 | |
|                                            + ") must be between 0 and 127.");
 | |
|       data[1] = (byte) data1;
 | |
|       if (length > 2)
 | |
|       {
 | |
|         if (data2 < 0 || data2 > 127)
 | |
|           throw new InvalidMidiDataException("data2 (" + data2
 | |
|                                              + ") must be between 0 and 127.");
 | |
|         data[2] = (byte) data2;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   public void setMessage(int command, int channel, int data1, int data2)
 | |
|     throws InvalidMidiDataException
 | |
|   {
 | |
|     // TODO: This could probably stand some error checking.
 | |
|     // It currently assumes command and channel are valid values.
 | |
|     setMessage(command + channel, data1, data2);
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Set the MIDI message to one that requires no data bytes.
 | |
|    *
 | |
|    * @param status the status byte for this message
 | |
|    * @throws InvalidMidiDataException if status is bad, or requires data
 | |
|    */
 | |
|   public void setMessage(int status) throws InvalidMidiDataException
 | |
|   {
 | |
|     int length = getDataLength(status);
 | |
|     if (length != 0)
 | |
|       throw new InvalidMidiDataException("Status byte 0x"
 | |
|                                          + Integer.toHexString(status)
 | |
|                                          + " requires "
 | |
|                                          + length + " bytes of data.");
 | |
|     setMessage(status, 0, 0);
 | |
|   }
 | |
| 
 | |
| 
 | |
|   /**
 | |
|    * Return the number of data bytes needed for a given MIDI status byte.
 | |
|    *
 | |
|    * @param status the status byte for a short MIDI message
 | |
|    * @return the number of data bytes needed for this status byte
 | |
|    * @throws InvalidMidiDataException if status is an invalid status byte
 | |
|    */
 | |
|   protected final int getDataLength(int status) throws InvalidMidiDataException
 | |
|   {
 | |
|     int originalStatus = status;
 | |
| 
 | |
|     if ((status & 0xF0) != 0xF0)
 | |
|       status &= 0xF0;
 | |
| 
 | |
|     switch (status)
 | |
|     {
 | |
|     case NOTE_OFF:
 | |
|     case NOTE_ON:
 | |
|     case POLY_PRESSURE:
 | |
|     case CONTROL_CHANGE:
 | |
|     case PITCH_BEND:
 | |
|     case SONG_POSITION_POINTER:
 | |
|       return 2;
 | |
| 
 | |
|     case PROGRAM_CHANGE:
 | |
|     case CHANNEL_PRESSURE:
 | |
|     case SONG_SELECT:
 | |
|     case 0xF5:  // FIXME: unofficial bus select.  Not in spec??
 | |
|       return 1;
 | |
| 
 | |
|     case TUNE_REQUEST:
 | |
|     case END_OF_EXCLUSIVE:
 | |
|     case TIMING_CLOCK:
 | |
|     case START:
 | |
|     case CONTINUE:
 | |
|     case STOP:
 | |
|     case ACTIVE_SENSING:
 | |
|     case SYSTEM_RESET:
 | |
|       return 0;
 | |
| 
 | |
|     default:
 | |
|       throw new InvalidMidiDataException("Invalid status: 0x"
 | |
|                                          + Integer.toHexString(originalStatus));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Get the channel information from this MIDI message, assuming it is a
 | |
|    * MIDI channel message.
 | |
|    *
 | |
|    * @return the MIDI channel for this message
 | |
|    */
 | |
|   public int getChannel()
 | |
|   {
 | |
|     return data[0] & 0x0F;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Get the command nibble from this MIDI message, assuming it is a MIDI
 | |
|    * channel message.
 | |
|    *
 | |
|    * @return the MIDI command for this message
 | |
|    */
 | |
|   public int getCommand()
 | |
|   {
 | |
|     return data[0] & 0xF0;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Get the first data byte from this message, assuming it exists, and
 | |
|    * zero otherwise.
 | |
|    *
 | |
|    * @return the first data byte or zero if none exists.
 | |
|    */
 | |
|   public int getData1()
 | |
|   {
 | |
|     if (length > 1)
 | |
|       return data[1];
 | |
|     else
 | |
|       return 0;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Get the second data byte from this message, assuming it exists, and
 | |
|    * zero otherwise.
 | |
|    *
 | |
|    * @return the second date byte or zero if none exists.
 | |
|    */
 | |
|   public int getData2()
 | |
|   {
 | |
|     if (length > 2)
 | |
|       return data[2];
 | |
|     else
 | |
|       return 0;
 | |
|   }
 | |
| 
 | |
|   /* Create a deep-copy clone of this object.
 | |
|    * @see java.lang.Object#clone()
 | |
|    */
 | |
|   public Object clone()
 | |
|   {
 | |
|     byte message[] = new byte[length];
 | |
|     System.arraycopy(data, 0, message, 0, length);
 | |
|     return new ShortMessage(message);
 | |
|   }
 | |
| }
 |