annotate projects/heavy/hello-world/ControlBinop.c @ 163:20b52283c7b4 heavy-updated

- added circular buffer pd/heavy example (works but process needs to be killed manually if launched via ssh?)
author chnrx <chris.heinrichs@gmail.com>
date Thu, 12 Nov 2015 15:55:30 +0000
parents 5bcf04234f80
children
rev   line source
chris@160 1 /**
chris@160 2 * Copyright (c) 2014, 2015, Enzien Audio Ltd.
chris@160 3 *
chris@160 4 * Permission to use, copy, modify, and/or distribute this software for any
chris@160 5 * purpose with or without fee is hereby granted, provided that the above
chris@160 6 * copyright notice and this permission notice appear in all copies.
chris@160 7 *
chris@160 8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
chris@160 9 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
chris@160 10 * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
chris@160 11 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
chris@160 12 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
chris@160 13 * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
chris@160 14 * PERFORMANCE OF THIS SOFTWARE.
chris@160 15 */
chris@160 16
chris@160 17 #include "ControlBinop.h"
chris@160 18
chris@160 19 hv_size_t cBinop_init(ControlBinop *o, float k) {
chris@160 20 o->k = k;
chris@160 21 return 0;
chris@160 22 }
chris@160 23
chris@160 24 static float cBinop_perform_op(BinopType op, float f, const float k) {
chris@160 25 switch (op) {
chris@160 26 case HV_BINOP_ADD: return f + k;
chris@160 27 case HV_BINOP_SUBTRACT: return f - k;
chris@160 28 case HV_BINOP_MULTIPLY: return f * k;
chris@160 29 case HV_BINOP_DIVIDE: return (k != 0.0f) ? f / k : 0.0f;
chris@160 30 case HV_BINOP_INT_DIV: return (float) ((int) f / (int) k);
chris@160 31 case HV_BINOP_MOD_BIPOLAR: return (float) ((int) f % (int) k);
chris@160 32 case HV_BINOP_MOD_UNIPOLAR: {
chris@160 33 f = (k == 0.0f) ? 0.0f : (float) ((int) f % (int) k);
chris@160 34 return (f < 0.0f) ? f + fabsf(k) : f;
chris@160 35 }
chris@160 36 case HV_BINOP_BIT_LEFTSHIFT: return (float) (((int) f) << ((int) k));
chris@160 37 case HV_BINOP_BIT_RIGHTSHIFT: return (float) (((int) f) >> ((int) k));
chris@160 38 case HV_BINOP_BIT_AND: return (float) ((int) f & (int) k);
chris@160 39 case HV_BINOP_BIT_XOR: return (float) ((int) f ^ (int) k);
chris@160 40 case HV_BINOP_BIT_OR: return (float) ((int) f | (int) k);
chris@160 41 case HV_BINOP_EQ: return (f == k) ? 1.0f : 0.0f;
chris@160 42 case HV_BINOP_NEQ: return (f != k) ? 1.0f : 0.0f;
chris@160 43 case HV_BINOP_LOGICAL_AND: return ((f == 0.0f) || (k == 0.0f)) ? 0.0f : 1.0f;
chris@160 44 case HV_BINOP_LOGICAL_OR: return ((f == 0.0f) && (k == 0.0f)) ? 0.0f : 1.0f;
chris@160 45 case HV_BINOP_LESS_THAN: return (f < k) ? 1.0f : 0.0f;
chris@160 46 case HV_BINOP_LESS_THAN_EQL: return (f <= k) ? 1.0f : 0.0f;
chris@160 47 case HV_BINOP_GREATER_THAN: return (f > k) ? 1.0f : 0.0f;
chris@160 48 case HV_BINOP_GREATER_THAN_EQL: return (f >= k) ? 1.0f : 0.0f;
chris@160 49 case HV_BINOP_MAX: return hv_max_f(f, k);
chris@160 50 case HV_BINOP_MIN: return hv_min_f(f, k);
chris@160 51 case HV_BINOP_POW: return (f > 0.0f) ? powf(f, k) : 0.0f;
chris@160 52 case HV_BINOP_ATAN2: return ((f == 0.0f) && (k == 0.0f)) ? 0.0f : atan2f(f, k);
chris@160 53 default: return 0.0f;
chris@160 54 }
chris@160 55 }
chris@160 56
chris@160 57 void cBinop_onMessage(HvBase *_c, ControlBinop *o, BinopType op, int letIn,
chris@160 58 const HvMessage *const m,
chris@160 59 void (*sendMessage)(HvBase *, int, const HvMessage *const)) {
chris@160 60 switch (letIn) {
chris@160 61 case 0: {
chris@160 62 if (msg_isFloat(m, 0)) {
chris@160 63 // Note(joe): supporting Pd's ability to perform operations of packs
chris@160 64 // of floats is likely to not be supported in the future.
chris@160 65 if (msg_isFloat(m, 1)) o->k = msg_getFloat(m, 1);
chris@160 66 HvMessage *n = HV_MESSAGE_ON_STACK(1);
chris@160 67 float f = cBinop_perform_op(op, msg_getFloat(m, 0), o->k);
chris@160 68 msg_initWithFloat(n, msg_getTimestamp(m), f);
chris@160 69 sendMessage(_c, 0, n);
chris@160 70 }
chris@160 71 break;
chris@160 72 }
chris@160 73 case 1: {
chris@160 74 if (msg_isFloat(m, 0)) {
chris@160 75 o->k = msg_getFloat(m, 0);
chris@160 76 }
chris@160 77 break;
chris@160 78 }
chris@160 79 default: break;
chris@160 80 }
chris@160 81 }
chris@160 82
chris@160 83 void cBinop_k_onMessage(HvBase *_c, void *o, BinopType op, const float k,
chris@160 84 int letIn, const HvMessage *const m,
chris@160 85 void (*sendMessage)(HvBase *, int, const HvMessage *const)) {
chris@160 86 if (msg_isFloat(m, 0)) {
chris@160 87 // NOTE(mhroth): Heavy does not support sending bangs to binop objects to return the previous output
chris@160 88 float f = (msg_isFloat(m, 1)) ? msg_getFloat(m, 1) : k;
chris@160 89 HvMessage *n = HV_MESSAGE_ON_STACK(1);
chris@160 90 f = cBinop_perform_op(op, msg_getFloat(m, 0), f);
chris@160 91 msg_initWithFloat(n, msg_getTimestamp(m), f);
chris@160 92 sendMessage(_c, 0, n);
chris@160 93 }
chris@160 94 }