/* * Copyright (C) 2004 Steve Harris * * This program 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 2.1 of the * License, or (at your option) any later version. * * This program 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. * * $Id$ */ #include #include #include #include "lo_types_internal.h" #include "lo.h" lo_bundle lo_bundle_new(lo_timetag tt) { lo_bundle b = calloc(1, sizeof(struct _lo_bundle)); b->size = 4; b->len = 0; b->ts = tt; b->msgs = calloc(b->size, sizeof(lo_message)); b->paths = calloc(b->size, sizeof(char *)); return b; } int lo_bundle_add_message(lo_bundle b, const char *path, lo_message m) { if (!m) return 0; if (b->len >= b->size) { b->size *= 2; b->msgs = realloc(b->msgs, b->size * sizeof(lo_message)); b->paths = realloc(b->paths, b->size * sizeof(char *)); if (!b->msgs || !b->paths) return -1; } b->msgs[b->len] = m; b->paths[b->len] = (char *)path; (b->len)++; return 0; } size_t lo_bundle_length(lo_bundle b) { size_t size = 16; /* "#bundle" and the timetag */ int i; if (!b) { return 0; } size += b->len * 4; /* sizes */ for (i = 0; i < b->len; i++) { size += lo_message_length(b->msgs[i], b->paths[i]); } return size; } void *lo_bundle_serialise(lo_bundle b, void *to, size_t *size) { size_t s, skip; int32_t *bes; int i; char *pos; lo_pcast32 be; if (!b) { return NULL; } s = lo_bundle_length(b); if (size) { *size = s; } if (!to) { to = calloc(1, s); } pos = to; strcpy(pos, "#bundle"); pos += 8; be.nl = lo_htoo32(b->ts.sec); memcpy(pos, &be, 4); pos += 4; be.nl = lo_htoo32(b->ts.frac); memcpy(pos, &be, 4); pos += 4; for (i = 0; i < b->len; i++) { lo_message_serialise(b->msgs[i], b->paths[i], pos + 4, &skip); bes = (int32_t *)pos; *bes = lo_htoo32(skip); pos += skip + 4; if (pos > (char *)to + s) { fprintf(stderr, "liblo: data integrity error at message %d\n", i); return NULL; } } if (pos != (char*)to + s) { fprintf(stderr, "liblo: data integrity error\n"); return NULL; } return to; } void lo_bundle_free(lo_bundle b) { if (!b) { return; } free(b->msgs); free(b->paths); free(b); } static int _lo_internal_compare_ptrs( const void* a, const void* b ) { if (*(void**)a < *(void**)b) return -1; if (*(void**)a == *(void**)b) return 0; return 1; } void lo_bundle_free_messages(lo_bundle b) { int i; lo_message tmp = 0; if (!b) return; /* avoid freeing the same message twice */ if (b->len > 2) qsort(b->msgs, b->len, sizeof(lo_message*), _lo_internal_compare_ptrs); for(i = 0; i < b->len; i++) { if (b->msgs[i] != tmp) { tmp = b->msgs[i]; lo_message_free(b->msgs[i]); } } free(b->msgs); free(b->paths); free(b); } void lo_bundle_pp(lo_bundle b) { int i; if (!b) return; printf("bundle(%f):\n", (double)b->ts.sec + b->ts.frac / 4294967296.0); for (i = 0; i < b->len; i++) { lo_message_pp(b->msgs[i]); } printf("\n"); } /* vi:set ts=8 sts=4 sw=4: */