/* xdrv_52_3_berry_native.ino - Berry scripting language, native fucnctions Copyright (C) 2021 Stephan Hadinger, Berry language by Guan Wenliang https://github.com/Skiars/berry This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #ifdef USE_BERRY #if defined(USE_ONEWIRE) || defined(USE_DS18x20) #include #include // // read the `bus` attribute and return `Wire` or `Wire1` // TwoWire & getWire(bvm *vm); // TwoWire & getWire(bvm *vm) { // be_getmember(vm, 1, "bus"); // int32_t bus = be_toint(vm, -1); // bus is 1 or 2 // be_pop(vm, 1); // if (2 != bus) { // return Wire; // } else { // return Wire1; // } // } // int32_t getBus(bvm *vm); // 1 or 2 // int32_t getBus(bvm *vm) { // be_getmember(vm, 1, "bus"); // int32_t bus = be_toint(vm, -1); // be_pop(vm, 1); // return bus; // } /*********************************************************************************************\ * Native functions mapped to Berry functions * * import wire * * wire.get_free_heap() -> int * \*********************************************************************************************/ extern "C" { OneWire * b_onewire_get(struct bvm *vm) { be_getmember(vm, 1, ".p"); OneWire * ow = (OneWire *) be_tocomptr(vm, -1); be_pop(vm, 1); return ow; } // Berry: `init(gpio:int)` int32_t b_onewire_init(struct bvm *vm); int32_t b_onewire_init(struct bvm *vm) { int32_t argc = be_top(vm); // Get the number of arguments if (argc > 1 && be_isint(vm, 2)) { int gpio = be_toint(vm, 2); OneWire * ow = new OneWire(gpio); be_pushcomptr(vm, (void*) ow); be_setmember(vm, 1, ".p"); be_return_nil(vm); } be_raise(vm, kTypeError, "gpio number expected"); } // Berry: `deinit(void)` int32_t b_onewire_deinit(struct bvm *vm); int32_t b_onewire_deinit(struct bvm *vm) { be_getmember(vm, 1, ".p"); OneWire * ow = (OneWire *) be_tocomptr(vm, -1); if (ow) { delete ow; // clear be_pushcomptr(vm, (void*) NULL); be_setmember(vm, 1, ".p"); } be_return_nil(vm); } // Berry: `reset(void) -> bool` // returns true if a device is present int32_t b_onewire_reset(struct bvm *vm); int32_t b_onewire_reset(struct bvm *vm) { OneWire * ow = b_onewire_get(vm); if (ow) { uint8_t resp = ow->reset(); be_pushbool(vm, resp != 0); be_return(vm); } be_return_nil(vm); } // Berry: `skip(void) -> nil` int32_t b_onewire_skip(struct bvm *vm); int32_t b_onewire_skip(struct bvm *vm) { OneWire * ow = b_onewire_get(vm); if (ow) { ow->skip(); } be_return_nil(vm); } // Berry: `depower(void) -> nil` int32_t b_onewire_depower(struct bvm *vm); int32_t b_onewire_depower(struct bvm *vm) { OneWire * ow = b_onewire_get(vm); if (ow) { ow->depower(); } be_return_nil(vm); } // Berry: `select(bytes) -> nil` // returns true if a device is present int32_t b_onewire_select(struct bvm *vm); int32_t b_onewire_select(struct bvm *vm) { int32_t argc = be_top(vm); // Get the number of arguments if (argc > 1 && be_isbytes(vm, 2)) { OneWire * ow = b_onewire_get(vm); if (ow) { uint8_t rom[8] = {0}; size_t len; const void * buf = be_tobytes(vm, 2, &len); if (len > 8) { len = 8; } /* 8 bytes max */ memmove(rom, buf, len); ow->select(rom); } be_return_nil(vm); } be_raise(vm, kTypeError, NULL); } // Berry: `write(byte:int | bytes [, power:bool = false]) -> nil` int32_t b_onewire_write(struct bvm *vm); int32_t b_onewire_write(struct bvm *vm) { int32_t argc = be_top(vm); // Get the number of arguments bool power = false; if (argc > 1 && (be_isint(vm, 2) || be_isbytes(vm, 2))) { if (argc > 2 && (be_isbool(vm, 3) || be_isint(vm,3))) { power = be_tobool(vm, 3); } uint8_t val = 0; if (be_isint(vm, 2)) { val = be_toint(vm, 2); } else { /* bytes */ size_t len = 0; const uint8_t * buf = (const uint8_t *) be_tobytes(vm, 2, &len); if (len > 0) { val = buf[0]; } else { be_return_nil(vm); /* abort */ } } OneWire * ow = b_onewire_get(vm); if (ow) { ow->write(val, power); } be_return_nil(vm); } be_raise(vm, kTypeError, NULL); } // Berry: `read(void) -> bytes()` int32_t b_onewire_read(struct bvm *vm); int32_t b_onewire_read(struct bvm *vm) { OneWire * ow = b_onewire_get(vm); if (ow) { uint8_t val = ow->read(); be_pushbytes(vm, &val, 1); be_return(vm); } be_return_nil(vm); } // Berry: `reset_search(void) -> nil` int32_t b_onewire_reset_search(struct bvm *vm); int32_t b_onewire_reset_search(struct bvm *vm) { OneWire * ow = b_onewire_get(vm); if (ow) { ow->reset_search(); } be_return_nil(vm); } // Berry: `target_search(family_code:int) -> nil` int32_t b_onewire_target_search(struct bvm *vm); int32_t b_onewire_target_search(struct bvm *vm) { int32_t argc = be_top(vm); // Get the number of arguments if (argc > 1 && be_isint(vm, 2)) { OneWire * ow = b_onewire_get(vm); if (ow) { ow->target_search(be_toint(vm, 2)); } be_return_nil(vm); } be_raise(vm, kTypeError, NULL); } // Berry: `search(void) -> bytes() or nil` int32_t b_onewire_search(struct bvm *vm); int32_t b_onewire_search(struct bvm *vm) { uint8_t rom[8] = {0}; OneWire * ow = b_onewire_get(vm); if (ow) { uint8_t res = ow->search(rom); if (res) { be_pushbytes(vm, rom, sizeof(rom)); be_return(vm); } else { be_return_nil(vm); } } be_return_nil(vm); } } #endif // USE_I2C #endif // USE_BERRY