2021-05-11 09:12:06 +01:00
|
|
|
// Copyright (c) 2021 Cesanta Software Limited
|
|
|
|
// All rights reserved
|
|
|
|
|
|
|
|
#include "device.h"
|
|
|
|
#include "mongoose.h"
|
|
|
|
|
2021-05-17 17:36:57 +01:00
|
|
|
static const char *s_debug_level = "4";
|
2021-05-12 08:43:34 +01:00
|
|
|
static const char *s_listening_address = "http://0.0.0.0:80";
|
2021-05-11 09:12:06 +01:00
|
|
|
|
|
|
|
// Event handler for the listening connection.
|
2021-05-12 08:43:34 +01:00
|
|
|
static void cb(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
|
|
|
if (ev == MG_EV_HTTP_MSG) {
|
2021-05-19 08:10:38 +01:00
|
|
|
#if MG_ENABLE_FS
|
|
|
|
struct mg_http_serve_opts opts = {.root_dir = "/"};
|
|
|
|
mg_http_serve_dir(c, ev_data, &opts);
|
|
|
|
#else
|
2021-05-12 08:43:34 +01:00
|
|
|
mg_http_reply(c, 200, "", "hello, %s!\n", "world");
|
2021-05-19 08:10:38 +01:00
|
|
|
#endif
|
2021-05-12 08:43:34 +01:00
|
|
|
}
|
|
|
|
(void) fn_data;
|
|
|
|
(void) ev_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void server(void *args) {
|
|
|
|
struct mg_mgr mgr;
|
|
|
|
mg_log_set(s_debug_level);
|
|
|
|
mg_mgr_init(&mgr);
|
2021-05-17 17:36:57 +01:00
|
|
|
LOG(LL_INFO, ("Starting Mongoose v%s", MG_VERSION)); // Tell the world
|
2021-05-12 08:43:34 +01:00
|
|
|
mg_http_listen(&mgr, s_listening_address, cb, &mgr); // Web listener
|
|
|
|
while (args == NULL) mg_mgr_poll(&mgr, 1000); // Infinite event loop
|
|
|
|
mg_mgr_free(&mgr); // Unreachable
|
|
|
|
}
|
|
|
|
|
|
|
|
static void blinker(void *args) {
|
2021-05-13 15:05:09 +01:00
|
|
|
uint16_t pin = ((char *) args)[0] == '1' ? LED2 : LED3;
|
2021-05-17 17:36:57 +01:00
|
|
|
int ms = pin == LED2 ? 3750 : 5130;
|
2021-05-13 15:05:09 +01:00
|
|
|
for (;;) {
|
|
|
|
gpio_toggle(pin);
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(ms));
|
2021-05-17 17:36:57 +01:00
|
|
|
LOG(LL_INFO, ("blink %s, RAM: %u", (char *) args, xPortGetFreeHeapSize()));
|
2021-05-12 08:43:34 +01:00
|
|
|
}
|
2021-05-11 09:12:06 +01:00
|
|
|
}
|
|
|
|
|
2021-05-17 17:36:57 +01:00
|
|
|
// Start Mongoose server when network is ready
|
|
|
|
void vApplicationIPNetworkEventHook(eIPCallbackEvent_t ev) {
|
|
|
|
static bool mongoose_started = false;
|
|
|
|
LOG(LL_INFO, ("FreeRTOS net event %d, up: %d", ev, eNetworkUp));
|
|
|
|
if (ev == eNetworkUp && mongoose_started == false) {
|
|
|
|
xTaskCreate(server, "server", 8192, NULL, configMAX_PRIORITIES - 1, NULL);
|
|
|
|
mongoose_started = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void init_heap(void) {
|
|
|
|
extern uint32_t _end, _estack;
|
|
|
|
uint8_t *ptr = (uint8_t *) ((((uint32_t) &_end) + 7) & ~7U);
|
|
|
|
uint32_t len = (uint32_t)((char *) &_estack - (char *) &_end);
|
|
|
|
HeapRegion_t regions[] = {{ptr, len}, {NULL, 0}};
|
|
|
|
vPortDefineHeapRegions(regions);
|
|
|
|
}
|
|
|
|
|
2021-05-11 09:12:06 +01:00
|
|
|
int main(void) {
|
|
|
|
init_hardware();
|
2021-05-17 17:36:57 +01:00
|
|
|
init_heap();
|
|
|
|
|
|
|
|
// Initialise networking task. Can be done before scheduler is started
|
|
|
|
static const uint8_t macaddr[6] = {0x00, 0x11, 0x22, 0x33, 0x44, 0x55};
|
|
|
|
static const uint8_t ipaddr[4] = {192, 168, 0, 77};
|
|
|
|
static const uint8_t netmask[4] = {255, 255, 255, 0};
|
|
|
|
static const uint8_t dnsaddr[4] = {8, 8, 8, 8};
|
|
|
|
static const uint8_t gwaddr[4] = {192, 168, 0, 1};
|
|
|
|
FreeRTOS_IPInit(ipaddr, netmask, gwaddr, dnsaddr, macaddr);
|
|
|
|
|
|
|
|
xTaskCreate(blinker, "blinker", 512, "1", configMAX_PRIORITIES - 1, NULL);
|
|
|
|
xTaskCreate(blinker, "blinker", 512, "2", configMAX_PRIORITIES - 1, NULL);
|
2021-05-11 20:44:14 +01:00
|
|
|
vTaskStartScheduler(); // This blocks
|
|
|
|
return 0; // Unreachable
|
2021-05-11 09:12:06 +01:00
|
|
|
}
|
2021-05-12 08:43:34 +01:00
|
|
|
|
|
|
|
// Stubs for FreeRTOS-TCP network interface
|
2021-05-13 22:03:46 +01:00
|
|
|
uint32_t ulApplicationGetNextSequenceNumber(uint32_t a, uint16_t b, uint32_t c,
|
|
|
|
uint16_t d) {
|
|
|
|
return a ^ b ^ c ^ d;
|
|
|
|
}
|
2021-05-17 17:36:57 +01:00
|
|
|
|
2021-05-13 22:03:46 +01:00
|
|
|
BaseType_t xApplicationGetRandomNumber(uint32_t *p) {
|
|
|
|
*p = 0;
|
|
|
|
return 1;
|
|
|
|
}
|
2021-05-17 17:36:57 +01:00
|
|
|
|
2021-05-13 22:03:46 +01:00
|
|
|
uint32_t SystemCoreClock = 216000000;
|
|
|
|
uint32_t HAL_RCC_GetHCLKFreq(void) {
|
|
|
|
return SystemCoreClock;
|
2021-05-12 08:43:34 +01:00
|
|
|
}
|
2021-05-17 17:36:57 +01:00
|
|
|
|
|
|
|
uint32_t HAL_GetTick(void) {
|
|
|
|
return configTICK_RATE_HZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "stm32f7xx_hal_conf.h"
|
|
|
|
#include "stm32fxx_hal_eth.h"
|
|
|
|
void HAL_ETH_MspInit( ETH_HandleTypeDef* heth ) {
|
|
|
|
GPIO_InitTypeDef GPIO_InitStructure;
|
|
|
|
if (heth->Instance == ETH) {
|
|
|
|
__HAL_RCC_GPIOA_CLK_ENABLE();
|
|
|
|
__HAL_RCC_GPIOC_CLK_ENABLE();
|
|
|
|
__HAL_RCC_GPIOG_CLK_ENABLE();
|
|
|
|
|
|
|
|
GPIO_InitStructure.Speed = GPIO_SPEED_HIGH;
|
|
|
|
GPIO_InitStructure.Mode = GPIO_MODE_AF_PP;
|
|
|
|
GPIO_InitStructure.Pull = GPIO_NOPULL;
|
|
|
|
GPIO_InitStructure.Alternate = GPIO_AF11_ETH;
|
|
|
|
GPIO_InitStructure.Pin = GPIO_PIN_1 | GPIO_PIN_2 | GPIO_PIN_7;
|
|
|
|
HAL_GPIO_Init(GPIOA, &GPIO_InitStructure);
|
|
|
|
|
|
|
|
GPIO_InitStructure.Pin = GPIO_PIN_1 | GPIO_PIN_4 | GPIO_PIN_5;
|
|
|
|
HAL_GPIO_Init(GPIOC, &GPIO_InitStructure);
|
|
|
|
|
|
|
|
GPIO_InitStructure.Pin = GPIO_PIN_2 | GPIO_PIN_11 | GPIO_PIN_13 | GPIO_PIN_14;
|
|
|
|
HAL_GPIO_Init(GPIOG, &GPIO_InitStructure);
|
|
|
|
|
|
|
|
HAL_NVIC_SetPriority(ETH_IRQn, 0x7, 0);
|
|
|
|
HAL_NVIC_EnableIRQ(ETH_IRQn);
|
|
|
|
|
|
|
|
__HAL_RCC_ETH_CLK_ENABLE();
|
|
|
|
HAL_NVIC_SetPriorityGrouping( NVIC_PRIORITYGROUP_4 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int vLoggingPrintf(const char *fmt, ...) {
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
char buf[100];
|
|
|
|
int n = vsnprintf(buf, sizeof(buf), fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
write(1, buf, n);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *_sbrk(ptrdiff_t incr) {
|
|
|
|
(void) incr;
|
|
|
|
return NULL;
|
2021-05-19 08:10:38 +01:00
|
|
|
}
|