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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
/*
 *  Copyright 2010,2011,2012 Reality Jockey, Ltd.
 *                 info@rjdj.me
 *                 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/>.
 *
 */
#ifndef _ZENGARDEN_H_
#define _ZENGARDEN_H_
#include "ZGCallbackFunction.h"
/**
 * This header file defines the C interface between ZenGarden and the outside world. Include this header
 * along with the <code>libzengarden</code> library in your project in order to integrate it.
 */
#ifdef __cplusplus
class pd::Context;
class PdGraph;
class MessageObject;
class PdMessage;
typedef pd::Context ZGContext;
typedef PdGraph ZGGraph;
typedef MessageObject ZGObject;
typedef PdMessage ZGMessage;
extern "C" {
#else
typedef void ZGGraph;
typedef void ZGContext;
typedef void ZGObject;
typedef void ZGMessage;
#endif
typedef struct ZGConnectionPair {
  ZGObject *object;
  unsigned int letIndex;
} ZGConnectionPair;
/** Enumerates the types of elements of which messages are composed. */
typedef enum ZGMessageElementType {
  ZG_MESSAGE_ELEMENT_FLOAT,
  ZG_MESSAGE_ELEMENT_SYMBOL,
  ZG_MESSAGE_ELEMENT_BANG
} ZGMessageElementType;
/**
 * A pointer to this structure is supplied in the callback function with ZG_RECEIVER_MESSAGE.
 * The structure will not persistent after the function returns and the pointer will become invalid.
 */
typedef struct ZGReceiverMessagePair {
  const char *receiver_name;
  ZGMessage *message;
} ZGReceiverMessagePair;
/** Enumerates the kinds of connections in ZenGarden; Message and DSP */
typedef enum ZGConnectionType {
  ZG_CONNECTION_MESSAGE,
  ZG_CONNECTION_DSP
} ZGConnectionType;
#pragma mark - Context
  /** Create a new context to which graphs can be added. */
  ZGContext *zg_context_new(int num_input_channels, int num_output_channels, int block_size, float sample_rate,
      void *(*callback_function)(ZGCallbackFunction function, void *userData, void *ptr), void *userData);
  /** Create a new empty graph in the given context. Ideal for building graphs on the fly. */
  ZGGraph *zg_context_new_empty_graph(ZGContext *context);
  /** Create a new graph from a Pd file. */
  ZGGraph *zg_context_new_graph_from_file(ZGContext *context, const char *directory, const char *filename);
  /** Create a new graph based on a string representation of the netlist. */
  ZGGraph *zg_context_new_graph_from_string(ZGContext *context, const char *netlist);
  /** Remove the graph from the context. */
  //void zg_remove_graph(ZGContext *context, ZGGraph *graph);
  /**
   * Delete the given context. All attached graphs are also deleted. Unattached graphs are not
   * automatically deleted, but should be by the user. They are thereafter useless.
   */
  void zg_context_delete(ZGContext *context);
  /** Returns the userinfo pointer used with the callback function. */
  void *zg_context_get_userinfo(ZGContext *context);
  /**
   * Returns all root graphs attached to this context. The returned array, with length n, must
   * be freed by the caller.
   */
  ZGGraph *zg_context_get_graphs(ZGContext *context, unsigned int *n);
  /**
   * Register an external such that the context can instantiate instances of it. If an object
   * with the same label already exists, then the factory method is replaced with the new one.
   */
  void zg_context_register_external_object(ZGContext *context, const char *object_label,
      ZGObject *(*factory)(ZGMessage *message, ZGGraph *graph));
  /** Unregister an external such that the context will be unaware of it. */
  void zg_context_unregister_external_object(ZGContext *context, const char *object_label);
#pragma mark - Abstractions from Context
  /** Register an abstraction from memory. */
  void zg_context_register_memorymapped_abstraction(ZGContext *context, const char *object_label,
      const char *abstraction);
  /** Unregister an abstraction. */
  void zg_context_unregister_memorymapped_abstraction(ZGContext *context, const char *object_label);
#pragma mark - Objects from Context
  /** Returns the global table object with the given name. NULL if the table does not exist. */
  ZGObject *zg_context_get_table_for_name(ZGObject *table, const char *name);
#pragma mark - Graph
  /** Deletes the given graph. If attached, the graph is automatically removed from its context. */
  void zg_graph_delete(ZGGraph *graph);
  /** Returns the $0 argument to a graph, allowing graph-specific receivers to be addressed. */
  unsigned int zg_graph_get_dollar_zero(ZGGraph *graph);
  /** Attaches a graph to its context */
  void zg_graph_attach(ZGGraph *graph);
  /** Unattaches a graph to its context */
  void zg_graph_unattach(ZGGraph *graph);
  /** Returns all objects in this graph. The returned array, with length n, must be freed by the caller. */
  ZGObject **zg_graph_get_objects(ZGGraph *graph, unsigned int *n);
#pragma mark - Manage Connections
  /**
   * Add a connection between two objects, both of which are in the given graph. The new connection
   * may cause the object graph to be reordered and cause audio dropouts. If the arguments do
   * not define a valid connection, then this function does nothing.
   */
  void zg_graph_add_connection(ZGGraph *graph, ZGObject *fromObject, int outlet_index, ZGObject *toObject, int inlet_index);
  /**
   * Remove a connection between two objects, both of which are in the given graph. If the arguments
   * do not define a valid connection, then this function does nothing.
   */
  void zg_graph_remove_connection(ZGGraph *graph, ZGObject *fromObject, int outlet_index, ZGObject *toObject, int inlet_index);
  /** Returns the connection type of the outlet of the given object. */
  ZGConnectionType zg_object_get_connection_type(ZGObject *object, unsigned int outlet_index);
  unsigned int zg_object_get_num_inlets(ZGObject *object);
  unsigned int zg_object_get_num_outlets(ZGObject *object);
#pragma mark - Context Process
  /** Process the given context. Audio buffers are channel-uninterleaved with float (32-bit) samples. */
  void zg_context_process(ZGContext *context, float *input_buffers, float *output_buffers);
  /** Process the given context. Audio buffers are channel-interleaved with signed short (16-bit) samples. */
  void zg_context_process_s(ZGContext *context, short *input_buffers, short *output_buffers);
#pragma mark - Context Send Message
  /** Send a message to the named receiver. */
  void zg_context_send_message(ZGContext *context, const char *receiver_name, ZGMessage *message);
  /** Send a message described by the <code>initString</code> to the named receiver at the given timestamp. */
  void zg_context_send_message_from_string(ZGContext *context, const char *receiver_name,
      double timestamp, const char *initString);
  /**
   * Send a message to the named receiver with the given format at the beginning of the next audio block.
   * If no receiver exists with the given name, then this funtion does nothing.
   * E.g., zg_send_message(graph, "#accelerate", "fff", 0.0f, 0.0f, 0.0f);
   * sends a message containing three floats, each with value 0.0f, to all receivers named "#accelerate".
   * Messages may also be formatted with "s" and "b" for symbols and bangs, respectively.
   * E.g., zg_send_message(graph, "test", "s", "hello");
   * E.g., zg_send_message(graph, "test", "b");
   */
  void zg_context_send_messageV(ZGContext *context, const char *receiver_name, double timestamp,
      const char *messageFormat, ...);
  /**
   * Send a message to the named receiver with the given format at the given block index. If the
   * block index is negative or greater than the block size (given when instantiating the graph)
   * the the message will be sent at the very beginning of the next block. A fractional block index
   * may be given, and the message will be evaluated between rendered samples. If the given block
   * index falls outside of the block size (either positive or negative), then the message will be
   * delivered at the beginning of the block. If no receiver exists with the given name, then this
   * funtion does nothing.
   * This function is equivalent to e.g., zg_send_message(graph, "#accelerate", 0.0, "fff", 0.0f, 0.0f, 0.0f)
   * E.g., zg_send_message_at_blockindex(graph, "#accelerate", 56.3, "fff", 0.0f, 0.0f, 0.0f);
   * sends a message containing three floats, each with value 0.0f, to all receivers named "#accelerate"
   * between samples 56th and 57th samples (counting from zero) of the block.
   */
  void zg_context_send_message_at_blockindex(ZGContext *context, const char *receiver_name,
      double blockIndex, const char *messageFormat, ...);
  /**
   * Send a midi note message on the given channel to occur at the given block index. The
   * <code>blockIndex</code> parameter behaves in the same way as in <code>zg_send_message_at_blockindex()</code>.
   * All messages are sent to <code>notein</code> objects, i.e. omni. Channels are zero-index and only
   * 16 are supported. A note off message is generally interpreted as having velocity zero.
   */
  void zg_context_send_midinote(ZGContext *context, int channel, int noteNumber, int velocity, double blockIndex);
#pragma mark - Context Un/Register External Receivers
  void zg_context_register_receiver(ZGContext *context, const char *receiver_name);
  void zg_context_unregister_receiver(ZGContext *context, const char *receiver_name);
#pragma mark - Object
  /**
   * Create a new object with a string, e.g. "osc~ 440", "+", or "pack t t s, and add it to the graph.
   * If the graph is currently attached then audio may be interrupted
   * while the object is attached the and graph reconfigured (if necessary). If the graph is unattached
   * then no audio interruption will take place, even if reconfiguration takes place. The canvasX
   * and canvasY arguments specify the canvas location of the object. This is only relevant for
   * input/~ and output/~ objects, otherwise 0 may be specified.
   */
  ZGObject *zg_graph_add_new_object(ZGGraph *graph, const char *objectString, float canvasX, float canvasY);
  /**
   * Removes the object from the graph and deletes it from memory. Any connections that this object
   * may have had in the graph are also deleted. The reference to the object after this function
   * completes is invalid.
   */
  void zg_object_remove(ZGObject *object);
  /**
   * Returns an array of ZGConnectionPair structs indicating the objects and outlets from which
   * the connections are comming. The result in n is the length of the array (i.e. the number of
   * connections at the given inlet). The returned array is owned and must be freed by the caller.
   */
  ZGConnectionPair *zg_object_get_connections_at_inlet(ZGObject *object, unsigned int inlet_index, unsigned int *n);
  ZGConnectionPair *zg_object_get_connections_at_outlet(ZGObject *object, unsigned int outlet_index, unsigned int *n);
  /**
   * Send a message directly to an object. The message will be evaluated at the beginning of the
   * next block, before any other messages otherwise scheduled are evaluated. The timestamp of
   * this message is ignored. If the message should be delivered at a specific time, use
   * zg_context_send_message() and its variants in order to send the message to a named receiver.
   */
  void zg_object_send_message(ZGObject *object, unsigned int inlet_index, ZGMessage *message);
  /**
   * Returns the canvas position of the object. Position coordinates are represented as floats
   * and are real valued, though Pd uses only non-negative values.
   */
  Coordinates zg_object_get_canvas_position(ZGObject *object);
  /** Sets the canvas position of the object. Coordinates may be positive or negative. */
  void zg_object_set_canvas_position(ZGObject *object, Coordinates coordinates);
  /** Returns the object label, e.g. "osc~" or "+". */
  const char *zg_object_get_label(ZGObject *object);
  /**
   * Returns a canonical string description of the object. The description reflects the state of the
   * object when at the time of the request. The pointer must be freed by the caller.
   * For example, "osc~ 440" or "trigger b b".
   */
  char *zg_object_to_string(ZGObject *object);
#pragma mark - Table
  /**
   * Returns a direct pointer to the table's buffer with a given length. Note that if elements
   * of the buffer are modified while the context is being processed, a race condition may occur
   * between the timing of the write and the read by zg_context_process().
   */
  float *zg_table_get_buffer(ZGObject *table, unsigned int *n);
  /**
   * The table's buffer is resized and copied from the given buffer. This set operation is thread-safe
   * especially with regards to zg_context_process().
   */
  void zg_table_set_buffer(ZGObject *table, float *buffer, unsigned int n);
#pragma mark - Message
  /** Returns a new message with the given timestamp and the number of elements. */
  ZGMessage *zg_message_new(double timetamp, unsigned int numElemets);
  /** Returns a new message based on the given string. */
  ZGMessage *zg_message_new_from_string(double timetamp, const char *init_message);
  /** The message is released from memory. */
  void zg_message_delete(ZGMessage *message);
  void zg_message_set_float(ZGMessage *message, unsigned int index, float f);
  /** The symbol parameter is copied into the message. Any previous symbol is freed from memory. */
  void zg_message_set_symbol(ZGMessage *message, unsigned int index, const char *s);
  void zg_message_set_bang(ZGMessage *message, unsigned int index);
  unsigned int zg_message_get_num_elements(ZGMessage *message);
  double zg_message_get_timestamp(ZGMessage *message);
  ZGMessageElementType zg_message_get_element_type(ZGMessage *message, unsigned int index);
  float zg_message_get_float(ZGMessage *message, unsigned int index);
  const char *zg_message_get_symbol(ZGMessage *message, unsigned int index);
  /** Returns a string representation of the message. The string must be freed by the caller. */
  char *zg_message_to_string(ZGMessage *message);
#ifdef __cplusplus
}
#endif
#endif // _ZENGARDEN_H_