445 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			445 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "pxt.h"
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <sys/time.h>
 | |
| #include <time.h>
 | |
| #include <cstdarg>
 | |
| #include <pthread.h>
 | |
| #include <assert.h>
 | |
| #include <unistd.h>
 | |
| #include <dirent.h>
 | |
| #include <signal.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <fcntl.h>
 | |
| 
 | |
| void *operator new(size_t size) {
 | |
|     return malloc(size);
 | |
| }
 | |
| void *operator new[](size_t size) {
 | |
|     return malloc(size);
 | |
| }
 | |
| 
 | |
| void operator delete(void *p) {
 | |
|     free(p);
 | |
| }
 | |
| void operator delete[](void *p) {
 | |
|     free(p);
 | |
| }
 | |
| 
 | |
| namespace pxt {
 | |
| 
 | |
| static int startTime;
 | |
| static pthread_mutex_t execMutex;
 | |
| static pthread_mutex_t eventMutex;
 | |
| static pthread_cond_t newEventBroadcast;
 | |
| 
 | |
| struct Thread {
 | |
|     struct Thread *next;
 | |
|     Action act;
 | |
|     TValue arg0;
 | |
|     pthread_t pid;
 | |
|     pthread_cond_t waitCond;
 | |
|     int waitSource;
 | |
|     int waitValue;
 | |
|     TValue data0;
 | |
|     TValue data1;
 | |
| };
 | |
| 
 | |
| static struct Thread *allThreads;
 | |
| static struct Event *eventHead, *eventTail;
 | |
| static int usbFD;
 | |
| static int dmesgPtr;
 | |
| static char dmesgBuf[4096];
 | |
| 
 | |
| struct Event {
 | |
|     struct Event *next;
 | |
|     int source;
 | |
|     int value;
 | |
| };
 | |
| 
 | |
| Event lastEvent;
 | |
| 
 | |
| Event *mkEvent(int source, int value) {
 | |
|     auto res = new Event();
 | |
|     memset(res, 0, sizeof(Event));
 | |
|     res->source = source;
 | |
|     res->value = value;
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| #define USB_MAGIC 0x3d3f
 | |
| #define USB_SERIAL 1
 | |
| 
 | |
| struct UsbPacket {
 | |
|     uint16_t size;
 | |
|     uint16_t msgcount;
 | |
|     uint16_t magic;
 | |
|     uint16_t code;
 | |
|     char buf[1000];
 | |
| };
 | |
| 
 | |
| void sendUsb(uint16_t code, const char *data, int len) {
 | |
|     if (usbFD == 0)
 | |
|         usbFD = open("/dev/lms_usbdev", O_RDWR, 0666);
 | |
| 
 | |
|     while (len > 0) {
 | |
|         int sz = len;
 | |
|         if (sz > 1000)
 | |
|             sz = 1000;
 | |
|         UsbPacket pkt = {(uint16_t)(6 + sz), 0, USB_MAGIC, code, {}};
 | |
|         memcpy(pkt.buf, data, sz);
 | |
|         write(usbFD, &pkt, sizeof(pkt));
 | |
|         len -= sz;
 | |
|         data += sz;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void sendSerial(const char *data, int len) {
 | |
|     sendUsb(USB_SERIAL, data, len);
 | |
| }
 | |
| 
 | |
| extern "C" void target_panic(int error_code) {
 | |
|     char buf[50];
 | |
|     snprintf(buf, sizeof(buf), "\nPANIC %d\n", error_code);
 | |
|     sendSerial(buf, strlen(buf));
 | |
|     DMESG("PANIC %d", error_code);
 | |
|     target_reset();
 | |
| }
 | |
| 
 | |
| void startUser() {
 | |
|     pthread_mutex_lock(&execMutex);
 | |
| }
 | |
| 
 | |
| void stopUser() {
 | |
|     pthread_mutex_unlock(&execMutex);
 | |
| }
 | |
| 
 | |
| void sleep_core_us(uint64_t us) {
 | |
|     struct timespec ts;
 | |
|     ts.tv_sec = us / 1000000;
 | |
|     ts.tv_nsec = (us % 1000000) * 1000;
 | |
|     while (nanosleep(&ts, &ts))
 | |
|         ;
 | |
| }
 | |
| 
 | |
| void sleep_ms(uint32_t ms) {
 | |
|     stopUser();
 | |
|     sleep_core_us(ms * 1000);
 | |
|     startUser();
 | |
| }
 | |
| 
 | |
| void sleep_us(uint64_t us) {
 | |
|     if (us > 50000) {
 | |
|         sleep_ms(us / 1000);
 | |
|     }
 | |
|     sleep_core_us(us);
 | |
| }
 | |
| 
 | |
| uint64_t currTime() {
 | |
|     struct timeval tv;
 | |
|     gettimeofday(&tv, NULL);
 | |
|     return tv.tv_sec * 1000 + tv.tv_usec / 1000;
 | |
| }
 | |
| 
 | |
| int current_time_ms() {
 | |
|     return currTime() - startTime;
 | |
| }
 | |
| 
 | |
| int getSerialNumber() {
 | |
|     return 42; // TODO
 | |
| }
 | |
| 
 | |
| void disposeThread(Thread *t) {
 | |
|     if (allThreads == t) {
 | |
|         allThreads = t->next;
 | |
|     } else {
 | |
|         for (auto tt = allThreads; tt; tt = tt->next) {
 | |
|             if (tt->next == t) {
 | |
|                 tt->next = t->next;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     decr(t->act);
 | |
|     decr(t->arg0);
 | |
|     decr(t->data0);
 | |
|     decr(t->data1);
 | |
|     pthread_cond_destroy(&t->waitCond);
 | |
|     delete t;
 | |
| }
 | |
| 
 | |
| static void runAct(Thread *thr) {
 | |
|     startUser();
 | |
|     pxt::runAction1(thr->act, thr->arg0);
 | |
|     stopUser();
 | |
|     disposeThread(thr);
 | |
| }
 | |
| 
 | |
| void setupThread(Action a, TValue arg = 0, void (*runner)(Thread *) = NULL, TValue d0 = 0,
 | |
|                  TValue d1 = 0) {
 | |
|     if (runner == NULL)
 | |
|         runner = runAct;
 | |
|     auto thr = new Thread();
 | |
|     memset(thr, 0, sizeof(Thread));
 | |
|     thr->next = allThreads;
 | |
|     allThreads = thr;
 | |
|     thr->act = incr(a);
 | |
|     thr->arg0 = incr(arg);
 | |
|     thr->data0 = incr(d0);
 | |
|     thr->data1 = incr(d1);
 | |
|     pthread_cond_init(&thr->waitCond, NULL);
 | |
|     pthread_create(&thr->pid, NULL, (void *(*)(void *))runner, thr);
 | |
|     pthread_detach(thr->pid);
 | |
| }
 | |
| 
 | |
| void runInBackground(Action a) {
 | |
|     setupThread(a);
 | |
| }
 | |
| 
 | |
| static void runFor(Thread *t) {
 | |
|     startUser();
 | |
|     while (true) {
 | |
|         pxt::runAction0(t->act);
 | |
|         sleep_ms(20);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void runForever(Action a) {
 | |
|     setupThread(a, 0, runFor);
 | |
| }
 | |
| 
 | |
| void waitForEvent(int source, int value) {
 | |
|     auto self = pthread_self();
 | |
|     for (auto t = allThreads; t; t = t->next) {
 | |
|         if (t->pid == self) {
 | |
|             pthread_mutex_lock(&eventMutex);
 | |
|             t->waitSource = source;
 | |
|             t->waitValue = value;
 | |
|             // spourious wake ups may occur they say
 | |
|             while (t->waitSource) {
 | |
|                 pthread_cond_wait(&t->waitCond, &eventMutex);
 | |
|             }
 | |
|             pthread_mutex_unlock(&eventMutex);
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
|     assert(0);
 | |
| }
 | |
| 
 | |
| static void dispatchEvent(Event &e) {
 | |
|     lastEvent = e;
 | |
| 
 | |
|     auto curr = findBinding(e.source, e.value);
 | |
|     if (curr)
 | |
|         setupThread(curr->action, fromInt(e.value));
 | |
| 
 | |
|     curr = findBinding(e.source, DEVICE_EVT_ANY);
 | |
|     if (curr)
 | |
|         setupThread(curr->action, fromInt(e.value));
 | |
| }
 | |
| 
 | |
| static void *evtDispatcher(void *dummy) {
 | |
|     pthread_mutex_lock(&eventMutex);
 | |
|     while (true) {
 | |
|         pthread_cond_wait(&newEventBroadcast, &eventMutex);
 | |
|         while (eventHead != NULL) {
 | |
|             Event *ev = eventHead;
 | |
|             eventHead = ev->next;
 | |
|             if (eventHead == NULL)
 | |
|                 eventTail = NULL;
 | |
| 
 | |
|             for (auto thr = allThreads; thr; thr = thr->next) {
 | |
|                 if (thr->waitSource == 0)
 | |
|                     continue;
 | |
|                 if (thr->waitValue != ev->value && thr->waitValue != DEVICE_EVT_ANY)
 | |
|                     continue;
 | |
|                 if (thr->waitSource == ev->source) {
 | |
|                     thr->waitSource = 0; // once!
 | |
|                     pthread_cond_broadcast(&thr->waitCond);
 | |
|                 } else if (thr->waitSource == DEVICE_ID_NOTIFY &&
 | |
|                            ev->source == DEVICE_ID_NOTIFY_ONE) {
 | |
|                     thr->waitSource = 0; // once!
 | |
|                     pthread_cond_broadcast(&thr->waitCond);
 | |
|                     break; // do not wake up any other threads
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             dispatchEvent(*ev);
 | |
|             delete ev;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| int allocateNotifyEvent() {
 | |
|     static volatile int notifyId;
 | |
|     pthread_mutex_lock(&eventMutex);
 | |
|     int res = ++notifyId;
 | |
|     pthread_mutex_unlock(&eventMutex);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| void raiseEvent(int id, int event) {
 | |
|     auto e = mkEvent(id, event);
 | |
|     pthread_mutex_lock(&eventMutex);
 | |
|     if (eventTail == NULL) {
 | |
|         assert(eventHead == NULL);
 | |
|         eventHead = eventTail = e;
 | |
|     } else {
 | |
|         eventTail->next = e;
 | |
|         eventTail = e;
 | |
|     }
 | |
|     pthread_cond_broadcast(&newEventBroadcast);
 | |
|     pthread_mutex_unlock(&eventMutex);
 | |
| }
 | |
| 
 | |
| void registerWithDal(int id, int event, Action a) {
 | |
|     setBinding(id, event, a);
 | |
| }
 | |
| 
 | |
| static void runPoller(Thread *thr) {
 | |
|     Action query = thr->data0;
 | |
|     auto us = (uint64_t)toInt(thr->data1) * 1000;
 | |
| 
 | |
|     // note that this is run without the user mutex held - it should not modify any state!
 | |
|     TValue prev = pxt::runAction0(query);
 | |
| 
 | |
|     while (true) {
 | |
|         sleep_core_us(us);
 | |
|         TValue curr = pxt::runAction0(query);
 | |
|         if (curr != prev) {
 | |
|             startUser();
 | |
|             pxt::runAction2(thr->act, prev, curr);
 | |
|             stopUser();
 | |
|             decr(prev);
 | |
|             prev = curr;
 | |
|         }
 | |
|     }
 | |
|     //    disposeThread(thr);
 | |
| }
 | |
| 
 | |
| //%
 | |
| void unsafePollForChanges(int ms, Action query, Action handler) {
 | |
|     setupThread(handler, 0, runPoller, query, fromInt(ms));
 | |
| }
 | |
| 
 | |
| uint32_t afterProgramPage() {
 | |
|     return 0;
 | |
| }
 | |
| void dumpDmesg() {
 | |
|     sendSerial("\nDMESG:\n", 8);
 | |
|     sendSerial(dmesgBuf, dmesgPtr);
 | |
|     sendSerial("\n\n", 2);
 | |
| }
 | |
| 
 | |
| int lmsPid;
 | |
| void stopLMS() {
 | |
|     struct dirent *ent;
 | |
|     DIR *dir;
 | |
| 
 | |
|     dir = opendir("/proc");
 | |
|     if (dir == NULL)
 | |
|         return;
 | |
| 
 | |
|     while ((ent = readdir(dir)) != NULL) {
 | |
|         int pid = atoi(ent->d_name);
 | |
|         if (!pid)
 | |
|             continue;
 | |
|         char namebuf[100];
 | |
|         snprintf(namebuf, 1000, "/proc/%d/cmdline", pid);
 | |
|         FILE *f = fopen(namebuf, "r");
 | |
|         if (f) {
 | |
|             fread(namebuf, 1, 99, f);
 | |
|             if (strcmp(namebuf, "./lms2012") == 0) {
 | |
|                 lmsPid = pid;
 | |
|             }
 | |
|             fclose(f);
 | |
|             if (lmsPid)
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     closedir(dir);
 | |
| 
 | |
|     lmsPid = 0; // disable SIGSTOP for now - rethink if problems with I2C (runs on a thread)
 | |
| 
 | |
|     if (lmsPid) {
 | |
|         DMESG("SIGSTOP to lmsPID=%d", lmsPid);
 | |
|         if (kill(lmsPid, SIGSTOP))
 | |
|             DMESG("SIGSTOP failed");
 | |
|     }
 | |
| }
 | |
| 
 | |
| void runLMS() {
 | |
|     DMESG("re-starting LMS2012");
 | |
|     kill(lmsPid, SIGCONT);
 | |
|     sleep_core_us(200000);
 | |
|     exit(0);
 | |
|     /*
 | |
|     chdir("/home/root/lms2012/sys");
 | |
|     for (int fd = 3; fd < 9999; ++fd)
 | |
|         close(fd);
 | |
|     execl("lms2012", "./lms2012");
 | |
|     exit(100); // should not be reached
 | |
|     */
 | |
| }
 | |
| 
 | |
| extern "C" void target_reset() {
 | |
|     if (lmsPid)
 | |
|         runLMS();
 | |
|     else
 | |
|         exit(0);
 | |
| }
 | |
| 
 | |
| void screen_init();
 | |
| void initRuntime() {
 | |
|     // daemon(1, 1);
 | |
|     startTime = currTime();
 | |
|     DMESG("runtime starting...");
 | |
|     stopLMS();
 | |
|     pthread_t disp;
 | |
|     pthread_create(&disp, NULL, evtDispatcher, NULL);
 | |
|     pthread_detach(disp);
 | |
|     target_init();
 | |
|     screen_init();
 | |
|     startUser();
 | |
| }
 | |
| 
 | |
| static FILE *dmesgFile;
 | |
| 
 | |
| void dmesgRaw(const char *buf, uint32_t len) {
 | |
|     if (!dmesgFile) {
 | |
|         dmesgFile = fopen("/tmp/dmesg.txt", "w");
 | |
|         if (!dmesgFile)
 | |
|             dmesgFile = stderr;
 | |
|     }
 | |
| 
 | |
|     if (len > sizeof(dmesgBuf) / 2)
 | |
|         return;
 | |
|     if (dmesgPtr + len > sizeof(dmesgBuf)) {
 | |
|         dmesgPtr = 0;
 | |
|     }
 | |
|     memcpy(dmesgBuf + dmesgPtr, buf, len);
 | |
|     dmesgPtr += len;
 | |
|     fwrite(buf, 1, len, dmesgFile);
 | |
| }
 | |
| 
 | |
| void dmesg(const char *format, ...) {
 | |
|     char buf[500];
 | |
| 
 | |
|     snprintf(buf, sizeof(buf), "[%8d] ", current_time_ms());
 | |
|     dmesgRaw(buf, strlen(buf));
 | |
| 
 | |
|     va_list arg;
 | |
|     va_start(arg, format);
 | |
|     vsnprintf(buf, sizeof(buf), format, arg);
 | |
|     va_end(arg);
 | |
|     dmesgRaw(buf, strlen(buf));
 | |
|     dmesgRaw("\n", 1);
 | |
| 
 | |
|     fflush(dmesgFile);
 | |
|     fdatasync(fileno(dmesgFile));
 | |
| }
 | |
| }
 |