-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathMessagePack.cpp
107 lines (100 loc) · 3.74 KB
/
MessagePack.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
/*
* Copyright 2009,2010,2011 Reality Jockey, Ltd.
* http://rjdj.me/
*
* This file is part of ZenGarden.
*
* ZenGarden 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 3 of the License, or
* (at your option) any later version.
*
* ZenGarden 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 ZenGarden. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "MessagePack.h"
#include "PdGraph.h"
MessageObject *MessagePack::newObject(PdMessage *initMessage, PdGraph *graph) {
return new MessagePack(initMessage, graph);
}
MessagePack::MessagePack(PdMessage *initMessage, PdGraph *graph) :
MessageObject(initMessage->getNumElements(), 1, graph) {
int numElements = initMessage->getNumElements();
PdMessage *message = PD_MESSAGE_ON_STACK(numElements);
message->initWithTimestampAndNumElements(0.0, numElements);
memcpy(message->getElement(0), initMessage->getElement(0), numElements*sizeof(MessageAtom));
message->resolveSymbolsToType();
outgoingMessage = message->copyToHeap();
}
MessagePack::~MessagePack() {
outgoingMessage->freeMessage();
}
string MessagePack::toString() {
std::string out = MessagePack::getObjectLabel();
for (int i = 0; i < outgoingMessage->getNumElements(); i++) {
switch (outgoingMessage->getType(i)) {
case FLOAT: out += " f"; break;
case SYMBOL: out += " s"; break;
case BANG: out += " b"; break;
case LIST: out += " l"; break;
case ANYTHING:
default: out += " a"; break;
}
}
return out;
}
void MessagePack::processMessage(int inletIndex, PdMessage *message) {
switch (message->getType(0)) {
case FLOAT: {
if (outgoingMessage->isFloat(inletIndex)) {
outgoingMessage->setFloat(inletIndex, message->getFloat(0));
onBangAtInlet(inletIndex, message->getTimestamp());
} else {
graph->printErr("pack: type mismatch: %s expected but got %s at inlet %i.\n",
StaticUtils::messageElementTypeToString(outgoingMessage->getType(inletIndex)),
StaticUtils::messageElementTypeToString(message->getType(0)),
inletIndex + 1);
return;
}
break;
}
case SYMBOL: {
if (outgoingMessage->isSymbol(inletIndex)) {
// NOTE(mhroth): this approach can lead to a lot of fragemented memory if symbols are
// replaced often
free(outgoingMessage->getSymbol(inletIndex)); // free the preexisting symbol on the heap
// create a new symbol on the heap and store it in the outgoing message
outgoingMessage->setSymbol(inletIndex, StaticUtils::copyString(message->getSymbol(0)));
onBangAtInlet(inletIndex, message->getTimestamp());
} else {
graph->printErr("pack: type mismatch: %s expected but got %s at inlet %i.\n",
StaticUtils::messageElementTypeToString(outgoingMessage->getType(inletIndex)),
StaticUtils::messageElementTypeToString(message->getType(0)),
inletIndex + 1);
return;
}
break;
}
case BANG: {
onBangAtInlet(inletIndex, message->getTimestamp());
break;
}
default: {
break;
}
}
}
void MessagePack::onBangAtInlet(int inletIndex, double timestamp) {
if (inletIndex == 0) {
// send the outgoing message
outgoingMessage->setTimestamp(timestamp);
sendMessage(0, outgoingMessage);
}
}