Berry support for AES GCM and raw MQTT send/receive

This commit is contained in:
Stephan Hadinger 2021-07-29 19:58:23 +02:00
parent 3b5180d1e8
commit bbe3babe8b
11 changed files with 1946 additions and 1644 deletions

View File

@ -0,0 +1,44 @@
/********************************************************************
* Berry module `webserver`
*
* To use: `import webserver`
*
* Allows to respond to HTTP request
*******************************************************************/
#include "be_constobj.h"
#ifdef USE_ALEXA_AVS
extern int m_aes_gcm_init(bvm *vm);
extern int m_aes_gcm_encryt(bvm *vm);
extern int m_aes_gcm_decryt(bvm *vm);
extern int m_aes_gcm_tag(bvm *vm);
#if BE_USE_PRECOMPILED_OBJECT
#include "../generate/be_fixed_be_class_aes_gcm.h"
#endif
void be_load_aes_gcm_lib(bvm *vm) {
// insert the class GCM in module AES
be_newmodule(vm);
be_setname(vm, -1, "AES");
be_setglobal(vm, "AES");
be_pushntvclass(vm, &be_class_aes_gcm);
be_setmember(vm, -2, "GCM");
be_pop(vm, 2);
}
/* @const_object_info_begin
class be_class_aes_gcm (scope: global, name: GCM) {
.p1, var
.p2, var
init, func(m_aes_gcm_init)
encrypt, func(m_aes_gcm_encryt)
decrypt, func(m_aes_gcm_decryt)
tag, func(m_aes_gcm_tag)
}
@const_object_info_end */
#endif // USE_ALEXA_AVS

View File

@ -103,6 +103,7 @@ extern void be_load_wirelib(bvm *vm);
extern void be_load_Driver_class(bvm *vm);
extern void be_load_driver_i2c_lib(bvm *vm);
extern void be_load_md5_lib(bvm *vm);
extern void be_load_aes_gcm_lib(bvm *vm);
#ifdef USE_I2S_AUDIO_BERRY
extern void be_load_driver_audio_lib(bvm *vm);
@ -133,6 +134,9 @@ BERRY_API void be_load_custom_libs(bvm *vm)
be_load_tasmota_ntvlib(vm);
be_load_Driver_class(vm);
be_load_md5_lib(vm);
#ifdef USE_ALEXA_AVS
be_load_aes_gcm_lib(vm);
#endif
#ifdef USE_I2C
be_load_wirelib(vm);
be_load_driver_i2c_lib(vm);

View File

@ -1364,8 +1364,8 @@ be_local_closure(gc, /* name */
********************************************************************/
be_local_closure(event, /* name */
be_nested_proto(
18, /* nstack */
5, /* argc */
19, /* nstack */
6, /* argc */
0, /* has upvals */
NULL, /* no upvals */
0, /* has sup protos */
@ -1390,85 +1390,86 @@ be_local_closure(event, /* name */
}),
(be_nested_const_str("event", -30355297, 5)),
(be_nested_const_str("input", -103256197, 5)),
( &(const binstruction[78]) { /* code */
0xA4160000, // 0000 IMPORT R5 R256
0x1C180301, // 0001 EQ R6 R1 R257
0x781A0001, // 0002 JMPF R6 #0005
0x8C180102, // 0003 GETMET R6 R0 R258
0x7C180200, // 0004 CALL R6 1
0x1C180303, // 0005 EQ R6 R1 R259
0x781A0006, // 0006 JMPF R6 #000E
0x8C180104, // 0007 GETMET R6 R0 R260
0x5C200400, // 0008 MOVE R8 R2
0x5C240600, // 0009 MOVE R9 R3
0x5C280800, // 000A MOVE R10 R4
0x7C180800, // 000B CALL R6 4
0x80040C00, // 000C RET 1 R6
0x7002003E, // 000D JMP #004D
0x1C180305, // 000E EQ R6 R1 R261
0x781A0004, // 000F JMPF R6 #0015
0x8C180106, // 0010 GETMET R6 R0 R262
0x5C200800, // 0011 MOVE R8 R4
0x7C180400, // 0012 CALL R6 2
0x80040C00, // 0013 RET 1 R6
0x70020037, // 0014 JMP #004D
0x1C180307, // 0015 EQ R6 R1 R263
0x781A0003, // 0016 JMPF R6 #001B
0x8C180107, // 0017 GETMET R6 R0 R263
0x7C180200, // 0018 CALL R6 1
0x80040C00, // 0019 RET 1 R6
0x70020031, // 001A JMP #004D
0x88180108, // 001B GETMBR R6 R0 R264
0x781A002F, // 001C JMPF R6 #004D
0x60180000, // 001D GETGBL R6 G0
0x881C0108, // 001E GETMBR R7 R0 R264
0x7C180200, // 001F CALL R6 1
0xA8020026, // 0020 EXBLK 0 #0048
0x5C1C0C00, // 0021 MOVE R7 R6
0x7C1C0000, // 0022 CALL R7 0
0x8C200B09, // 0023 GETMET R8 R5 R265
0x5C280E00, // 0024 MOVE R10 R7
0x5C2C0200, // 0025 MOVE R11 R1
0x7C200600, // 0026 CALL R8 3
0x60240015, // 0027 GETGBL R9 G21
0x5C281000, // 0028 MOVE R10 R8
0x7C240200, // 0029 CALL R9 1
0x1C24130A, // 002A EQ R9 R9 R266
0x7826001A, // 002B JMPF R9 #0047
0xA802000D, // 002C EXBLK 0 #003B
0x5C241000, // 002D MOVE R9 R8
0x5C280E00, // 002E MOVE R10 R7
0x5C2C0400, // 002F MOVE R11 R2
0x5C300600, // 0030 MOVE R12 R3
0x5C340800, // 0031 MOVE R13 R4
0x7C240800, // 0032 CALL R9 4
0x50280200, // 0033 LDBOOL R10 1 0
0x1C28120A, // 0034 EQ R10 R9 R10
0x782A0002, // 0035 JMPF R10 #0039
0x50280200, // 0036 LDBOOL R10 1 0
0xA8040002, // 0037 EXBLK 1 2
0x80041400, // 0038 RET 1 R10
0xA8040001, // 0039 EXBLK 1 1
0x7002000B, // 003A JMP #0047
0xAC240002, // 003B CATCH R9 0 2
0x70020008, // 003C JMP #0046
0xA42E1600, // 003D IMPORT R11 R267
0x6030000F, // 003E GETGBL R12 G15
0x8C34170C, // 003F GETMET R13 R11 R268
0x583C000D, // 0040 LDCONST R15 K13
0x5C401200, // 0041 MOVE R16 R9
( &(const binstruction[79]) { /* code */
0xA41A0000, // 0000 IMPORT R6 R256
0x1C1C0301, // 0001 EQ R7 R1 R257
0x781E0001, // 0002 JMPF R7 #0005
0x8C1C0102, // 0003 GETMET R7 R0 R258
0x7C1C0200, // 0004 CALL R7 1
0x1C1C0303, // 0005 EQ R7 R1 R259
0x781E0006, // 0006 JMPF R7 #000E
0x8C1C0104, // 0007 GETMET R7 R0 R260
0x5C240400, // 0008 MOVE R9 R2
0x5C280600, // 0009 MOVE R10 R3
0x5C2C0800, // 000A MOVE R11 R4
0x7C1C0800, // 000B CALL R7 4
0x80040E00, // 000C RET 1 R7
0x7002003F, // 000D JMP #004E
0x1C1C0305, // 000E EQ R7 R1 R261
0x781E0004, // 000F JMPF R7 #0015
0x8C1C0106, // 0010 GETMET R7 R0 R262
0x5C240800, // 0011 MOVE R9 R4
0x7C1C0400, // 0012 CALL R7 2
0x80040E00, // 0013 RET 1 R7
0x70020038, // 0014 JMP #004E
0x1C1C0307, // 0015 EQ R7 R1 R263
0x781E0003, // 0016 JMPF R7 #001B
0x8C1C0107, // 0017 GETMET R7 R0 R263
0x7C1C0200, // 0018 CALL R7 1
0x80040E00, // 0019 RET 1 R7
0x70020032, // 001A JMP #004E
0x881C0108, // 001B GETMBR R7 R0 R264
0x781E0030, // 001C JMPF R7 #004E
0x601C0000, // 001D GETGBL R7 G0
0x88200108, // 001E GETMBR R8 R0 R264
0x7C1C0200, // 001F CALL R7 1
0xA8020027, // 0020 EXBLK 0 #0049
0x5C200E00, // 0021 MOVE R8 R7
0x7C200000, // 0022 CALL R8 0
0x8C240D09, // 0023 GETMET R9 R6 R265
0x5C2C1000, // 0024 MOVE R11 R8
0x5C300200, // 0025 MOVE R12 R1
0x7C240600, // 0026 CALL R9 3
0x60280015, // 0027 GETGBL R10 G21
0x5C2C1200, // 0028 MOVE R11 R9
0x7C280200, // 0029 CALL R10 1
0x1C28150A, // 002A EQ R10 R10 R266
0x782A001B, // 002B JMPF R10 #0048
0xA802000E, // 002C EXBLK 0 #003C
0x5C281200, // 002D MOVE R10 R9
0x5C2C1000, // 002E MOVE R11 R8
0x5C300400, // 002F MOVE R12 R2
0x5C340600, // 0030 MOVE R13 R3
0x5C380800, // 0031 MOVE R14 R4
0x5C3C0A00, // 0032 MOVE R15 R5
0x7C280A00, // 0033 CALL R10 5
0x502C0200, // 0034 LDBOOL R11 1 0
0x1C2C140B, // 0035 EQ R11 R10 R11
0x782E0002, // 0036 JMPF R11 #003A
0x502C0200, // 0037 LDBOOL R11 1 0
0xA8040002, // 0038 EXBLK 1 2
0x80041600, // 0039 RET 1 R11
0xA8040001, // 003A EXBLK 1 1
0x7002000B, // 003B JMP #0048
0xAC280002, // 003C CATCH R10 0 2
0x70020008, // 003D JMP #0047
0xA4321600, // 003E IMPORT R12 R267
0x6034000F, // 003F GETGBL R13 G15
0x8C38190C, // 0040 GETMET R14 R12 R268
0x5840000D, // 0041 LDCONST R16 K13
0x5C441400, // 0042 MOVE R17 R10
0x7C340800, // 0043 CALL R13 4
0x7C300200, // 0044 CALL R12 1
0x70020000, // 0045 JMP #0047
0xB0080000, // 0046 RAISE 2 R0 R0
0x7001FFD8, // 0047 JMP #0021
0x5818000E, // 0048 LDCONST R6 K14
0xAC180200, // 0049 CATCH R6 1 0
0xB0080000, // 004A RAISE 2 R0 R0
0x50180000, // 004B LDBOOL R6 0 0
0x80040C00, // 004C RET 1 R6
0x80000000, // 004D RET 0 R0
0x5C481600, // 0043 MOVE R18 R11
0x7C380800, // 0044 CALL R14 4
0x7C340200, // 0045 CALL R13 1
0x70020000, // 0046 JMP #0048
0xB0080000, // 0047 RAISE 2 R0 R0
0x7001FFD7, // 0048 JMP #0021
0x581C000E, // 0049 LDCONST R7 K14
0xAC1C0200, // 004A CATCH R7 1 0
0xB0080000, // 004B RAISE 2 R0 R0
0x501C0000, // 004C LDBOOL R7 0 0
0x80040E00, // 004D RET 1 R7
0x80000000, // 004E RET 0 R0
})
)
);

View File

@ -241,7 +241,7 @@ class Tasmota
end
def event(event_type, cmd, idx, payload)
def event(event_type, cmd, idx, payload, raw)
import introspect
if event_type=='every_50ms' self.run_deferred() end #- first run deferred events -#
@ -253,7 +253,7 @@ class Tasmota
var f = introspect.get(d, event_type) # try to match a function or method with the same name
if type(f) == 'function'
try
var done = f(d, cmd, idx, payload)
var done = f(d, cmd, idx, payload, raw)
if done == true return true end
except .. as e,m
import string

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,22 @@
#include "be_constobj.h"
static be_define_const_map_slots(be_class_aes_gcm_map) {
{ be_const_key(encrypt, 4), be_const_func(m_aes_gcm_encryt) },
{ be_const_key(dot_p2, -1), be_const_index(0) },
{ be_const_key(decrypt, -1), be_const_func(m_aes_gcm_decryt) },
{ be_const_key(init, -1), be_const_func(m_aes_gcm_init) },
{ be_const_key(dot_p1, -1), be_const_index(1) },
{ be_const_key(tag, -1), be_const_func(m_aes_gcm_tag) },
};
static be_define_const_map(
be_class_aes_gcm_map,
6
);
BE_EXPORT_VARIABLE be_define_const_class(
be_class_aes_gcm,
2,
NULL,
GCM
);

View File

@ -0,0 +1,196 @@
/*
xdrv_52_3_berry_md5.ino - Berry scripting language, Md5 class
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 <http://www.gnu.org/licenses/>.
*/
#ifdef USE_BERRY
#ifdef USE_ALEXA_AVS
#include <berry.h>
#include "be_mem.h"
/*********************************************************************************************\
* AES class
*
\*********************************************************************************************/
extern "C" {
int free_br_obj(bvm* vm) {
int argc = be_top(vm);
if (argc > 0) {
void * obj = be_tocomptr(vm, 1);
if (obj != NULL) { be_os_free(obj); }
}
be_return_nil(vm);
}
// `AES_GCM.init(secret_key:bytes(32), iv:bytes(12)) -> instance`
int32_t m_aes_gcm_init(struct bvm *vm);
int32_t m_aes_gcm_init(struct bvm *vm) {
int32_t argc = be_top(vm); // Get the number of arguments
if (argc >= 3 && be_isinstance(vm, 2) && be_isinstance(vm, 3)) {
do {
be_getglobal(vm, "bytes"); /* get the bytes class */ /* TODO eventually replace with be_getbuiltin */
if (!be_isderived(vm, 2)) break;
size_t length = 0;
const void * bytes = be_tobytes(vm, 2, &length);
if (!bytes) break;
if (length != 32) {
be_raise(vm, "value_error", "Key size must be 32 bytes");
}
be_getglobal(vm, "bytes"); /* get the bytes class */ /* TODO eventually replace with be_getbuiltin */
if (!be_isderived(vm, 3)) break;
size_t iv_length = 0;
const void * iv_bytes = be_tobytes(vm, 3, &iv_length);
if (!iv_bytes) break;
// Initialize an AES CTR structure with the secret key
br_aes_small_ctr_keys * ctr_ctx = (br_aes_small_ctr_keys *) be_os_malloc(sizeof(br_aes_small_ctr_keys));
if (!ctr_ctx) { be_throw(vm, BE_MALLOC_FAIL); }
br_aes_small_ctr_init(ctr_ctx, bytes, length);
be_newcomobj(vm, ctr_ctx, &free_br_obj);
be_setmember(vm, 1, ".p1");
// Initialize an AES GCM structure based on this CTR engine
br_gcm_context * gcm_ctx = (br_gcm_context *) be_os_malloc(sizeof(br_gcm_context));
if (!gcm_ctx) { be_throw(vm, BE_MALLOC_FAIL); }
br_gcm_init(gcm_ctx, &ctr_ctx->vtable, &br_ghash_ctmul32);
be_newcomobj(vm, gcm_ctx, &free_br_obj);
be_setmember(vm, 1, ".p2");
// Reset GCM context with provided IV
br_gcm_reset(gcm_ctx, iv_bytes, iv_length);
// We don't have any additional authenticated data so we flip instantly
br_gcm_flip(gcm_ctx);
be_return_nil(vm);
// success
} while (0);
}
be_raise(vm, kTypeError, nullptr);
}
int32_t m_aes_gcm_encryt(bvm *vm);
int32_t m_aes_gcm_decryt(bvm *vm);
int32_t m_aes_gcm_encrypt_or_decryt(bvm *vm, int encrypt) {
int32_t argc = be_top(vm); // Get the number of arguments
if (argc >= 2 && be_isinstance(vm, 2)) {
do {
be_getglobal(vm, "bytes"); /* get the bytes class */ /* TODO eventually replace with be_getbuiltin */
if (!be_isderived(vm, 2)) break;
// get GCM context
be_getmember(vm, 1, ".p2");
br_gcm_context * gcm_ctx = (br_gcm_context *) be_tocomptr(vm, -1);
be_pop(vm, 1);
// copy the input buffer
be_getmember(vm, 2, "copy"); // stack: bytes.copy()
be_pushvalue(vm, 2); // stack: bytes.copy(), bytes instance
be_call(vm, 1); // call copy with self parameter
be_pop(vm, 1); // stack: clone of input bytes
size_t length = 0;
// we are changing bytes in place
void * bytes = (void*) be_tobytes(vm, -1, &length);
if (!bytes) break;
br_gcm_run(gcm_ctx, encrypt, bytes, length);
be_return(vm);
// success
} while (0);
}
be_raise(vm, kTypeError, nullptr);
}
int32_t m_aes_gcm_encryt(bvm *vm) {
return m_aes_gcm_encrypt_or_decryt(vm, 1);
}
int32_t m_aes_gcm_decryt(bvm *vm) {
return m_aes_gcm_encrypt_or_decryt(vm, 0);
}
int32_t m_aes_gcm_tag(bvm *vm) {
do {
be_getglobal(vm, "bytes"); /* get the bytes class */ /* TODO eventually replace with be_getbuiltin */
// get GCM context
be_getmember(vm, 1, ".p2");
br_gcm_context * gcm_ctx = (br_gcm_context *) be_tocomptr(vm, -1);
be_pop(vm, 1);
// create a bytes buffer of 16 bytes
uint8_t tag[16] = {0};
br_gcm_get_tag(gcm_ctx, tag);
be_pushbytes(vm, tag, sizeof(tag));
be_return(vm);
// success
} while (0);
be_raise(vm, kTypeError, nullptr);
}
// // `Md5.update(content:bytes()) -> nil`
// //
// // Add raw bytes to the MD5 calculation
// int32_t m_md5_update(struct bvm *vm);
// int32_t m_md5_update(struct bvm *vm) {
// int32_t argc = be_top(vm); // Get the number of arguments
// if (argc >= 2 && be_isinstance(vm, 2)) {
// do {
// be_getglobal(vm, "bytes"); /* get the bytes class */ /* TODO eventually replace with be_getbuiltin */
// if (!be_isderived(vm, 2)) break;
// size_t length = 0;
// const void * bytes = be_tobytes(vm, 2, &length);
// if (!bytes) break;
// be_getmember(vm, 1, ".p");
// struct MD5Context * ctx;
// ctx = (struct MD5Context *) be_tocomptr(vm, -1);
// if (!ctx) break;
// if (length > 0) {
// MD5Update(ctx, (const uint8_t*) bytes, length);
// }
// be_return_nil(vm);
// // success
// } while (0);
// }
// be_raise(vm, kTypeError, nullptr);
// }
// // `Md5.update(content:bytes()) -> nil`
// //
// // Add raw bytes to the MD5 calculation
// int32_t m_md5_finish(struct bvm *vm);
// int32_t m_md5_finish(struct bvm *vm) {
// be_getmember(vm, 1, ".p");
// struct MD5Context * ctx;
// ctx = (struct MD5Context *) be_tocomptr(vm, -1);
// uint8_t output[16];
// MD5Final(output, ctx);
// be_pushbytes(vm, output, sizeof(output));
// be_return(vm);
// }
}
#endif // USE_ALEXA_AVS
#endif // USE_BERRY

View File

@ -33,7 +33,8 @@ extern "C" {
int free_ctx(bvm* vm) {
int argc = be_top(vm);
if (argc > 0) {
struct MD5Context * ctx = (struct MD5Context *) be_tocomptr(vm, 1);
be_getmember(vm, 1, ".p");
struct MD5Context * ctx = (struct MD5Context *) be_tocomptr(vm, -1);
if (ctx != NULL) {
be_os_free(ctx);
}

View File

@ -72,17 +72,27 @@ extern "C" {
int32_t l_publish(struct bvm *vm);
int32_t l_publish(struct bvm *vm) {
int32_t top = be_top(vm); // Get the number of arguments
if (top >= 3 && be_isstring(vm, 2) && be_isstring(vm, 3)) { // 2 mandatory string arguments
if (top >= 3 && be_isstring(vm, 2) && (be_isstring(vm, 3) || be_isinstance(vm, 3))) { // 2 mandatory string arguments
if (top == 3 || (top == 4 && be_isbool(vm, 4))) { // 3rd optional argument must be bool
const char * topic = be_tostring(vm, 2);
const char * payload = be_tostring(vm, 3);
const char * payload = nullptr;
size_t payload_len = 0;
if (be_isstring(vm, 3)) {
payload = be_tostring(vm, 3);
payload_len = strlen(payload);
} else {
be_getglobal(vm, "bytes"); /* get the bytes class */ /* TODO eventually replace with be_getbuiltin */
if (be_isderived(vm, 3)) {
payload = (const char *) be_tobytes(vm, 3, &payload_len);
}
}
bool retain = false;
if (top == 4) {
retain = be_tobool(vm, 4);
}
Response_P(payload);
MqttPublish(topic, retain);
be_return(vm); // Return
if (!payload) { be_raise(vm, "value_error", "Empty payload"); }
MqttPublishPayload(topic, payload, payload_len, retain);
be_return_nil(vm); // Return
}
}
be_raise(vm, kTypeError, nullptr);

View File

@ -37,6 +37,8 @@ void (* const BerryCommand[])(void) PROGMEM = {
CmndBrRun,
};
int32_t callBerryEventDispatcher(const char *type, const char *cmd, int32_t idx, const char *payload, uint32_t data_len = 0);
//
// Sanity Check for be_top()
//
@ -191,7 +193,8 @@ bool callMethodObjectWithArgs(const char * objname, const char * method, size_t
// call the event dispatcher from Tasmota object
int32_t callBerryEventDispatcher(const char *type, const char *cmd, int32_t idx, const char *payload) {
// if data_len is non-zero, the event is also sent as raw `bytes()` object because the string may lose data
int32_t callBerryEventDispatcher(const char *type, const char *cmd, int32_t idx, const char *payload, uint32_t data_len) {
int32_t ret = 0;
bvm *vm = berry.vm;
@ -206,7 +209,13 @@ int32_t callBerryEventDispatcher(const char *type, const char *cmd, int32_t idx,
be_pushstring(vm, cmd != nullptr ? cmd : "");
be_pushint(vm, idx);
be_pushstring(vm, payload != nullptr ? payload : "{}"); // empty json
if (data_len > 0) {
be_pushbytes(vm, payload, data_len); // if data_len is set, we also push raw bytes
ret = be_pcall(vm, 6); // 6 arguments
be_pop(vm, 1);
} else {
ret = be_pcall(vm, 5); // 5 arguments
}
if (ret != 0) {
BerryDumpErrorAndClear(vm, false); // log in Tasmota console only
return ret;
@ -725,7 +734,7 @@ bool Xdrv52(uint8_t function)
result = callBerryRule();
break;
case FUNC_MQTT_DATA:
result = callBerryEventDispatcher(PSTR("mqtt_data"), XdrvMailbox.topic, 0, XdrvMailbox.data);
result = callBerryEventDispatcher(PSTR("mqtt_data"), XdrvMailbox.topic, 0, XdrvMailbox.data, XdrvMailbox.data_len);
break;
case FUNC_EVERY_50_MSECOND:
callBerryEventDispatcher(PSTR("every_50ms"), nullptr, 0, nullptr);