#include <esp8266.h> #include <httpd.h> #include "page_waveform.h" #include "ftoa.h" #include "sampling.h" #include "serial.h" #include "payload_parser.h" // ------------------------------------------------------------------------------- // Read multiple samples from the ADC as JSON typedef struct { uint16_t total_count; uint16_t done_count; uint32_t freq; bool success; } tplReadSamplesJSON_state; static int FLASH_FN tplSamplesJSON(MEAS_FORMAT fmt, HttpdConnData *connData, char *token, void **arg); int FLASH_FN tplWaveformJSON(HttpdConnData *connData, char *token, void **arg) { return tplSamplesJSON(RAW, connData, token, arg); } int FLASH_FN tplFourierJSON(HttpdConnData *connData, char *token, void **arg) { return tplSamplesJSON(FFT, connData, token, arg); } static int FLASH_FN tplSamplesJSON(MEAS_FORMAT fmt, HttpdConnData *connData, char *token, void **arg) { char buff[128]; int len; tplReadSamplesJSON_state *st = *arg; if (token == NULL) { // end of template, cleanup if (st != NULL) free(st); return HTTPD_CGI_DONE; // cleanup } if (st == NULL) { // first call - allocate the struct st = malloc(sizeof(tplReadSamplesJSON_state)); *arg = st; // check how many samples are requested uint16_t count = 1; len = httpdFindArg(connData->getArgs, "n", buff, sizeof(buff)); if (len != -1) count = (uint16_t)atoi(buff); if (count == 0) { error("Count == 0"); st->success = false; return HTTPD_CGI_DONE; } uint32_t freq = 0; len = httpdFindArg(connData->getArgs, "fs", buff, sizeof(buff)); if (len != -1) freq = (uint32_t)atoi(buff); if (freq == 0) { error("Freq == 0"); st->success = false; return HTTPD_CGI_DONE; } st->total_count = count; st->done_count = 0; st->freq = freq; // --- REQUEST THE DATA --- // This actually asks the STM32 over SBMP to start the ADC DMA, // and we'll wait for the data to arrive. st->success = meas_request_data(fmt, count, freq); if (!st->success) { error("Failed to start sampling"); } } // the "success" field is after the data, // so if readout fails, success can be set to false. if (strcmp(token, "values") == 0) { if (!st->success) { // failed to start sampling return HTTPD_CGI_DONE; } // Wait for a chunk uint8_t *chunk = NULL; uint16_t chunk_len = 0; // 10 secs or 100 ms - longer wait for intial data. int timeout = (st->done_count == 0 ? (int)meas_estimate_duration(st->total_count, st->freq): SAMP_READOUT_TMEO); for (int i = 0; i < timeout*100; i++) { uart_poll(); if (meas_chunk_ready() || meas_is_closed()) break; // We have some data! --- or transaction aborted by peer :( os_delay_us(10); // 1 ms system_soft_wdt_feed(); // Feed the dog, or it'll bite. } chunk = meas_get_chunk(&chunk_len); if (chunk == NULL) { // abort, proceed to the next field. meas_close(); st->success = false; return HTTPD_CGI_DONE; } PayloadParser pp = pp_start(chunk, chunk_len); // chunk of data... for (; pp.ptr < pp.len; st->done_count++) { // preceding comma if not the first number if (st->done_count > 0) { httpdSend(connData, ", ", 2); } float samp = pp_float(&pp); my_ftoa(buff, samp, 3); httpdSend(connData, buff, -1); } // wait for more data in this % tag if (!meas_is_last_chunk()) { meas_request_next_chunk(); return HTTPD_CGI_MORE; // more numbers to come } else { // we're done meas_close(); return HTTPD_CGI_DONE; } } else if (strcmp(token, "stats") == 0) { // the STATS json block if (!st->success) { httpdSend(connData, "null", 4); } else { // no %f in sprintf :( MeasStats *stats = meas_get_stats(); httpdSend(connData, "{", 1); sprintf(buff, "\"count\": %d, ", stats->count); httpdSend(connData, buff, -1); httpdSend(connData, "\"freq\": ", -1); my_ftoa(buff, stats->freq, 3); httpdSend(connData, buff, -1); httpdSend(connData, ", \"min\": ", -1); my_ftoa(buff, stats->min, 3); httpdSend(connData, buff, -1); httpdSend(connData, ", \"max\": ", -1); my_ftoa(buff, stats->max, 3); httpdSend(connData, buff, -1); httpdSend(connData, ", \"rms\": ", -1); my_ftoa(buff, stats->rms, 3); httpdSend(connData, buff, -1); if (fmt == FFT) { // ... maybe something special for fft ... } sprintf(buff, ", \"format\": \"%s\"", fmt == RAW ? "RAW" : fmt == FFT ? "FFT" : "UNKNOWN"); httpdSend(connData, buff, -1); httpdSend(connData, "}", 1); } } else if (strcmp(token, "success") == 0) { // success status httpdSend(connData, (st->success ? "true" : "false"), -1); } return HTTPD_CGI_DONE; }