fun experiment ig
This commit is contained in:
parent
1e61d057bb
commit
425391a3a8
|
@ -33,8 +33,8 @@ if(FAMILY STREQUAL "rp2040")
|
||||||
family_initialize_project(PROJECT ${CMAKE_CURRENT_LIST_DIR}) # calls pico_sdk_init()
|
family_initialize_project(PROJECT ${CMAKE_CURRENT_LIST_DIR}) # calls pico_sdk_init()
|
||||||
#pico_sdk_init()
|
#pico_sdk_init()
|
||||||
|
|
||||||
#pico_set_program_name(${PROJECT} "${PROJECT}")
|
#pico_set_program_name(PROJECT "${PROJECT}")
|
||||||
#pico_set_program_version(${PROJECT} "0.1")
|
#pico_set_program_version(PROJECT "0.1")
|
||||||
|
|
||||||
add_executable(${PROJECT})
|
add_executable(${PROJECT})
|
||||||
|
|
||||||
|
@ -64,6 +64,8 @@ target_sources(${PROJECT} PUBLIC
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/bsp/${FAMILY}/tempsensor.c
|
${CMAKE_CURRENT_SOURCE_DIR}/bsp/${FAMILY}/tempsensor.c
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/bsp/${FAMILY}/unique.c
|
${CMAKE_CURRENT_SOURCE_DIR}/bsp/${FAMILY}/unique.c
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/src/cdc_serprog.c
|
${CMAKE_CURRENT_SOURCE_DIR}/src/cdc_serprog.c
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/src/cdc_dynexec.c
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/src/dynexec.S
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/src/vnd_i2ctinyusb.c
|
${CMAKE_CURRENT_SOURCE_DIR}/src/vnd_i2ctinyusb.c
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/src/main.c
|
${CMAKE_CURRENT_SOURCE_DIR}/src/main.c
|
||||||
${CMAKE_CURRENT_SOURCE_DIR}/src/rtconf.c
|
${CMAKE_CURRENT_SOURCE_DIR}/src/rtconf.c
|
||||||
|
|
|
@ -19,6 +19,7 @@ enum {
|
||||||
#ifdef USE_USBCDC_FOR_STDIO
|
#ifdef USE_USBCDC_FOR_STDIO
|
||||||
CDC_N_STDIO,
|
CDC_N_STDIO,
|
||||||
#endif
|
#endif
|
||||||
|
CDC_N_DYNEXEC
|
||||||
|
|
||||||
CDC_N__NITF
|
CDC_N__NITF
|
||||||
};
|
};
|
||||||
|
@ -28,9 +29,9 @@ enum {
|
||||||
|
|
||||||
#define CFG_TUD_HID 1
|
#define CFG_TUD_HID 1
|
||||||
#ifdef USE_USBCDC_FOR_STDIO
|
#ifdef USE_USBCDC_FOR_STDIO
|
||||||
#define CFG_TUD_CDC 3
|
#define CFG_TUD_CDC 4
|
||||||
#else
|
#else
|
||||||
#define CFG_TUD_CDC 2
|
#define CFG_TUD_CDC 3
|
||||||
#endif
|
#endif
|
||||||
#define CFG_TUD_VENDOR 0
|
#define CFG_TUD_VENDOR 0
|
||||||
|
|
||||||
|
|
12
dmctl.py
12
dmctl.py
|
@ -143,6 +143,14 @@ def main():
|
||||||
#do_xfer(1, 1, "/dev/ttyACM1")
|
#do_xfer(1, 1, "/dev/ttyACM1")
|
||||||
#do_xfer(1, 0, "/dev/ttyACM1")
|
#do_xfer(1, 0, "/dev/ttyACM1")
|
||||||
|
|
||||||
if __name__ == '__main__':
|
#if __name__ == '__main__':
|
||||||
main()
|
# main()
|
||||||
|
|
||||||
|
with serial.Serial('/dev/ttyACM2', 115200, timeout=1) as ser:
|
||||||
|
regs = [x+0x18 for x in range(8)]
|
||||||
|
ops = [0x1889] # adds r1, r2
|
||||||
|
olen = len(ops)
|
||||||
|
|
||||||
|
ser.write(struct.pack('<IIIIIIII', *regs))
|
||||||
|
ser.write(struct.pack('<I', olen))
|
||||||
|
ser.write(struct.pack('<'+('H'*olen), *ops))
|
||||||
|
|
|
@ -0,0 +1,113 @@
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "tusb.h"
|
||||||
|
|
||||||
|
#include "protocfg.h"
|
||||||
|
|
||||||
|
#include "protos.h"
|
||||||
|
#include "util.h"
|
||||||
|
|
||||||
|
extern uint32_t dynexec_regs[8];
|
||||||
|
uint32_t dynexec_regs[8];
|
||||||
|
extern uint32_t dynexec_codesize;
|
||||||
|
uint32_t dynexec_codesize;
|
||||||
|
extern void* dynexec_code;
|
||||||
|
void* dynexec_code;
|
||||||
|
|
||||||
|
static uint32_t codecap;
|
||||||
|
static uint16_t codestatic[32];
|
||||||
|
|
||||||
|
static uint8_t rx_buf[CFG_TUD_CDC_RX_BUFSIZE];
|
||||||
|
static uint32_t rxavail, rxpos;
|
||||||
|
|
||||||
|
extern void dynexec_exec(void);
|
||||||
|
|
||||||
|
void cdc_dynexec_init(void) {
|
||||||
|
dynexec_codesize = 0;
|
||||||
|
memset(dynexec_regs, 0, sizeof dynexec_regs);
|
||||||
|
|
||||||
|
codecap = 32;
|
||||||
|
dynexec_code = codestatic;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint8_t read_byte(void) {
|
||||||
|
while (rxavail <= 0) {
|
||||||
|
if (!tud_cdc_n_connected(CDC_N_DYNEXEC) || !tud_cdc_n_available(CDC_N_DYNEXEC)) {
|
||||||
|
thread_yield();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
rxpos = 0;
|
||||||
|
rxavail = tud_cdc_n_read(CDC_N_DYNEXEC, rx_buf, sizeof rx_buf);
|
||||||
|
|
||||||
|
if (rxavail == 0) thread_yield();
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t rv = rx_buf[rxpos];
|
||||||
|
++rxpos;
|
||||||
|
--rxavail;
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t read_hword(void) {
|
||||||
|
uint16_t r;
|
||||||
|
r = (uint16_t)read_byte();
|
||||||
|
r |= (uint16_t)read_byte() << 8;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
static uint32_t read_word(void) {
|
||||||
|
uint32_t r;
|
||||||
|
r = (uint32_t)read_byte();
|
||||||
|
r |= (uint32_t)read_byte() << 8;
|
||||||
|
r |= (uint32_t)read_byte() << 16;
|
||||||
|
r |= (uint32_t)read_byte() << 24;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct dynexec_hdr {
|
||||||
|
uint32_t regs[8];
|
||||||
|
uint32_t codelen;
|
||||||
|
};
|
||||||
|
|
||||||
|
void cdc_dynexec_task(void) {
|
||||||
|
for (int i = 0; i < 8; ++i)
|
||||||
|
dynexec_regs[i] = read_word();
|
||||||
|
dynexec_codesize = read_word() /*+ 3*/;
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; ++i)
|
||||||
|
printf("r%d = 0x%08x\n", i, dynexec_regs[i]);
|
||||||
|
printf("codesize = 0x%04x\n", dynexec_codesize);
|
||||||
|
|
||||||
|
while (dynexec_codesize > codecap) {
|
||||||
|
codecap <<= 1;
|
||||||
|
|
||||||
|
if (codecap == 64) {
|
||||||
|
dynexec_code = malloc(codecap*sizeof(uint16_t));
|
||||||
|
} else {
|
||||||
|
dynexec_code = realloc(dynexec_code, codecap*sizeof(uint16_t));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (uint32_t i = 0; i < dynexec_codesize; ++i) {
|
||||||
|
((uint16_t*)dynexec_code)[2+i] = read_hword();
|
||||||
|
}
|
||||||
|
|
||||||
|
// all little-endian...
|
||||||
|
((uint16_t*)dynexec_code)[0] = 0xbc80; // pop {r7}
|
||||||
|
((uint16_t*)dynexec_code)[1] = 0xb500; // push {lr}
|
||||||
|
((uint16_t*)dynexec_code)[2+dynexec_codesize] = 0xbd00; // pop {pc}
|
||||||
|
|
||||||
|
for (uint32_t i = 0; i < dynexec_codesize+3; ++i) {
|
||||||
|
printf("code[%u] = %04x\n", i, ((uint16_t*)dynexec_code)[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("exec!\n");
|
||||||
|
dynexec_exec();
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; ++i)
|
||||||
|
printf("r%d = 0x%08x\n", i, dynexec_regs[i]);
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,43 @@
|
||||||
|
@ vim: set ft=armv5:
|
||||||
|
|
||||||
|
.cpu cortex-m0
|
||||||
|
.thumb
|
||||||
|
|
||||||
|
.extern dynexec_regs
|
||||||
|
.extern dynexec_code
|
||||||
|
|
||||||
|
.section .text.dynexec_exec, "ax", %progbits
|
||||||
|
.type dynexec_exec, %function
|
||||||
|
.thumb_func
|
||||||
|
.global dynexec_exec
|
||||||
|
dynexec_exec:
|
||||||
|
push {r0-r7,lr}
|
||||||
|
|
||||||
|
@ load regs from saved ones
|
||||||
|
ldr r7, =dynexec_regs
|
||||||
|
ldmia r7!, {r0-r6}
|
||||||
|
ldr r7, [r7]
|
||||||
|
push {r7}
|
||||||
|
|
||||||
|
ldr r7, =dynexec_code
|
||||||
|
ldr r7, [r7]
|
||||||
|
add r7, #1 @ bluh
|
||||||
|
blx r7
|
||||||
|
|
||||||
|
@ save regs in saved ones
|
||||||
|
|
||||||
|
push {r7}
|
||||||
|
ldr r7, =dynexec_regs
|
||||||
|
stmia r7!, {r0-r6}
|
||||||
|
pop {r0}
|
||||||
|
str r0, [r7]
|
||||||
|
|
||||||
|
pop {r0-r7,pc}
|
||||||
|
|
||||||
|
.pool
|
||||||
|
|
||||||
|
/*to_exec_code:
|
||||||
|
pop {r7}
|
||||||
|
push {lr}
|
||||||
|
@ TODO
|
||||||
|
pop {pc}*/
|
17
src/main.c
17
src/main.c
|
@ -82,6 +82,18 @@ static void serprog_thread_fn(void) {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static cothread_t dynexecthread;
|
||||||
|
static uint8_t dynexecstack[DEFAULT_STACK_SIZE];
|
||||||
|
|
||||||
|
static void dynexec_thread_fn(void) {
|
||||||
|
cdc_dynexec_init();
|
||||||
|
thread_yield();
|
||||||
|
while (1) {
|
||||||
|
cdc_dynexec_task();
|
||||||
|
thread_yield();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// FIXME
|
// FIXME
|
||||||
extern uint32_t co_active_buffer[64];
|
extern uint32_t co_active_buffer[64];
|
||||||
uint32_t co_active_buffer[64];
|
uint32_t co_active_buffer[64];
|
||||||
|
@ -114,6 +126,8 @@ int main(void) {
|
||||||
serprogthread = co_derive(serprogstack, sizeof serprogstack, serprog_thread_fn);
|
serprogthread = co_derive(serprogstack, sizeof serprogstack, serprog_thread_fn);
|
||||||
co_switch(serprogthread); // will call cdc_serprog_init() on correct thread
|
co_switch(serprogthread); // will call cdc_serprog_init() on correct thread
|
||||||
#endif
|
#endif
|
||||||
|
dynexecthread = co_derive(dynexecstack, sizeof dynexecstack, dynexec_thread_fn);
|
||||||
|
co_switch(dynexecthread);
|
||||||
#ifdef DBOARD_HAS_CMSISDAP
|
#ifdef DBOARD_HAS_CMSISDAP
|
||||||
DAP_Setup();
|
DAP_Setup();
|
||||||
#endif
|
#endif
|
||||||
|
@ -139,6 +153,9 @@ int main(void) {
|
||||||
#ifdef DBOARD_HAS_SERPROG
|
#ifdef DBOARD_HAS_SERPROG
|
||||||
co_switch(serprogthread);
|
co_switch(serprogthread);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
tud_task();
|
||||||
|
co_switch(dynexecthread);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -33,5 +33,8 @@ void itu_init(void);
|
||||||
void itu_task(void);
|
void itu_task(void);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
void dynexec_init(void);
|
||||||
|
void dynexec_task(void);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -100,6 +100,7 @@ extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
//------------- CLASS -------------//
|
//------------- CLASS -------------//
|
||||||
|
|
||||||
#define CFG_TUD_MSC 0
|
#define CFG_TUD_MSC 0
|
||||||
#define CFG_TUD_MIDI 0
|
#define CFG_TUD_MIDI 0
|
||||||
#define CFG_TUD_NET 0
|
#define CFG_TUD_NET 0
|
||||||
|
|
|
@ -57,6 +57,7 @@ enum {
|
||||||
STRID_IF_CDC_UART,
|
STRID_IF_CDC_UART,
|
||||||
STRID_IF_CDC_SERPROG,
|
STRID_IF_CDC_SERPROG,
|
||||||
STRID_IF_CDC_STDIO,
|
STRID_IF_CDC_STDIO,
|
||||||
|
STRID_IF_CDC_DYNEXEC,
|
||||||
};
|
};
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
|
@ -124,6 +125,8 @@ enum {
|
||||||
ITF_NUM_CDC_SERPROG_COM,
|
ITF_NUM_CDC_SERPROG_COM,
|
||||||
ITF_NUM_CDC_SERPROG_DATA,
|
ITF_NUM_CDC_SERPROG_DATA,
|
||||||
#endif
|
#endif
|
||||||
|
ITF_NUM_CDC_DYNEXEC_COM,
|
||||||
|
ITF_NUM_CDC_DYNEXEC_DATA,
|
||||||
#ifdef USE_USBCDC_FOR_STDIO
|
#ifdef USE_USBCDC_FOR_STDIO
|
||||||
ITF_NUM_CDC_STDIO_COM,
|
ITF_NUM_CDC_STDIO_COM,
|
||||||
ITF_NUM_CDC_STDIO_DATA,
|
ITF_NUM_CDC_STDIO_DATA,
|
||||||
|
@ -154,6 +157,7 @@ enum {
|
||||||
#ifdef USE_USBCDC_FOR_STDIO
|
#ifdef USE_USBCDC_FOR_STDIO
|
||||||
+ TUD_CDC_DESC_LEN
|
+ TUD_CDC_DESC_LEN
|
||||||
#endif
|
#endif
|
||||||
|
+ TUD_CDC_DESC_LEN
|
||||||
};
|
};
|
||||||
|
|
||||||
#define EPNUM_CDC_UART_OUT 0x02
|
#define EPNUM_CDC_UART_OUT 0x02
|
||||||
|
@ -163,9 +167,12 @@ enum {
|
||||||
#define EPNUM_CDC_SERPROG_OUT 0x05
|
#define EPNUM_CDC_SERPROG_OUT 0x05
|
||||||
#define EPNUM_CDC_SERPROG_IN 0x85
|
#define EPNUM_CDC_SERPROG_IN 0x85
|
||||||
#define EPNUM_CDC_SERPROG_NOTIF 0x86
|
#define EPNUM_CDC_SERPROG_NOTIF 0x86
|
||||||
#define EPNUM_CDC_STDIO_OUT 0x07
|
#define EPNUM_CDC_DYNEXEC_OUT 0x07
|
||||||
#define EPNUM_CDC_STDIO_IN 0x87
|
#define EPNUM_CDC_DYNEXEC_IN 0x87
|
||||||
#define EPNUM_CDC_STDIO_NOTIF 0x88
|
#define EPNUM_CDC_DYNEXEC_NOTIF 0x88
|
||||||
|
#define EPNUM_CDC_STDIO_OUT 0x09
|
||||||
|
#define EPNUM_CDC_STDIO_IN 0x89
|
||||||
|
#define EPNUM_CDC_STDIO_NOTIF 0x8a
|
||||||
|
|
||||||
// NOTE: if you modify this table, don't forget to keep tusb_config.h up to date as well!
|
// NOTE: if you modify this table, don't forget to keep tusb_config.h up to date as well!
|
||||||
// TODO: maybe add some strings to all these interfaces
|
// TODO: maybe add some strings to all these interfaces
|
||||||
|
@ -188,6 +195,8 @@ uint8_t const desc_configuration[] = {
|
||||||
TUD_CDC_DESCRIPTOR(ITF_NUM_CDC_SERPROG_COM, STRID_IF_CDC_SERPROG, EPNUM_CDC_SERPROG_NOTIF, 64, EPNUM_CDC_SERPROG_OUT, EPNUM_CDC_SERPROG_IN, 64),
|
TUD_CDC_DESCRIPTOR(ITF_NUM_CDC_SERPROG_COM, STRID_IF_CDC_SERPROG, EPNUM_CDC_SERPROG_NOTIF, 64, EPNUM_CDC_SERPROG_OUT, EPNUM_CDC_SERPROG_IN, 64),
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
TUD_CDC_DESCRIPTOR(ITF_NUM_CDC_DYNEXEC_COM, STRID_IF_CDC_DYNEXEC, EPNUM_CDC_DYNEXEC_NOTIF, 64, EPNUM_CDC_DYNEXEC_OUT, EPNUM_CDC_DYNEXEC_IN, 64),
|
||||||
|
|
||||||
#ifdef USE_USBCDC_FOR_STDIO
|
#ifdef USE_USBCDC_FOR_STDIO
|
||||||
TUD_CDC_DESCRIPTOR(ITF_NUM_CDC_STDIO_COM, STRID_IF_CDC_STDIO, EPNUM_CDC_STDIO_NOTIF, 64, EPNUM_CDC_STDIO_OUT, EPNUM_CDC_STDIO_IN, 64),
|
TUD_CDC_DESCRIPTOR(ITF_NUM_CDC_STDIO_COM, STRID_IF_CDC_STDIO, EPNUM_CDC_STDIO_NOTIF, 64, EPNUM_CDC_STDIO_OUT, EPNUM_CDC_STDIO_IN, 64),
|
||||||
#endif
|
#endif
|
||||||
|
@ -218,6 +227,7 @@ char const* string_desc_arr [] = {
|
||||||
[STRID_IF_CDC_UART] = "UART CDC interface",
|
[STRID_IF_CDC_UART] = "UART CDC interface",
|
||||||
[STRID_IF_CDC_SERPROG] = "Serprog CDC interface",
|
[STRID_IF_CDC_SERPROG] = "Serprog CDC interface",
|
||||||
[STRID_IF_CDC_STDIO] = "stdio CDC interface (debug)",
|
[STRID_IF_CDC_STDIO] = "stdio CDC interface (debug)",
|
||||||
|
[STRID_IF_CDC_DYNEXEC] = "Dynamic execution CDC itf",
|
||||||
};
|
};
|
||||||
|
|
||||||
// Invoked when received GET STRING DESCRIPTOR request
|
// Invoked when received GET STRING DESCRIPTOR request
|
||||||
|
|
Loading…
Reference in New Issue