modded version CNMAT/OSC https://github.com/CNMAT/OSC

Dependents:   CVtoOSCConverter

Fork of osc-cnmat by Asperius van Hansen

OSCMessage.cpp

Committer:
casiotone401
Date:
2016-02-13
Revision:
4:107c23eb31b6
Parent:
2:61caa2495215

File content as of revision 4:107c23eb31b6:

/*
 Written by Yotam Mann, The Center for New Music and Audio Technologies,
 University of California, Berkeley.  Copyright (c) 2012, The Regents of
 the University of California (Regents).
 
 Permission to use, copy, modify, distribute, and distribute modified versions
 of this software and its documentation without fee and without a signed
 licensing agreement, is hereby granted, provided that the above copyright
 notice, this paragraph and the following two paragraphs appear in all copies,
 modifications, and distributions.
 
 IN NO EVENT SHALL REGENTS BE LIABLE TO ANY PARTY FOR DIRECT, INDIRECT,
 SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES, INCLUDING LOST PROFITS, ARISING
 OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN IF REGENTS HAS
 BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 
 REGENTS SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
 THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 PURPOSE. THE SOFTWARE AND ACCOMPANYING DOCUMENTATION, IF ANY, PROVIDED
 HEREUNDER IS PROVIDED "AS IS". REGENTS HAS NO OBLIGATION TO PROVIDE
 MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
 
 For bug reports and feature requests please email me at yotam@cnmat.berkeley.edu
 */

#include "OSCMessage.h"
#include "OSCMatch.h"

// Fast strlen function http://www.strchr.com/optimized_strlen_function
inline size_t strlengthgth(const char *s) 
{
    size_t len = 0;
    
    for (;;) 
    {
        unsigned x = *(unsigned*)s;
        if ((x & 0xFF) == 0) return len;
        if ((x & 0xFF00) == 0) return len + 1;
        if ((x & 0xFF0000) == 0) return len + 2;
        if ((x & 0xFF000000) == 0) return len + 3;
        s += 4, len += 4;
    }
}

/*=============================================================================
    CONSTRUCTORS / DESTRUCTOR
=============================================================================*/

//constructor with address
OSCMessage::OSCMessage(const char * _address){
    setupMessage();
    setAddress(_address);
}

//constructor with nothing
//just a placeholder since the message is invalid
OSCMessage::OSCMessage(){
    setupMessage();
    error = INVALID_OSC;
}

//variable length constructor
//for example OSCMessage msg("/address", "isf", 1, "two", 3.0);
/*
OSCMessage::OSCMessage(const char * _address, char * types, ... ){
    setupMessage(_address);
}
 */

//sets up a new message
void OSCMessage::setupMessage(){
    address = NULL;
    //setup the attributes
    dataCount = 0;
    error = OSC_OK;
    //setup the space for data
    data = NULL;
    //setup for filling the message
    incomingBuffer = NULL;
    incomingBufferSize = 0;
    incomingBufferFree = 0;
    clearIncomingBuffer();
    //set the decode state
    decodeState = STANDBY;
}

//DESTRUCTOR
OSCMessage::~OSCMessage(){
    //free everything that needs to be freed
    //free the address
    free(address);
    //free the data
    empty();
    //free the filling buffer
    free(incomingBuffer);
}

void OSCMessage::empty(){
    error = OSC_OK;
    //free each of hte data in the array
    for (int i = 0; i < dataCount; i++){
        OSCData * datum = getOSCData(i);
        //explicitly destruct the data
        //datum->~OSCData();
        delete datum;
    }
    //and free the array
    free(data);
    data = NULL;
    dataCount = 0;
    clearIncomingBuffer();
}

//COPY
OSCMessage::OSCMessage(OSCMessage * msg){
    //start with a message with the same address
    setupMessage();
    setAddress(msg->address);
    //add each of the data to the other message
    for (int i = 0; i < msg->dataCount; i++){
        add(msg->data[i]);
    }
}

/*=============================================================================
    GETTING DATA
=============================================================================*/

OSCData * OSCMessage::getOSCData(int position){
    if (position < dataCount){
        OSCData * datum = data[position];
        return datum;
    } else {
        error = INDEX_OUT_OF_BOUNDS;
        return NULL;
    }
}

int32_t OSCMessage::getInt(int position){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->getInt();
    } else {
        return NULL;
    }
}
uint64_t OSCMessage::getTime(int position){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->getTime();
    } else {
        return NULL;
    }
}
float OSCMessage::getFloat(int position){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->getFloat();
    } else {
        return NULL;
    }
}

double OSCMessage::getDouble(int position){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->getDouble();
    } else {
        return NULL;
    }
}

int OSCMessage::getString(int position, char * buffer, int bufferSize){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        //the number of bytes to copy is the smaller between the buffer size and the datum's byte length
        int copyBytes = bufferSize < datum->bytes? bufferSize : datum->bytes;
        return datum->getString(buffer, copyBytes);
    } else {
        return NULL;
    }
}

int OSCMessage::getBlob(int position, uint8_t * buffer, int bufferSize){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        //the number of bytes to copy is the smaller between the buffer size and the datum's byte length
        int copyBytes = bufferSize < datum->bytes? bufferSize : datum->bytes;
        return datum->getBlob(buffer, copyBytes);
    } else {
        return NULL;
    }
}

char OSCMessage::getType(int position){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->type;
    } else {
        return NULL;
    }
}

int OSCMessage::getDataLength(int position){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->bytes;
    } else {
        return 0;
    }
}

/*=============================================================================
    TESTING DATA
=============================================================================*/

bool OSCMessage::testType(int position, char type){
    OSCData * datum = getOSCData(position);
    if (!hasError()){
        return datum->type == type;
    } else {
        return false;
    }
}

bool OSCMessage::isInt(int position){
    return testType(position, 'i');
}

bool OSCMessage::isTime(int position){
    return testType(position, 't');
}

bool OSCMessage::isFloat(int position){
    return testType(position, 'f');
}

bool OSCMessage::isBlob(int position){
    return testType(position, 'b');
}

bool OSCMessage::isChar(int position){
    return testType(position, 'c');
}

bool OSCMessage::isString(int position){
    return testType(position, 's');
}

bool OSCMessage::isDouble(int position){
    return testType(position, 'd');
}
bool OSCMessage::isBoolean(int position){
    return testType(position, 'T') || testType(position, 'F');
}


/*=============================================================================
    PATTERN MATCHING
=============================================================================*/

int OSCMessage::match(const  char * pattern, int addr_offset){
    int pattern_offset;
    int address_offset;
    int ret = osc_match(address + addr_offset, pattern, &pattern_offset, &address_offset);
    char * next = (char *) (address + addr_offset + pattern_offset);
    if (ret==3){
        return pattern_offset;
    } else if (pattern_offset > 0 && *next == '/'){
        return pattern_offset;
    } else {
        return 0;
    }
}

bool OSCMessage::fullMatch( const char * pattern, int addr_offset){
    int pattern_offset;
    int address_offset;
    int ret = osc_match(address + addr_offset, pattern, &address_offset, &pattern_offset);
    return (ret==3);
}

bool OSCMessage::dispatch(const char * pattern, void (*callback)(OSCMessage &), int addr_offset){
    if (fullMatch(pattern, addr_offset)){
        callback(*this);
        return true;
    } else {
        return false;
    }
}

bool OSCMessage::route(const char * pattern, void (*callback)(OSCMessage &, int), int initial_offset){
    int match_offset = match(pattern, initial_offset);
    if (match_offset>0){
        callback(*this, match_offset + initial_offset);
        return true;
    } else {
        return false;
    }
}

/*=============================================================================
    ADDRESS
 =============================================================================*/

int OSCMessage::getAddress(char * buffer, int offset){
    strcpy(buffer, address+offset);
    return strlen(buffer);
}

int OSCMessage::getAddress(char * buffer, int offset, int len){
    strncpy(buffer, address+offset, len);
    return strlen(buffer);
}

void OSCMessage::setAddress(const char * _address){
    //free the previous address
    free(address); // are we sure address was allocated?
    //copy the address
    char * addressMemory = (char *) malloc( (strlen(_address) + 1) * sizeof(char) );
    if (addressMemory == NULL){
        error = ALLOCFAILED;
        address = NULL;
    } else {
        strcpy(addressMemory, _address);
        address = addressMemory;
    }
}

/*=============================================================================
    SIZE
=============================================================================*/

#ifdef SLOWpadcalculation
int OSCMessage::padSize(int _bytes){
    int space = (_bytes + 3) / 4;
    space *= 4;
    return space - _bytes;
}
#else
static inline  int padSize(int bytes) { return (4- (bytes&03))&3; }
#endif
//returns the number of OSCData in the OSCMessage
int OSCMessage::size(){
    return dataCount;
}

int OSCMessage::bytes(){
    int messageSize = 0;
    //send the address
    int addrLen = strlen(address) + 1;
    messageSize += addrLen;
    //padding amount
    int addrPad = padSize(addrLen);
    messageSize += addrPad;
    //add the comma seperator
    messageSize += 1;
    //add the types
    messageSize += dataCount;
    //pad the types
    int typePad = padSize(dataCount + 1);   //for the comma
    if (typePad == 0){
         typePad = 4; // to make sure the type string is null terminated
    }
    messageSize += typePad;
    //then the data
    for (int i = 0; i < dataCount; i++){
        OSCData * datum = getOSCData(i);
        messageSize += datum->bytes;
        messageSize += padSize(datum->bytes);
    }
    return messageSize;
}

/*=============================================================================
    ERROR HANDLING
=============================================================================*/

bool OSCMessage::hasError(){
    bool retError = error != OSC_OK;
    //test each of the data
    for (int i = 0; i < dataCount; i++){
        OSCData * datum = getOSCData(i);
        retError |= datum->error != OSC_OK;
    }
    return retError;
}

OSCErrorCode OSCMessage::getError(){
    return error;
}

/*=============================================================================
    SENDING
 =============================================================================*/


/*void OSCMessage::send(Stream &p){
    //don't send a message with errors
    if (hasError()){
        return;
    }
    uint8_t nullChar = '\0';
    //send the address
    int addrLen = strlen(address) + 1;
    //padding amount
    int addrPad = padSize(addrLen);
    //write it to the stream
    p.write((uint8_t *) address, addrLen);
    //add the padding
    while(addrPad--){
        p.write(nullChar);
    }
    //add the comma seperator
    p.write((uint8_t) ',');
    //add the types
#ifdef PAULSSUGGESTION
    // Paul suggested buffering on the stack
    // to improve performance. The problem is this could exhaust the stack
    // for long complex messages
    {
        uint8_t typstr[dataCount];
    
        for (int i = 0; i < dataCount; i++){
            typstr[i] =  getType(i);
        }
        p.write(typstr,dataCount);
    }
#else
    for (int i = 0; i < dataCount; i++){
        p.write((uint8_t) getType(i));
    }
#endif
    //pad the types
    int typePad = padSize(dataCount + 1); // 1 is for the comma
    if (typePad == 0){
            typePad = 4;  // This is because the type string has to be null terminated
    }
    while(typePad--){
        p.write(nullChar);
    }
    //write the data
    for (int i = 0; i < dataCount; i++){
        OSCData * datum = getOSCData(i);
        if ((datum->type == 's') || (datum->type == 'b')){
            p.write(datum->data.b, datum->bytes);
            int dataPad = padSize(datum->bytes);
            while(dataPad--){
                p.write(nullChar);
            }
        } else if(datum->type == 'b'){
            p.write(datum->data.b, datum->bytes);
            int dataPad = padSize(datum->bytes);
            while(dataPad--){
                p.write(nullChar);
            }
        } else if (datum->type == 'd'){
            double d = BigEndian(datum->data.d);
            uint8_t * ptr = (uint8_t *) &d;
            p.write(ptr, 8);
        } else if (datum->type == 't'){
            uint64_t d = BigEndian(datum->data.l);
            uint8_t * ptr = (uint8_t *)    &d;
            p.write(ptr, 8);

        } else if (datum->type == 'T' || datum->type == 'F')
                    { }
        else { // float or int
            uint32_t i = BigEndian(datum->data.i);
            uint8_t * ptr = (uint8_t *) &i;
            p.write(ptr, datum->bytes);
        }
    }
}*/


int OSCMessage::send(uint8_t * messageBuffer){
    int i, j;
    //don't send a message with errors
    if (hasError()){
        return -1;
    }
    uint8_t nullChar = '\0';
    //send the address
    int addrLen = strlen(address) + 1;
    //padding amount
    int addrPad = padSize(addrLen);
    //write it to the stream
    //p.write((uint8_t *) address, addrLen);
    int addrIdx = 0;
    int idx = 0 + bundleOffset;
    
    while(addrLen--) {
        messageBuffer[idx++] = address[addrIdx++];
    }
    //add the padding
    while(addrPad--){
        //p.write(nullChar);
        messageBuffer[idx++] = nullChar;
    }
    //add the comma seperator
    //p.write((uint8_t) ',');
    messageBuffer[idx++] = ',';
    //add the types
#ifdef PAULSSUGGESTION
    // Paul suggested buffering on the stack
    // to improve performance. The problem is this could exhaust the stack
    // for long complex messages
    {
        //uint8_t typstr[dataCount];
    
        for (i = 0; i < dataCount; i++){
            //typstr[i] =  getType(i);
            messageBuffer[idx++] = getType(i);
        }
        //p.write(typstr,dataCount);
    }
#else
    for (i = 0; i < dataCount; i++){
        //p.write((uint8_t) getType(i));
        messageBuffer[idx++] = getType(i);
    }
#endif
    //pad the types
    int typePad = padSize(dataCount + 1); // 1 is for the comma
    
    if (typePad == 0) {
        typePad = 4;  // This is because the type string has to be null terminated
    }
    
    while(typePad--){
        //p.write(nullChar);
        messageBuffer[idx++] = nullChar;
    }
    //write the data
    for (i = 0; i < dataCount; i++) {
        
        OSCData * datum = getOSCData(i);
        
        if ((datum->type == 's') || (datum->type == 'b')){
            //p.write(datum->data.b, datum->bytes);
            for (j = 0; j < datum->bytes; j++) {
                messageBuffer[idx++] = datum->data.b[j];
            }

            int dataPad = padSize(datum->bytes);
            while(dataPad--){
                //p.write(nullChar);
                messageBuffer[idx++] = nullChar;
            }
        } else if (datum->type == 'd'){
            double d = BigEndian(datum->data.d);
            //uint8_t * ptr = (uint8_t *) &d;
            uint8_t * ptr = (uint8_t *) &d;
            //p.write(ptr, 8);
            for (j = 0; j < 8; j++) {
                messageBuffer[idx++] = ptr[j];
            }
        } else if (datum->type == 't'){
            uint64_t d = BigEndian(datum->data.l);
            //uint8_t * ptr = (uint8_t *) &d;
            uint8_t * ptr = (uint8_t *) &d;
            //p.write(ptr, 8);
            for (j = 0; j < 8; j++) {
                messageBuffer[idx++] = ptr[j];
            }
        } else if (datum->type == 'T' || datum->type == 'F') { 
        
        } else { // float or int
            //uint32_t i = BigEndian(datum->data.i);
            uint32_t ui = BigEndian(datum->data.i);
            //uint8_t * ptr = (uint8_t *) &i;
            uint8_t * ptr = (uint8_t *) &ui;
            //p.write(ptr, datum->bytes);
            for (j = 0; j < datum->bytes; j++) {
                messageBuffer[idx++] = ptr[j];
            }
        }
    }
    
    bundleOffset = 0;
    return idx;
}

/*=============================================================================
    FILLING
 =============================================================================*/

void OSCMessage::fill(uint8_t incomingByte){
    decode(incomingByte);
}
/*
void OSCMessage::fill(uint8_t * incomingBytes, int length){
    while (length--){
        decode(*incomingBytes++);
    }
}
*/
/*=============================================================================
    DECODING
 =============================================================================*/

void OSCMessage::decodeAddress(){
    setAddress((char *) incomingBuffer);
    //change the error from invalide message
    error = OSC_OK;
    clearIncomingBuffer();
}

void OSCMessage::decodeType(uint8_t incomingByte){
    char type = incomingByte;
    add(type);
}

void OSCMessage::decodeData(uint8_t incomingByte){
    //get the first OSCData to re-set
    for (int i = 0; i < dataCount; i++){
        OSCData * datum = getOSCData(i);
        if (datum->error == INVALID_OSC){
            //set the contents of datum with the data received
            switch (datum->type){
                case 'i':
                    if (incomingBufferSize == 4){
                        //parse the buffer as an int
                        union {
                            int32_t i;
                            uint8_t b[4];
                        } u;
                        memcpy(u.b, incomingBuffer, 4);
                        int32_t dataVal = BigEndian(u.i);
                        set(i, dataVal);
                        clearIncomingBuffer();
                    }
                    break;
                case 'f':
                    if (incomingBufferSize == 4){
                        //parse the buffer as an int
                        union {
                            float f;
                            uint8_t b[4];
                        } u;
                        memcpy(u.b, incomingBuffer, 4);
                        float dataVal = BigEndian(u.f);
                        set(i, dataVal);
                        clearIncomingBuffer();
                    }
                    break;
                case 'd':
                    if (incomingBufferSize == 8){
                        //parse the buffer as an int
                        union {
                            double d;
                            uint8_t b[8];
                        } u;
                        memcpy(u.b, incomingBuffer, 8);
                        double dataVal = BigEndian(u.d);
                        set(i, dataVal);
                        clearIncomingBuffer();
                    }
                    break;
                case 't':
                    if (incomingBufferSize == 8){
                        //parse the buffer as an int
                        union {
                            uint64_t d;
                            uint8_t b[8];
                        } u;
                        memcpy(u.b, incomingBuffer, 8);
                         uint64_t dataVal = BigEndian(u.d);
                        set(i, dataVal);
                        clearIncomingBuffer();
                    }
                    break;

                case 's':
                    if (incomingByte == 0){
                        char * str = (char *) incomingBuffer;
                        set(i, str);
                        clearIncomingBuffer();
                        decodeState = DATA_PADDING;
                    }
                    break;
                case 'b':
                    if (incomingBufferSize > 4){
                        //compute the expected blob size
                        union {
                            uint32_t i;
                            uint8_t b[4];
                        } u;
                        memcpy(u.b, incomingBuffer, 4);
                        uint32_t blobLength = BigEndian(u.i);
                        if (incomingBufferSize == blobLength + 4){
                            set(i, incomingBuffer + 4, blobLength);
                            clearIncomingBuffer();
                            decodeState = DATA_PADDING;
                        }
                        
                    }
                    break;
            }
            //break out of the for loop once we've selected the first invalid message
            break;
        } 
    }

}

//does not validate the incoming OSC for correctness
void OSCMessage::decode(uint8_t incomingByte){
    addToIncomingBuffer(incomingByte);
    switch (decodeState){
        case STANDBY:
            if (incomingByte == '/'){
                decodeState = ADDRESS;
            }
            break;
        case ADDRESS:
            if (incomingByte == 0){
                //end of the address
                //decode the address
                decodeAddress();
                //next state
                decodeState = ADDRESS_PADDING;
            } 
            break;
        case ADDRESS_PADDING:
            //it does not count the padding
            if (incomingByte==','){
                //next state
                decodeState = TYPES;
                clearIncomingBuffer();
            }
            break;
        case TYPES:
            if (incomingByte != 0){
                //next state
                decodeType(incomingByte);
            } else {
                decodeState = TYPES_PADDING;
            }
            //FALL THROUGH to test if it should go to the data state
        case TYPES_PADDING: {
                //compute the padding size for the types
                //to determine the start of the data section
            int typePad = padSize(dataCount + 1); // 1 is the comma
                if (typePad == 0){
                    typePad = 4;     // to make sure it will be null terminated
                }
                if (incomingBufferSize == (typePad + dataCount)){
                    clearIncomingBuffer();
                    decodeState = DATA;
                }
            }
            break;
        case DATA:
            decodeData(incomingByte);
            break;
        case DATA_PADDING:{
                //get hte last valid data
                for (int i = dataCount - 1; i >= 0; i--){
                    OSCData * datum = getOSCData(i);
                    if (datum->error == OSC_OK){
                        //compute the padding size for the data
                        int dataPad = padSize(datum->bytes);
                        if (incomingBufferSize == dataPad){
                            clearIncomingBuffer();
                            decodeState = DATA;
                        }
                        break;
                    }
                }
            }
            break;

    }
}


/*=============================================================================
    INCOMING BUFFER MANAGEMENT
 =============================================================================*/
#define OSCPREALLOCATEIZE 16
void OSCMessage::addToIncomingBuffer(uint8_t incomingByte){
    //realloc some space for the new byte and stick it on the end
    if(incomingBufferFree>0)
    {
            incomingBuffer[incomingBufferSize++] = incomingByte;
            incomingBufferFree--;
    }
    else
    {

        incomingBuffer = (uint8_t *) realloc ( incomingBuffer, incomingBufferSize + 1 + OSCPREALLOCATEIZE);
        if (incomingBuffer != NULL){
            incomingBuffer[incomingBufferSize++] = incomingByte;
            incomingBufferFree = OSCPREALLOCATEIZE;
        } else {
            error = ALLOCFAILED;
        }
    }
}

void OSCMessage::clearIncomingBuffer(){
    incomingBuffer = (uint8_t *) realloc ( incomingBuffer, OSCPREALLOCATEIZE);
    if (incomingBuffer != NULL){
        incomingBufferFree = OSCPREALLOCATEIZE;
    } else {
        error = ALLOCFAILED;
        incomingBuffer = NULL;

    }    
    incomingBufferSize = 0;
}