All Downloads are FREE. Search and download functionalities are using the official Maven repository.

de.sciss.synth.osc.ServerMessages.scala Maven / Gradle / Ivy

/*
 *  ServerMessages.scala
 *  (ScalaCollider)
 *
 *  Copyright (c) 2008-2013 Hanns Holger Rutz. All rights reserved.
 *
 *  This software 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, june 1991 of the License, or (at your option) any later version.
 *
 *  This software 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 (gpl.txt) along with this software; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 *
 *  For further information, please contact Hanns Holger Rutz at
 *  [email protected]
 */

package de.sciss.synth
package osc

import java.nio.ByteBuffer
import collection.immutable.{ IndexedSeq => IIdxSeq}
import collection.mutable.ListBuffer
import java.io.PrintStream
import de.sciss.osc.{Bundle, Message, Packet, PacketCodec}

object ServerCodec extends PacketCodec {
   import Packet._

   private val superCodec = PacketCodec().scsynth().build

	private val decodeStatusReply: (String, ByteBuffer) => Message = (name, b) => {
		// ",iiiiiffdd"
		if( (b.getLong != 0x2C69696969696666L) || (b.getShort != 0x6464) ) decodeFail( name )
		skipToValues( b )

//		if( b.getInt() != 1) decodeFail  // was probably intended as a version number...
		b.getInt()
		val numUGens		= b.getInt()
		val numSynths		= b.getInt()
		val numGroups		= b.getInt()
		val numDefs			= b.getInt()
		val avgCPU			= b.getFloat()
		val peakCPU			= b.getFloat()
		val sampleRate		= b.getDouble()
		val actualSampleRate= b.getDouble()

		StatusReplyMessage( numUGens, numSynths, numGroups, numDefs,
		                    avgCPU, peakCPU, sampleRate, actualSampleRate )
	}

   private val decodeSynced: (String, ByteBuffer) => Message = (name, b) => {
      // ",i"
      if( b.getShort() != 0x2C69 ) decodeFail( name )
      skipToValues( b )

      val id = b.getInt()

      SyncedMessage( id )
   }

	private def decodeNodeChange( factory: NodeMessageFactory ) :
      (String, ByteBuffer) => Message = (name, b) => {

		// ",iiiii[ii]"
		if( (b.getInt() != 0x2C696969) || (b.getShort() != 0x6969) ) decodeFail( name )
		val extTags = b.getShort()
		if( (extTags & 0xFF) == 0x00 ) {
			skipToAlign( b )
		} else {
			skipToValues( b )
		}
		val nodeID		= b.getInt()
		val parentID	= b.getInt()
		val predID		= b.getInt()
		val succID		= b.getInt()
		val nodeType	= b.getInt()

      if( nodeType == 0 ) {
         factory.apply( nodeID, SynthInfo( parentID, predID, succID ))
      } else if( (nodeType == 1) && (extTags == 0x6969) ) {	// group
         val headID	= b.getInt()
         val tailID	= b.getInt()
         factory.apply( nodeID, GroupInfo( parentID, predID, succID, headID, tailID ))
		} else decodeFail( name )
	}

   private val decodeBufferInfo: (String, ByteBuffer) => Message = (name, b) => {
      // ",[iiif]*N"
      if( b.get() != 0x2C ) decodeFail( name )
      var cnt = 0
      var tag = b.getShort()
      while( tag != 0x0000 ) {
         if( (tag != 0x6969) || (b.getShort() != 0x6966) ) decodeFail( name )
         cnt += 1
         tag = b.getShort()
      }
      skipToAlign( b )
      var infos = new ListBuffer[ BufferInfo ]
      while( cnt > 0 ) {
         infos += BufferInfo( b.getInt(), b.getInt(), b.getInt(), b.getFloat() )
         cnt -= 1
      }
      BufferInfoMessage( infos: _* )
   }

   private val msgDecoders = Map[ String, (String, ByteBuffer) => Message ](
      "/status.reply"   -> decodeStatusReply,
      "/n_go"			   -> decodeNodeChange( NodeGoMessage ),
      "/n_end"		      -> decodeNodeChange( NodeEndMessage ),
      "/n_off"		      -> decodeNodeChange( NodeOffMessage ),
      "/n_on"			   -> decodeNodeChange( NodeOnMessage ),
      "/n_move"		   -> decodeNodeChange( NodeMoveMessage ),
      "/n_info"		   -> decodeNodeChange( NodeInfoMessage ),
      "/synced"         -> decodeSynced,
      "/b_info"         -> decodeBufferInfo,
      "status.reply"	   -> decodeStatusReply
   )

   private val superDecoder: (String, ByteBuffer ) => Message =
      (name, b) => superCodec.decodeMessage( name, b ) // super.decodeMessage( name, b )

   override /* protected */ def decodeMessage( name: String, b: ByteBuffer ) : Message = {
        msgDecoders.getOrElse( name, superDecoder ).apply( name, b )
/*		val dec = msgDecoders.get( name )
		if( dec.isDefined ) {
			dec.get.apply( name, b )
		} else {
			super.decodeMessage( name, b )
		}
*/	}

   def encodeMessage( msg: Message, b: ByteBuffer ) { superCodec.encodeMessage( msg, b )}
   def encodedMessageSize( msg: Message ) = superCodec.encodedMessageSize( msg )
   def encodeBundle( bndl: Bundle, b: ByteBuffer ) { superCodec.encodeBundle( bndl, b )}
   def printAtom( value: Any, stream: PrintStream, nestCount: Int ) { superCodec.printAtom( value, stream, nestCount )}
   val charsetName = superCodec.charsetName

   private def decodeFail( name : String ) : Nothing = throw PacketCodec.MalformedPacket( name )
}
// val nodeID: Int, val parentID: Int, val predID: Int, val succID: Int, val headID: Int, val tailID: Int )

/**
 *    Identifies messages received or sent by the
 *    SuperCollider server
 */
sealed trait ServerMessage

/**
 * Identifies messages sent to the SuperCollider server
 */
sealed trait Send extends ServerMessage {
   def isSynchronous : Boolean
}
/**
 * Identifies messages sent to the server which are
 * executed synchronously
 */
sealed trait SyncSend extends Send {
   final def isSynchronous = true
}
/**
 * Identifies command messages sent to the server which are
 * executed synchronously and do not return a message
 */
trait SyncCmd extends SyncSend
/**
 * Identifies query messages sent to the server which are
 * executed synchronously and produce a reply message
 */
trait SyncQuery extends SyncSend
/**
 * Identifies messages sent to the server which are
 * executed asynchronously and reply with a form of
 * done-message.
 */
sealed trait AsyncSend extends Send {
   final def isSynchronous = false
}
/**
 * Identifies messages returned by SuperCollider server
 */
trait Receive extends ServerMessage

/**
 * Represents a `/synced` message, a reply from the server acknowledging that
 * all asynchronous operations up to the corresponding `/sync` message (i.e. with
 * the same id) have been completed
 */
final case class SyncedMessage( id: Int ) extends Message( "/synced", id )
with Receive

/**
 * Represents a `/sync` message, which is queued with the asynchronous messages
 * on the server, and which, when executed, triggers a corresponding `/synced` reply
 * message (i.e. with the same id)
 *
 * @param   id    an arbitrary identifier which can be used to match the corresponding
 *                reply message. typically the id is incremented by 1 for each
 *                `/sync` message sent out.
 */
final case class SyncMessage( id: Int ) extends Message( "/sync", id )
with AsyncSend

final case class StatusReplyMessage( numUGens: Int, numSynths: Int, numGroups: Int,
                                  numDefs: Int, avgCPU: Float, peakCPU: Float,
                                  sampleRate: Double, actualSampleRate: Double )
extends Message( "/status.reply", 1, numUGens, numSynths, numGroups, numDefs, avgCPU, peakCPU,
                    sampleRate, actualSampleRate )
with Receive

case object StatusMessage extends Message( "/status" )
with SyncQuery

//trait NodeChange {
//	def name: String // aka command (/n_go, /n_end, /n_off, /n_on, /n_move, /n_info)
//	def nodeID:   Int
//	def parentID: Int
//	def predID:   Int
//	def succID:   Int
//}

abstract sealed class NodeInfo {
   def parentID: Int
   def predID:   Int
   def succID:   Int
   def toList( nodeID: Int ): List[ Any ]
}
final case class SynthInfo( parentID: Int, predID: Int, succID: Int ) extends NodeInfo {
   def toList( nodeID: Int ) = List( nodeID, parentID, predID, succID, 0 )
}
final case class GroupInfo( parentID: Int, predID: Int, succID: Int, headID: Int, tailID: Int ) extends NodeInfo {
   def toList( nodeID: Int ) = List( nodeID, parentID, predID, succID, 1, headID, tailID )
}

sealed trait NodeChange extends Receive {
   def nodeID: Int
   def info:   NodeInfo
}

protected[synth] sealed trait NodeMessageFactory {
   def apply( nodeID: Int, info: NodeInfo ) : Message
}

object NodeGoMessage extends NodeMessageFactory
final case class NodeGoMessage( nodeID: Int, info: NodeInfo )
extends Message( "/n_go", info.toList( nodeID ): _* ) with NodeChange

object NodeEndMessage extends NodeMessageFactory
final case class NodeEndMessage( nodeID: Int, info: NodeInfo )
extends Message( "/n_end", info.toList( nodeID ): _* ) with NodeChange

object NodeOnMessage extends NodeMessageFactory
final case class NodeOnMessage( nodeID: Int, info: NodeInfo )
extends Message( "/n_on", info.toList( nodeID ): _* ) with NodeChange

object NodeOffMessage extends NodeMessageFactory
final case class NodeOffMessage( nodeID: Int, info: NodeInfo )
extends Message( "/n_off", info.toList( nodeID ): _* ) with NodeChange

object NodeMoveMessage extends NodeMessageFactory
final case class NodeMoveMessage( nodeID: Int, info: NodeInfo )
extends Message( "/n_move", info.toList( nodeID ): _* ) with NodeChange

object NodeInfoMessage extends NodeMessageFactory
final case class NodeInfoMessage( nodeID: Int, info: NodeInfo )
extends Message( "/n_info", info.toList( nodeID ): _* ) with NodeChange

final case class BufferInfo( bufID: Int, numFrames: Int, numChannels: Int, sampleRate: Float )

// we need List[ Any ] as scala would otherwise expand to List[ Float ]!
final case class BufferInfoMessage( infos: BufferInfo* )
extends Message( "/b_info", infos.flatMap( info =>
   List[ Any ]( info.bufID, info.numFrames, info.numChannels, info.sampleRate )): _* )
with Receive

// ---- messages to the server ----
final case class ServerNotifyMessage( onOff: Boolean )
extends Message( "/notify", onOff )
with AsyncSend

case object ServerQuitMessage extends Message( "/quit" )
with AsyncSend

final case class BufferQueryMessage( ids: Int* ) extends Message( "/b_query", ids: _* )
with SyncQuery

final case class BufferFreeMessage( id: Int, completion: Option[ Packet ])
extends Message( "/b_free", (completion.map( m => List( id, m )) getOrElse List( id )): _* )
with AsyncSend

final case class BufferCloseMessage( id: Int, completion: Option[ Packet ])
extends Message( "/b_close", (completion.map( m => List( id, m )) getOrElse List( id )): _* )
with AsyncSend

final case class BufferAllocMessage( id: Int, numFrames: Int, numChannels: Int, completion: Option[ Packet ])
extends Message( "/b_alloc", (completion.map( m => List( id, numFrames, numChannels, m ))
                                                   getOrElse List( id, numFrames, numChannels )): _* )
with AsyncSend

final case class BufferAllocReadMessage( id: Int, path: String, startFrame: Int, numFrames: Int,
                                      completion: Option[ Packet ])
extends Message( "/b_allocRead", (completion.map( m => List( id, path, startFrame, numFrames, m ))
                                                       getOrElse List( id, path, startFrame, numFrames )): _* )
with AsyncSend

final case class BufferAllocReadChannelMessage( id: Int, path: String, startFrame: Int, numFrames: Int,
                                             channels: List[ Int ], completion: Option[ Packet ])
extends Message( "/b_allocReadChannel", (List( id, path, startFrame, numFrames ) ::: channels
   ::: completion.map( msg => List( msg )).getOrElse( Nil )): _* )
with AsyncSend

final case class BufferReadMessage( id: Int, path: String, fileStartFrame: Int, numFrames: Int, bufStartFrame: Int,
                                 leaveOpen: Boolean, completion: Option[ Packet ])
extends Message( "/b_read", (completion.map(
   m =>      List( id, path, fileStartFrame, numFrames, bufStartFrame, leaveOpen, m ))
   getOrElse List( id, path, fileStartFrame, numFrames, bufStartFrame, leaveOpen )): _* )
with AsyncSend

final case class BufferReadChannelMessage( id: Int, path: String, fileStartFrame: Int, numFrames: Int,
                                        bufStartFrame: Int, leaveOpen: Boolean, channels: List[ Int ],
                                        completion: Option[ Packet ])
extends Message( "/b_readChannel", (List( id, path, fileStartFrame, numFrames, bufStartFrame, leaveOpen ) :::
   channels ::: completion.map( msg => List( msg )).getOrElse( Nil )): _* )
with AsyncSend

final case class BufferZeroMessage( id: Int, completion: Option[ Packet ])
extends Message( "/b_zero", (completion.map( m => List( id, m )) getOrElse List( id )): _* )
with AsyncSend

final case class BufferWriteMessage( id: Int, path: String, fileType: io.AudioFileType, sampleFormat: io.SampleFormat,
                                  numFrames: Int, startFrame: Int, leaveOpen: Boolean,
                                  completion: Option[ Packet])
extends Message( "/b_write", (List( id, path, fileType.id, sampleFormat.id, numFrames, startFrame, leaveOpen ) :::
   completion.map( msg => List( msg )).getOrElse( Nil )): _* )
with AsyncSend

final case class BufferSetMessage( id: Int, indicesAndValues: (Int, Float)* )
extends Message( "/b_set", (id +: indicesAndValues.flatMap( iv => List[ Any ]( iv._1, iv._2 ))): _* )
with SyncCmd

final case class BufferSetnMessage( id: Int, indicesAndValues: (Int, IIdxSeq[ Float ])* )
extends Message( "/b_setn", (id +: indicesAndValues.flatMap( iv => Vector( iv._1, iv._2.size ) ++ iv._2 )): _* )
with SyncCmd

//case class BusValuePair( index: Int, value: Float )
final case class ControlBusSetMessage( indicesAndValues: (Int, Float)* )
extends Message( "/c_set", indicesAndValues.flatMap( iv => List[ Any ]( iv._1, iv._2 )): _* )
with SyncCmd

//case class BusValuesPair( index: Int, values: IIdxSeq[ Float ])
final case class ControlBusSetnMessage( indicesAndValues: (Int, IIdxSeq[ Float ])* )
extends Message( "/c_setn", indicesAndValues.flatMap( iv => Vector( iv._1, iv._2.size ) ++ iv._2 ): _* )
with SyncCmd

final case class ControlBusGetMessage( index: Int* ) // fucking hell: indices is defined for SeqLike
extends Message( "/c_get", index: _* )
with SyncQuery

final case class GroupNewInfo( groupID: Int, addAction: Int, targetID: Int )
final case class GroupNewMessage( groups: GroupNewInfo* )
extends Message( "/g_new", groups.flatMap( g => List( g.groupID, g.addAction, g.targetID )): _* )
with SyncCmd

//case class NodeFlagPair( id: Int, flag: Boolean )
final case class GroupDumpTreeMessage( groups: (Int, Boolean)* )
extends Message( "/g_dumpTree", groups.flatMap( g => List( g._1, g._2 )): _* )
with SyncCmd

final case class GroupQueryTreeMessage( groups: (Int, Boolean)* )
extends Message( "/g_queryTree", groups.flatMap( g => List( g._1, g._2 )): _* )
with SyncQuery

/**
 * Represents an `/g_head` message, which pair-wise places nodes at the head
 * of groups.
 * {{{
 * /g_head
 *   [
 *     Int - the ID of the group at which head a node is to be placed (B)
 *     int - the ID of the node to place (A)
 *   ] * N
 * }}}
 * So that for each pair, node A is moved to the head of group B.
 */
final case class GroupHeadMessage( groups: (Int, Int)* )
extends Message( "/g_head", groups.flatMap( g => List( g._1, g._2 )): _* )
with SyncCmd

/**
 * Represents an `/g_tail` message, which pair-wise places nodes at the tail
 * of groups.
 * {{{
 * /g_tail
 *   [
 *     Int - the ID of the group at which tail a node is to be placed (B)
 *     int - the ID of the node to place (A)
 *   ] * N
 * }}}
 * So that for each pair, node A is moved to the tail of group B.
 */
final case class GroupTailMessage( groups: (Int, Int)* )
extends Message( "/g_tail", groups.flatMap( g => List( g._1, g._2 )): _* )
with SyncCmd

final case class GroupFreeAllMessage( ids: Int* )
extends Message( "/g_freeAll", ids: _* )
with SyncCmd

final case class GroupDeepFreeMessage( ids: Int* )
extends Message( "/g_deepFree", ids: _* )
with SyncCmd

final case class SynthNewMessage( defName: String, id: Int, addAction: Int, targetID: Int, controls: ControlSetMap* )
extends Message( "/s_new",
   (Vector( defName, id, addAction, targetID ) ++ controls.flatMap( _.toSetSeq )): _* )
with SyncCmd

final case class NodeRunMessage( nodes: (Int, Boolean)* )
extends Message( "/n_run", nodes.flatMap( n => List( n._1, n._2 )): _* )
with SyncCmd

final case class NodeSetMessage( id: Int, pairs: ControlSetMap* )
extends Message( "/n_set", (id +: pairs.flatMap( _.toSetSeq )): _* )
with SyncCmd

final case class NodeSetnMessage( id: Int, pairs: ControlSetMap* )
extends Message( "/n_setn", (id +: pairs.flatMap( _.toSetnSeq )): _* )
with SyncCmd

final case class NodeTraceMessage( ids: Int* )
extends Message( "/n_trace", ids: _* )
with SyncCmd

final case class NodeNoIDMessage( ids: Int* )
extends Message( "/n_noid", ids: _* )
with SyncCmd

final case class NodeFreeMessage( ids: Int* )
extends Message( "/n_free", ids: _* )
with SyncCmd

final case class NodeMapMessage( id: Int, mappings: ControlKBusMap.Single* )
extends Message( "/n_map", (id +: mappings.flatMap( _.toMapSeq )): _* )
with SyncCmd

final case class NodeMapnMessage( id: Int, mappings: ControlKBusMap* )
extends Message( "/n_mapn", (id +: mappings.flatMap( _.toMapnSeq )): _* )
with SyncCmd

final case class NodeMapaMessage( id: Int, mappings: ControlABusMap.Single* )
extends Message( "/n_mapa", (id +: mappings.flatMap( _.toMapaSeq )): _* )
with SyncCmd

final case class NodeMapanMessage( id: Int, mappings: ControlABusMap* )
extends Message( "/n_mapan", (id +: mappings.flatMap( _.toMapanSeq )): _* )
with SyncCmd

final case class NodeFillInfo( control: Any, numChannels: Int, value: Float )

final case class NodeFillMessage( id: Int, fillings: NodeFillInfo* )
extends Message( "/n_fill", (id +: fillings.flatMap( f => Vector( f.control, f.numChannels, f.value ))): _* )
with SyncCmd

/**
 * Represents an `/n_before` message, which pair-wise places nodes before
 * other nodes.
 * {{{
 * /n_before
 *   [
 *     Int - the ID of the node to place (A)
 *     int - the ID of the node before which the above is placed (B)
 *   ] * N
 * }}}
 * So that for each pair, node A in the same group as node B, to execute immediately before node B.
 */
final case class NodeBeforeMessage( groups: (Int, Int)* )
extends Message( "/n_before", groups.flatMap( g => List( g._1, g._2 )): _* )
with SyncCmd

/**
 * Represents an `/n_after` message, which pair-wise places nodes after
 * other nodes.
 * {{{
 * /n_after
 *   [
 *     Int - the ID of the node to place (A)
 *     int - the ID of the node after which the above is placed (B)
 *   ] * N
 * }}}
 * So that for each pair, node A in the same group as node B, to execute immediately after node B.
 */
final case class NodeAfterMessage( groups: (Int, Int)* )
extends Message( "/n_after", groups.flatMap( g => List( g._1, g._2 )): _* )
with SyncCmd

final case class SynthDefRecvMessage( bytes: ByteBuffer, completion: Option[ Packet ])
extends Message( "/d_recv", (bytes :: (completion.map( List( _ )) getOrElse Nil)): _* )
with AsyncSend

final case class SynthDefFreeMessage( names: String* )
extends Message( "/d_free", names: _* )
with SyncCmd

final case class SynthDefLoadMessage( path: String, completion: Option[ Packet ])
extends Message( "/d_load", (path :: (completion.map( List( _ )) getOrElse Nil)): _* )
with AsyncSend

final case class SynthDefLoadDirMessage( path: String, completion: Option[ Packet ])
extends Message( "/d_loadDir", (path :: (completion.map( List( _ )) getOrElse Nil)): _* )
with AsyncSend




© 2015 - 2025 Weber Informatics LLC | Privacy Policy