tomwalters@509: /* tomwalters@509: oscpack -- Open Sound Control packet manipulation library tomwalters@509: http://www.audiomulch.com/~rossb/oscpack tomwalters@509: tomwalters@509: Copyright (c) 2004-2005 Ross Bencina tomwalters@509: tomwalters@509: Permission is hereby granted, free of charge, to any person obtaining tomwalters@509: a copy of this software and associated documentation files tomwalters@509: (the "Software"), to deal in the Software without restriction, tomwalters@509: including without limitation the rights to use, copy, modify, merge, tomwalters@509: publish, distribute, sublicense, and/or sell copies of the Software, tomwalters@509: and to permit persons to whom the Software is furnished to do so, tomwalters@509: subject to the following conditions: tomwalters@509: tomwalters@509: The above copyright notice and this permission notice shall be tomwalters@509: included in all copies or substantial portions of the Software. tomwalters@509: tomwalters@509: Any person wishing to distribute modifications to the Software is tomwalters@509: requested to send the modifications to the original developer so that tomwalters@509: they can be incorporated into the canonical version. tomwalters@509: tomwalters@509: THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, tomwalters@509: EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF tomwalters@509: MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. tomwalters@509: IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR tomwalters@509: ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF tomwalters@509: CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION tomwalters@509: WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. tomwalters@509: */ tomwalters@509: #include "OscPrintReceivedElements.h" tomwalters@509: tomwalters@509: #include tomwalters@509: #include tomwalters@509: #include tomwalters@509: tomwalters@509: tomwalters@509: namespace osc{ tomwalters@509: tomwalters@509: tomwalters@509: std::ostream& operator<<( std::ostream & os, tomwalters@509: const ReceivedMessageArgument& arg ) tomwalters@509: { tomwalters@509: switch( arg.TypeTag() ){ tomwalters@509: case TRUE_TYPE_TAG: tomwalters@509: os << "bool:true"; tomwalters@509: break; tomwalters@509: tomwalters@509: case FALSE_TYPE_TAG: tomwalters@509: os << "bool:false"; tomwalters@509: break; tomwalters@509: tomwalters@509: case NIL_TYPE_TAG: tomwalters@509: os << "(Nil)"; tomwalters@509: break; tomwalters@509: tomwalters@509: case INFINITUM_TYPE_TAG: tomwalters@509: os << "(Infinitum)"; tomwalters@509: break; tomwalters@509: tomwalters@509: case INT32_TYPE_TAG: tomwalters@509: os << "int32:" << arg.AsInt32Unchecked(); tomwalters@509: break; tomwalters@509: tomwalters@509: case FLOAT_TYPE_TAG: tomwalters@509: os << "float32:" << arg.AsFloatUnchecked(); tomwalters@509: break; tomwalters@509: tomwalters@509: case CHAR_TYPE_TAG: tomwalters@509: { tomwalters@509: char s[2] = {0}; tomwalters@509: s[0] = arg.AsCharUnchecked(); tomwalters@509: os << "char:'" << s << "'"; tomwalters@509: } tomwalters@509: break; tomwalters@509: tomwalters@509: case RGBA_COLOR_TYPE_TAG: tomwalters@509: { tomwalters@509: uint32 color = arg.AsRgbaColorUnchecked(); tomwalters@509: tomwalters@509: os << "RGBA:0x" tomwalters@509: << std::hex << std::setfill('0') tomwalters@509: << std::setw(2) << (int)((color>>24) & 0xFF) tomwalters@509: << std::setw(2) << (int)((color>>16) & 0xFF) tomwalters@509: << std::setw(2) << (int)((color>>8) & 0xFF) tomwalters@509: << std::setw(2) << (int)(color & 0xFF) tomwalters@509: << std::setfill(' '); tomwalters@509: os.unsetf(std::ios::basefield); tomwalters@509: } tomwalters@509: break; tomwalters@509: tomwalters@509: case MIDI_MESSAGE_TYPE_TAG: tomwalters@509: { tomwalters@509: uint32 m = arg.AsMidiMessageUnchecked(); tomwalters@509: os << "midi (port, status, data1, data2):<<" tomwalters@509: << std::hex << std::setfill('0') tomwalters@509: << "0x" << std::setw(2) << (int)((m>>24) & 0xFF) tomwalters@509: << " 0x" << std::setw(2) << (int)((m>>16) & 0xFF) tomwalters@509: << " 0x" << std::setw(2) << (int)((m>>8) & 0xFF) tomwalters@509: << " 0x" << std::setw(2) << (int)(m & 0xFF) tomwalters@509: << std::setfill(' ') << ">>"; tomwalters@509: os.unsetf(std::ios::basefield); tomwalters@509: } tomwalters@509: break; tomwalters@509: tomwalters@509: case INT64_TYPE_TAG: tomwalters@509: os << "int64:" << arg.AsInt64Unchecked(); tomwalters@509: break; tomwalters@509: tomwalters@509: case TIME_TAG_TYPE_TAG: tomwalters@509: { tomwalters@509: os << "OSC-timetag:" << arg.AsTimeTagUnchecked(); tomwalters@509: tomwalters@509: std::time_t t = tomwalters@509: (unsigned long)( arg.AsTimeTagUnchecked() >> 32 ); tomwalters@509: tomwalters@509: // strip trailing newline from string returned by ctime tomwalters@509: const char *timeString = std::ctime( &t ); tomwalters@509: size_t len = strlen( timeString ); tomwalters@509: char *s = new char[ len + 1 ]; tomwalters@509: strcpy( s, timeString ); tomwalters@509: if( len ) tomwalters@509: s[ len - 1 ] = '\0'; tomwalters@509: tomwalters@509: os << " " << s; tomwalters@509: } tomwalters@509: break; tomwalters@509: tomwalters@509: case DOUBLE_TYPE_TAG: tomwalters@509: os << "double:" << arg.AsDoubleUnchecked(); tomwalters@509: break; tomwalters@509: tomwalters@509: case STRING_TYPE_TAG: tomwalters@509: os << "OSC-string:`" << arg.AsStringUnchecked() << "'"; tomwalters@509: break; tomwalters@509: tomwalters@509: case SYMBOL_TYPE_TAG: tomwalters@509: os << "OSC-string (symbol):`" << arg.AsSymbolUnchecked() << "'"; tomwalters@509: break; tomwalters@509: tomwalters@509: case BLOB_TYPE_TAG: tomwalters@509: { tomwalters@509: unsigned long size; tomwalters@509: const void *data; tomwalters@509: arg.AsBlobUnchecked( data, size ); tomwalters@509: os << "OSC-blob:<<" << std::hex << std::setfill('0'); tomwalters@509: unsigned char *p = (unsigned char*)data; tomwalters@509: for( unsigned long i = 0; i < size; ++i ){ tomwalters@509: os << "0x" << std::setw(2) << int(p[i]); tomwalters@509: if( i != size-1 ) tomwalters@509: os << ' '; tomwalters@509: } tomwalters@509: os.unsetf(std::ios::basefield); tomwalters@509: os << ">>" << std::setfill(' '); tomwalters@509: } tomwalters@509: break; tomwalters@509: tomwalters@509: default: tomwalters@509: os << "unknown"; tomwalters@509: } tomwalters@509: tomwalters@509: return os; tomwalters@509: } tomwalters@509: tomwalters@509: tomwalters@509: std::ostream& operator<<( std::ostream & os, const ReceivedMessage& m ) tomwalters@509: { tomwalters@509: tomwalters@509: os << "[" << m.AddressPattern(); tomwalters@509: bool first = true; tomwalters@509: tomwalters@509: for( ReceivedMessage::const_iterator i = m.ArgumentsBegin(); tomwalters@509: i != m.ArgumentsEnd(); ++i ){ tomwalters@509: if( first ){ tomwalters@509: os << " "; tomwalters@509: first = false; tomwalters@509: }else{ tomwalters@509: os << ", "; tomwalters@509: } tomwalters@509: tomwalters@509: os << *i; tomwalters@509: } tomwalters@509: tomwalters@509: os << "]"; tomwalters@509: tomwalters@509: return os; tomwalters@509: } tomwalters@509: tomwalters@509: tomwalters@509: std::ostream& operator<<( std::ostream & os, const ReceivedBundle& b ) tomwalters@509: { tomwalters@509: static int indent = 0; tomwalters@509: tomwalters@509: for( int j=0; j < indent; ++j ) tomwalters@509: os << " "; tomwalters@509: os << "{ ( "; tomwalters@509: if( b.TimeTag() == 1 ) tomwalters@509: os << "immediate"; tomwalters@509: else tomwalters@509: os << b.TimeTag(); tomwalters@509: os << " )\n"; tomwalters@509: tomwalters@509: ++indent; tomwalters@509: tomwalters@509: for( ReceivedBundle::const_iterator i = b.ElementsBegin(); tomwalters@509: i != b.ElementsEnd(); ++i ){ tomwalters@509: if( i->IsBundle() ){ tomwalters@509: ReceivedBundle b(*i); tomwalters@509: os << b << "\n"; tomwalters@509: }else{ tomwalters@509: ReceivedMessage m(*i); tomwalters@509: for( int j=0; j < indent; ++j ) tomwalters@509: os << " "; tomwalters@509: os << m << "\n"; tomwalters@509: } tomwalters@509: } tomwalters@509: tomwalters@509: --indent; tomwalters@509: tomwalters@509: for( int j=0; j < indent; ++j ) tomwalters@509: os << " "; tomwalters@509: os << "}"; tomwalters@509: tomwalters@509: return os; tomwalters@509: } tomwalters@509: tomwalters@509: tomwalters@509: std::ostream& operator<<( std::ostream & os, const ReceivedPacket& p ) tomwalters@509: { tomwalters@509: if( p.IsBundle() ){ tomwalters@509: ReceivedBundle b(p); tomwalters@509: os << b << "\n"; tomwalters@509: }else{ tomwalters@509: ReceivedMessage m(p); tomwalters@509: os << m << "\n"; tomwalters@509: } tomwalters@509: tomwalters@509: return os; tomwalters@509: } tomwalters@509: tomwalters@509: } // namespace osc