prettyprint

2023年12月21日 星期四

Raspberry Pi Pico W || LwIP || MbedTLS: EP.2 HTTPS server Session management

本文章介紹Raspberry Pi Pico W使用LwIP & MbedTLS,利用Cookie建立Session管理。

實驗建立需要登入驗證的網頁與無須登入驗證的網頁兩類別。當開啟需要驗證的網頁時,若尚未登入,則重新導到驗證頁面。



將需要驗證後才能使用的uri存在list中


軟體主要使用LwIP httpd application, 加入Cookies功能的patch後,利用cookie建立client與server的session管理。

logout 清除client 端session cookie,下次開啟需要驗證的網頁時,需再登入一次。

為了實現session management增加三個實作functions:

char *httpd_set_cookies(const void *connection, const char *uri);
執行上述patch後在httpd.c新增的function call,用於加入Set-Cookie: header。

void httpd_received_cookies(const void *connection, char **cookies);
執行上述patch後在httpd.c新增的function call,本專案修改cookies的parameter屬性,因為server在收到 client送來cookie中的session-id,若在server端已不存在(也許timeout), 將重設cookie為NULL。

void httpd_redirect_uri(char* uri, u16_t *uri_len);
若client的cookie為NULL,則server 
將重新導到登入頁面。

每個session設有timeout時間,當在設定的timeout時間內沒有任何存取網頁,則清除server上該session object。當每次存取網頁時就會重設timeout時間。


流程如下圖所示:
本專案測試https與http service同時存在:

HTTPS:


HTTP:


當cookie設定 Secure ,則只能以https方式傳送cookie。因此以http開啟網頁時,因為cookie設定為Secure,即使登入成功,browser也不會送出cookie,因此server將認定為未登入。

詳細程式碼附於文章後面。實驗影片請參閱成果影片。

成果影片





程式碼

  • picow_https_auth.c(main)
  • #include <stdio.h>
    #include "pico/stdlib.h"
    #include "pico/cyw43_arch.h"
    #include "hardware/adc.h"
    #include "lwip/altcp_tls.h"
    #include "lwip/altcp_tcp.h"
    #include "lwip/altcp.h"
    #include "lwip/apps/httpd.h"
    #include "lwip/apps/sntp.h"
    #include "picow_https_cert.h"
    #include "ff.h"
    #include "pico_storage.h"
    #include "mbedtls/ssl.h"
    #define LIGHT_PIN 16
    #define WIFI_SSID "your-ssid"
    #define WIFI_PASSWORD "your-password"
    struct altcp_tls_config {
    mbedtls_ssl_config conf;
    mbedtls_x509_crt *cert;
    mbedtls_pk_context *pkey;
    u8_t cert_count;
    u8_t cert_max;
    u8_t pkey_count;
    u8_t pkey_max;
    mbedtls_x509_crt *ca;
    #if defined(MBEDTLS_SSL_CACHE_C) && ALTCP_MBEDTLS_USE_SESSION_CACHE
    /** Inter-connection cache for fast connection startup */
    struct mbedtls_ssl_cache_context cache;
    #endif
    #if defined(MBEDTLS_SSL_SESSION_TICKETS) && ALTCP_MBEDTLS_USE_SESSION_TICKETS
    mbedtls_ssl_ticket_context ticket_ctx;
    #endif
    };
    struct altcp_tls_config *tls_config = NULL;
    void fs_ex_init(const char *httpd_root_dir);
    void netif_status_cb(struct netif *netif) {
    if (netif_is_link_up(netif)) {
    ip_addr_t ipaddr;
    ip_addr_t netmask;
    ip_addr_t gateway;
    ipaddr_aton("192.168.1.70", &ipaddr);
    ipaddr_aton("255.255.255.0", &netmask);
    ipaddr_aton("192.168.1.1", &gateway);
    netif_set_addr(netif, &ipaddr, &netmask, &gateway);
    }
    }
    bool sntp_time_geted=false;
    void sntp_set_system_time(uint32_t sec, uint32_t us)
    {
    struct tm info;
    time_t tim = sec;
    struct timeval tmv;
    tmv.tv_sec=sec;
    tmv.tv_usec=us;
    settimeofday(&tmv, 0);
    printf("set time :%s\n", ctime(&tim));
    sntp_time_geted=true;
    }
    void ssi_handler_init();
    void cgi_handler_init();
    void format_current_time(char *time_str, uint8_t count) {
    while(!sntp_time_geted) {
    cyw43_arch_poll();
    tight_loop_contents();
    };
    time_t timer=time(NULL);
    struct tm *tm_info = localtime(&timer);
    strftime(time_str, count, "%a, %d-%b-%Y %X %Z", tm_info);
    }
    #include <malloc.h>
    uint32_t getTotalHeap(void) {
    extern char __StackLimit, __bss_end__;
    return &__StackLimit - &__bss_end__;
    }
    uint32_t getFreeHeap(void) {
    struct mallinfo m = mallinfo();
    return getTotalHeap() - m.uordblks;
    }
    DWORD get_sd_free_size() {
    FATFS *fs;
    DWORD fre_clust, fre_sect, tot_sect;
    /* Get volume information and free clusters of drive 1 */
    if (f_getfree(SDMMC_PATH, &fre_clust, &fs)!= FR_OK) return 0;
    /* Get total sectors and free sectors */
    tot_sect = (fs->n_fatent - 2) * fs->csize;
    fre_sect = fre_clust * fs->csize;
    /* Print the free space (assuming 512 bytes/sector) */
    // printf("%10lu KiB total drive space.\n%10lu KiB available.\n", tot_sect / 2, fre_sect / 2);
    return fre_sect / 2;
    }
    float read_onboard_temperature() {
    /* 12-bit conversion, assume max value == ADC_VREF == 3.3 V */
    const float conversionFactor = 3.3f / (1 << 12);
    float adc = (float)adc_read() * conversionFactor;
    float tempC = 27.0f - (adc - 0.706f) / 0.001721f;
    return tempC;
    }
    extern void check_timeout_sessions_timer();
    void system_error_blink() {
    while(1) {
    cyw43_arch_gpio_put(CYW43_WL_GPIO_LED_PIN, !cyw43_arch_gpio_get(CYW43_WL_GPIO_LED_PIN));
    sleep_ms(500);
    }
    }
    int main()
    {
    stdio_init_all();
    adc_init();
    adc_set_temp_sensor_enabled(true);
    adc_select_input(4);
    if (cyw43_arch_init()) {
    printf("cyw43 arch init error\n");
    system_error_blink();
    }
    gpio_init(LIGHT_PIN);
    gpio_set_dir(LIGHT_PIN, true);
    gpio_put(LIGHT_PIN, 0);
    //1. connect to WiFi network and set static IP
    cyw43_arch_enable_sta_mode();
    netif_set_status_callback(netif_default, netif_status_cb);
    int wifi_stat = cyw43_arch_wifi_connect_timeout_ms(WIFI_SSID, WIFI_PASSWORD, CYW43_AUTH_WPA2_AES_PSK, 30000);
    if (wifi_stat) {
    printf("wifi connect error:%d\n", wifi_stat);
    system_error_blink();
    }
    printf("Wifi connected\n");
    printf("get ip addr:%s\n",ipaddr_ntoa(&(cyw43_state.netif[0].ip_addr)));
    // 2. set SNTP
    sntp_setoperatingmode(SNTP_OPMODE_POLL);
    sntp_init();
    // 3. set TLS connection
    tls_config = altcp_tls_create_config_server_privkey_cert(SERVER_KEY, sizeof(SERVER_KEY), NULL, 0, SERVER_CERT, sizeof(SERVER_CERT));
    mbedtls_x509_crt_parse(tls_config->cert, CA_CERT, sizeof(CA_CERT));
    // for large file size
    mbedtls_ssl_conf_max_frag_len( &tls_config->conf, MBEDTLS_SSL_MAX_FRAG_LEN_4096 );
    //4. mount FS in SD card and start https server
    FATFS fs;
    FIL fil;
    FRESULT res;
    if (f_mount(&fs, SDMMC_PATH, 1) != FR_OK) {
    printf("mount error\n");
    system_error_blink();
    }
    int i;
    ssi_handler_init();
    cgi_handler_init();
    check_timeout_sessions_timer();
    fs_ex_init(SDMMC_PATH"/");
    httpd_inits(tls_config);
    //httpd_init();
    char buf[61];
    format_current_time(buf, 60);
    printf(buf);
    cyw43_arch_gpio_put(CYW43_WL_GPIO_LED_PIN,1);
    while(1) {
    cyw43_arch_poll();
    sleep_ms(50);
    }
    return 0;
    }
  • picow_https_ssi.c

  • #include <stdio.h>
    #include "pico/stdlib.h"
    #include "pico/cyw43_arch.h"
    #include "lwip/altcp_tls.h"
    #include "lwip/altcp_tcp.h"
    #include "lwip/altcp.h"
    #include "lwip/apps/httpd.h"
    #include "lwip/apps/sntp.h"
    #include "ff.h"
    #define DOMAIN "httpsserver"
    #define SESSION_TIMEOUT_SEC (10*60)
    #define AUTH_USER "picow"
    #define AUTH_PASS "passw0rd"
    #define LIGHT_PIN 16
    char* session_check_uri[]={
    "/light.shtml",
    "/demo1.shtml",
    "/demo2.shtml",
    "/light.cgi"
    };
    typedef struct _sess_id_t{
    uint8_t sess_id[26];
    absolute_time_t timeout;
    struct _sess_id_t *next;
    } sess_id_t;
    typedef struct _conn_t {
    void *connection;
    time_t start_time;
    enum {
    NORMAL_PAGE=0,
    LOGIN_PAGE=1,
    LOGIN_SUCCESSFUL=2,
    LOGOUT_PAGE=3,
    } verified;
    struct _conn_t *next;
    } conn_t;
    char light_img[20];
    sess_id_t *sess_header=NULL;
    conn_t * conn_header=NULL;
    void gen_sess_cookie(char *sess_id, char* sess_cookie, uint8_t count) {
    char time_str[40];
    uint8_t sid[26];
    time_t timer=time(NULL);
    struct tm *tm_info = localtime(&timer);
    srand((unsigned)time(NULL));
    sprintf(sid, "sess_%04d%04d%04d",rand()%10000, rand()%10000, rand()%10000);
    printf("sess_id:%s\n", sid);
    sprintf(sess_id, sid);
    timer+=86400; // 1 day
    tm_info = localtime(&timer);
    strftime(time_str, 40, "%a, %d-%b-%Y %X %Z", tm_info);
    sprintf(sess_cookie, "sess_id=%s; expires=%s; domain=%s; path=/; Secure", sid, time_str, DOMAIN);
    //sprintf(sess_cookie, "sess_id=%s; expires=%s; domain=%s; path=/; SameSite=Lax", sid, time_str, DOMAIN);
    }
    void new_session_id(char *sid) {
    sess_id_t *newid = (sess_id_t*)calloc(1,sizeof(sess_id_t));
    newid->next = NULL;
    newid->timeout=make_timeout_time_ms(SESSION_TIMEOUT_SEC*1000); // 10 min
    strcpy(newid->sess_id, sid);
    sess_id_t *tidp;
    if (sess_header == NULL) {
    sess_header = newid;
    return;
    } else {
    tidp = sess_header;
    while (tidp->next != NULL) {
    tidp = tidp->next;
    }
    }
    tidp->next=newid;
    return;
    }
    extern DWORD get_sd_free_size();
    extern float read_onboard_temperature();
    extern uint32_t getFreeHeap(void);
    /* ==== cgi begin ======*/
    const char *
    cgi_light_handler(int iIndex, int iNumParams, char *pcParam[], char *pcValue[]) {
    if (iIndex == 0 && iNumParams ==1) {
    if (strcmp(pcParam[0],"action") == 0) {
    strcpy(light_img,"");
    if (strcmp(pcValue[0], "ON")==0) {
    gpio_put(LIGHT_PIN, 1);
    strcpy(light_img, "/img/light_on.png");
    }
    if (strcmp(pcValue[0],"OFF")==0) {
    gpio_put(LIGHT_PIN, 0);
    strcpy(light_img, "/img/light_off.png");
    }
    return "/light.shtml";
    }
    }
    return "/404.html";
    }
    static const tCGI cgi_handlers[] = {
    {
    "/light.cgi", cgi_light_handler
    },
    };
    void cgi_handler_init() {
    http_set_cgi_handlers(cgi_handlers, LWIP_ARRAYSIZE(cgi_handlers));
    }
    /* ==== cgi end ======*/
    /* === ssi begin =====*/
    const char* ssi_tags[] = {
    "temp",
    "freeram",
    "freesd",
    "logout",
    "light",
    };
    u16_t __time_critical_func(ssi_handler)(int iIndex, char *pcInsert, int iInsertLen, u16_t current_tag_part, u16_t *next_tag_part)
    {
    size_t printed;
    static uint32_t counter=0;
    switch (iIndex) {
    case 0: //temp, Temperature
    printed = snprintf(pcInsert, iInsertLen, "%.02f°C", read_onboard_temperature());
    break;
    case 1: // free RAM
    printed = sniprintf(pcInsert, iInsertLen, "%d B", getFreeHeap());
    break;
    case 2: // SD Free Size
    printed = sniprintf(pcInsert, iInsertLen, "%ld KB", get_sd_free_size());
    break;
    case 3: // logout uri
    printed = snprintf(pcInsert, iInsertLen,
    "<button onClick='javascript:logout();'>Logout</button>");
    break;
    case 4: // light
    printed = snprintf(pcInsert, iInsertLen,light_img);
    break;
    default:
    printed = snprintf(pcInsert, iInsertLen, "");
    }
    return printed;
    }
    void ssi_handler_init() {
    http_set_ssi_handler(ssi_handler,
    #if LWIP_HTTPD_SSI_RAW
    NULL, 0
    #else
    ssi_tags, LWIP_ARRAYSIZE(ssi_tags)
    #endif
    );
    if (gpio_get(LIGHT_PIN))
    strcpy(light_img, "/img/light_on.png");
    else
    strcpy(light_img, "/img/light_off.png");
    }
    /* === ssi end =====*/
    /*===== post begin =====*/
    #define USER_PASS_BUFSIZE 16
    err_t
    httpd_post_begin(void *connection, const char *uri, const char *http_request,
    u16_t http_request_len, int content_len, char *response_uri,
    u16_t response_uri_len, u8_t *post_auto_wnd)
    {
    LWIP_UNUSED_ARG(connection);
    LWIP_UNUSED_ARG(http_request);
    LWIP_UNUSED_ARG(http_request_len);
    LWIP_UNUSED_ARG(content_len);
    LWIP_UNUSED_ARG(post_auto_wnd);
    conn_t *conn = (conn_t*)calloc(1, sizeof(conn_t));
    conn->connection = connection;
    conn->start_time = time(NULL);
    conn->verified = NORMAL_PAGE;
    conn->next=NULL;
    if (conn_header == NULL) {
    conn_header = conn;
    } else {
    conn->next=conn_header;
    conn_header = conn;
    }
    if (!memcmp(uri, "/login.cgi", 11)) {
    conn->verified = LOGIN_PAGE;
    snprintf(response_uri, response_uri_len, "/loginfail.html");
    *post_auto_wnd = 1;
    return ERR_OK;
    }
    return ERR_VAL;
    }
    err_t
    httpd_post_receive_data(void *connection, struct pbuf *p)
    {
    err_t ret;
    LWIP_ASSERT("NULL pbuf", p != NULL);
    ret = ERR_VAL;
    conn_t *start = conn_header;
    while(start != NULL) {
    if (start->connection == connection) {
    if (start->verified == LOGIN_PAGE) {
    u16_t token_user = pbuf_memfind(p, "user=", 5, 0);
    u16_t token_pass = pbuf_memfind(p, "pass=", 5, 0);
    if ((token_user != 0xFFFF) && (token_pass != 0xFFFF)) {
    u16_t value_user = token_user + 5;
    u16_t value_pass = token_pass + 5;
    u16_t len_user = 0;
    u16_t len_pass = 0;
    u16_t tmp;
    /* find user len */
    tmp = pbuf_memfind(p, "&", 1, value_user);
    if (tmp != 0xFFFF) {
    len_user = tmp - value_user;
    } else {
    len_user = p->tot_len - value_user;
    }
    /* find pass len */
    tmp = pbuf_memfind(p, "&", 1, value_pass);
    if (tmp != 0xFFFF) {
    len_pass = tmp - value_pass;
    } else {
    len_pass = p->tot_len - value_pass;
    }
    if ((len_user > 0) && (len_user < USER_PASS_BUFSIZE) &&
    (len_pass > 0) && (len_pass < USER_PASS_BUFSIZE)) {
    /* provide contiguous storage if p is a chained pbuf */
    char buf_user[USER_PASS_BUFSIZE];
    char buf_pass[USER_PASS_BUFSIZE];
    char *user = (char *)pbuf_get_contiguous(p, buf_user, sizeof(buf_user), len_user, value_user);
    char *pass = (char *)pbuf_get_contiguous(p, buf_pass, sizeof(buf_pass), len_pass, value_pass);
    if (user && pass) {
    user[len_user] = 0;
    pass[len_pass] = 0;
    if (!strcmp(user, AUTH_USER) && !strcmp(pass, AUTH_PASS)) {
    /* user and password are correct, create a "session" */
    start->verified=LOGIN_SUCCESSFUL;
    }
    }
    }
    }
    /* not returning ERR_OK aborts the connection, so return ERR_OK unless the
    connection is unknown */
    ret = ERR_OK;
    break;
    }
    }
    start = start->next;
    }
    /* this function must ALWAYS free the pbuf it is passed or it will leak memory */
    pbuf_free(p);
    return ret;
    }
    void
    httpd_post_finished(void *connection, char *response_uri, u16_t response_uri_len)
    {
    /* default page is "login failed" */
    snprintf(response_uri, response_uri_len, "/loginfail.html");
    conn_t *start = conn_header;
    conn_t *pre_conn = conn_header;
    while (start != NULL) {
    if (start->connection == connection) {
    if (start->verified == LOGIN_SUCCESSFUL) {
    /* login succeeded */
    snprintf(response_uri, response_uri_len, "/session.html");
    break;
    }
    if (start == conn_header) {
    conn_header = start->next;
    }
    else {
    pre_conn->next = start->next;
    }
    free(start);
    break;
    }
    pre_conn = start;
    start=start->next;
    break;
    }
    }
    /*===== post end =====*/
    /* ==== Session manager ===*/
    #define SET_COOKIE_SIZE 120
    char setCookie[SET_COOKIE_SIZE];
    char *httpd_set_cookies(const void *connection, const char *uri) {
    LWIP_UNUSED_ARG(uri);
    char cookie[SET_COOKIE_SIZE];
    conn_t *start = conn_header;
    conn_t *pre_conn = start;
    while(start!=NULL) {
    if (start->connection == connection && start->verified == LOGIN_SUCCESSFUL) {
    char sessid[26];
    gen_sess_cookie(sessid, cookie, SET_COOKIE_SIZE);
    new_session_id(sessid);
    snprintf(setCookie, SET_COOKIE_SIZE, HDR_HTTP_RES_SET_COOKIE "%s\r\n", cookie);
    if (start == conn_header) {
    conn_header = start->next;
    }
    else {
    pre_conn->next = start->next;
    }
    free(start);
    return setCookie;
    }
    pre_conn = start;
    start=start->next;
    }
    return NULL;
    }
    void httpd_received_cookies(const void *connection, char **cookies) {
    char *session = strstr(*cookies, "sess_id=");
    if (session) session+=8;
    sess_id_t* start = sess_header;
    while (start!= NULL) {
    if (strcmp(start->sess_id, session) == 0) {
    // match! reset timeout
    start->timeout = make_timeout_time_ms(SESSION_TIMEOUT_SEC*1000);
    return;
    break;
    } else {
    start = start->next;
    }
    }
    // not an avail connection cookie, reset cookie to NULL(invalid cookie)
    *cookies=NULL;
    }
    void httpd_redirect_uri(char* uri, u16_t *uri_len) {
    char new_uri[(*uri_len)+1];
    strncpy(new_uri, uri, *uri_len);
    new_uri[*uri_len]=0;
    for (int idx=0; idx < LWIP_ARRAYSIZE(session_check_uri); idx++) {
    if(strstr(new_uri, session_check_uri[idx])){
    // redirect to authenticate web page
    strcpy(uri, "/login.html");
    *uri_len = strlen("/login.html");
    break;
    }
    }
    return ;
    }
    repeating_timer_t timer;
    bool check_timeout_sessions_isr(repeating_timer_t *t) {
    sess_id_t *start = sess_header;
    sess_id_t *pre_sess = start;
    while (start != NULL) {
    if (absolute_time_diff_us(get_absolute_time(), start->timeout) < 0) {
    if (start == sess_header) {
    sess_header = NULL;
    free(start);
    break;
    } else {
    pre_sess->next = start->next;
    free(start);
    start=pre_sess->next;
    }
    } else {
    pre_sess = start;
    start = start->next;
    }
    }
    return true;
    }
    void check_timeout_sessions_timer() {
    add_repeating_timer_ms(1000*60, check_timeout_sessions_isr, NULL, &timer);
    }
    /* ==== Session manager end ===*/
  • picow_https_fs.c

  • /**
    * @file
    * HTTPD custom file system example
    *
    * This file demonstrates how to add support for an external file system to httpd.
    * It provides access to the specified root directory and uses stdio.h file functions
    * to read files.
    *
    * ATTENTION: This implementation is *not* secure: no checks are added to ensure
    * files are only read below the specified root directory!
    */
    /*
    * Copyright (c) 2017 Simon Goldschmidt
    * All rights reserved.
    *
    * Redistribution and use in source and binary forms, with or without modification,
    * are permitted provided that the following conditions are met:
    *
    * 1. Redistributions of source code must retain the above copyright notice,
    * this list of conditions and the following disclaimer.
    * 2. Redistributions in binary form must reproduce the above copyright notice,
    * this list of conditions and the following disclaimer in the documentation
    * and/or other materials provided with the distribution.
    * 3. The name of the author may not be used to endorse or promote products
    * derived from this software without specific prior written permission.
    *
    * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
    * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
    * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
    * SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
    * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
    * OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
    * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
    * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
    * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
    * OF SUCH DAMAGE.
    *
    * This file is part of the lwIP TCP/IP stack.
    *
    * Author: Simon Goldschmidt <goldsimon@gmx.de>
    *
    */
    #include "lwip/opt.h"
    #include "lwip/apps/fs.h"
    #include "lwip/def.h"
    #include "lwip/mem.h"
    #include "stdio.h"
    #include "string.h"
    #include "ff.h"
    #include "pico_storage.h"
    /////////////////////////////////////////////////////////
    #if !LWIP_HTTPD_CUSTOM_FILES
    #error This needs LWIP_HTTPD_CUSTOM_FILES
    #endif
    //#if !LWIP_HTTPD_DYNAMIC_HEADERS
    //#error This needs LWIP_HTTPD_DYNAMIC_HEADERS
    //#endif
    //#if !LWIP_HTTPD_DYNAMIC_FILE_READ
    //#error This wants to demonstrate read-after-open, so LWIP_HTTPD_DYNAMIC_FILE_READ is required!
    //#endif
    //#if !LWIP_HTTPD_FS_ASYNC_READ
    //#error This needs LWIP_HTTPD_FS_ASYNC_READ
    //#endif
    #if !LWIP_HTTPD_FILE_EXTENSION
    #error This needs LWIP_HTTPD_FILE_EXTENSION
    #endif
    struct fs_custom_data {
    //FILE *f;
    FIL *f;
    };
    const char* fs_ex_root_dir;
    void
    fs_ex_init(const char *httpd_root_dir)
    {
    fs_ex_root_dir = strdup(httpd_root_dir);
    }
    #if LWIP_HTTPD_CUSTOM_FILES
    int
    fs_open_custom(struct fs_file *file, const char *name)
    {
    char full_filename[256];
    //FILE *f;
    FIL f;
    FRESULT res;
    snprintf(full_filename, 255, "%s%s", fs_ex_root_dir, name);
    full_filename[255] = 0;
    //f = fopen(full_filename, "rb");
    res =f_open(&f, full_filename, FA_READ);
    //if (f != NULL) {
    if (res == FR_OK) {
    //if (!fseek(f, 0, SEEK_END)) {
    if (f_lseek(&f, 0) == FR_OK) {
    //int len = (int)ftell(f);
    int len = (int)f_size(&f);
    //if(!fseek(f, 0, SEEK_SET)) {
    struct fs_custom_data *data = (struct fs_custom_data *)mem_malloc(sizeof(struct fs_custom_data));
    LWIP_ASSERT("out of memory?", data != NULL);
    memset(file, 0, sizeof(struct fs_file));
    file->len = len;
    file->flags = FS_FILE_FLAGS_HEADER_PERSISTENT;
    data->f = &f;
    file->pextension = data;
    return 1;
    //}
    } else {
    printf("open file error:%s\n", name);
    }
    //fclose(f);
    f_close(&f);
    }
    return 0;
    }
    void
    fs_close_custom(struct fs_file *file)
    {
    //FRESULT res;
    if (file && file->pextension) {
    struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
    if (data->f != NULL) {
    //fclose(data->f);
    f_close(data->f);
    data->f = NULL;
    }
    mem_free(data);
    }
    }
    #if LWIP_HTTPD_FS_ASYNC_READ
    u8_t
    fs_canread_custom(struct fs_file *file)
    {
    /* This function is only necessary for asynchronous I/O:
    If reading would block, return 0 and implement fs_wait_read_custom() to call the
    supplied callback if reading works. */
    LWIP_UNUSED_ARG(file);
    return 1;
    }
    u8_t
    fs_wait_read_custom(struct fs_file *file, fs_wait_cb callback_fn, void *callback_arg)
    {
    //LWIP_ASSERT("not implemented in this example configuration", 0);
    LWIP_UNUSED_ARG(file);
    LWIP_UNUSED_ARG(callback_fn);
    LWIP_UNUSED_ARG(callback_arg);
    /* Return
    - 0 if ready to read (at least one byte)
    - 1 if reading should be delayed (call 'tcpip_callback(callback_fn, callback_arg)' when ready) */
    return 1;
    }
    int
    fs_read_async_custom(struct fs_file *file, char *buffer, int count, fs_wait_cb callback_fn, void *callback_arg)
    {
    struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
    //FILE *f;
    FIL *f;
    int len;
    int read_count = count;
    LWIP_ASSERT("data not set", data != NULL);
    f = data->f;
    //len = (int)fread(buffer, 1, read_count, f);
    f_read(f, buffer,read_count, &len);
    LWIP_UNUSED_ARG(callback_fn);
    LWIP_UNUSED_ARG(callback_arg);
    file->index += len;
    /* Return
    - FS_READ_EOF if all bytes have been read
    - FS_READ_DELAYED if reading is delayed (call 'tcpip_callback(callback_fn, callback_arg)' when done) */
    if (len == 0) {
    /* all bytes read already */
    return FS_READ_EOF;
    }
    return len;
    }
    #else /* LWIP_HTTPD_FS_ASYNC_READ */
    int
    fs_read_custom(struct fs_file *file, char *buffer, int count)
    {
    struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
    //FILE *f;
    FIL *f;
    int len;
    int read_count = count;
    LWIP_ASSERT("data not set", data != NULL);
    f = data->f;
    //len = (int)fread(buffer, 1, read_count, f);
    f_read(f, buffer, read_count, &len);
    file->index += len;
    /* Return FS_READ_EOF if all bytes have been read */
    return len;
    }
    #endif /* LWIP_HTTPD_FS_ASYNC_READ */
    #endif /* LWIP_HTTPD_CUSTOM_FILES */
  • CMakeLists.txt

  • # Generated Cmake Pico project file
    cmake_minimum_required(VERSION 3.13)
    set(CMAKE_C_STANDARD 11)
    set(CMAKE_CXX_STANDARD 17)
    # Initialise pico_sdk from installed location
    # (note this can come from environment, CMake cache etc)
    set(PICO_SDK_PATH "/home/duser/pico/pico-sdk")
    set(PICO_BOARD pico_w CACHE STRING "Board type")
    # Pull in Raspberry Pi Pico SDK (must be before project)
    include(pico_sdk_import.cmake)
    if (PICO_SDK_VERSION_STRING VERSION_LESS "1.4.0")
    message(FATAL_ERROR "Raspberry Pi Pico SDK version 1.4.0 (or later) required. Your version is ${PICO_SDK_VERSION_STRING}")
    endif()
    project(picow_https_auth C CXX ASM)
    # Initialise the Raspberry Pi Pico SDK
    pico_sdk_init()
    # Add executable. Default name is the project name, version 0.1
    add_executable(picow_https_auth
    picow_https_auth.c
    picow_https_ssi.c
    picow_httpd_fs.c
    )
    pico_set_program_name(picow_https_auth "picow_https_auth")
    pico_set_program_version(picow_https_auth "0.1")
    pico_enable_stdio_uart(picow_https_auth 1)
    pico_enable_stdio_usb(picow_https_auth 0)
    # Add the standard library to the build
    target_link_libraries(picow_https_auth
    pico_stdlib
    hardware_adc)
    # Add the standard include files to the build
    target_include_directories(picow_https_auth PRIVATE
    ${CMAKE_CURRENT_LIST_DIR}
    ${CMAKE_CURRENT_LIST_DIR}/.. # for our common lwipopts or any other standard includes, if required
    )
    # Add any user requested libraries
    target_link_libraries(picow_https_auth
    pico_cyw43_arch_lwip_poll
    pico_lwip_sntp
    pico_lwip_http
    pico_mbedtls
    pico_lwip_mbedtls
    )
    add_subdirectory(pico_storage_drv)
    target_link_libraries(picow_https_auth
    pico_storage_drv
    )
    pico_add_extra_outputs(picow_https_auth)
    # Ignore warnings from lwip code
    set_source_files_properties(
    ${PICO_LWIP_PATH}/src/apps/altcp_tls/altcp_tls_mbedtls.c
    PROPERTIES
    COMPILE_OPTIONS "-Wno-unused-result"
    )
    add_definitions(
    -DSNTP_SERVER_DNS=1
    -DSNTP_SERVER_ADDRESS="pool.ntp.org"
    -DSNTP_SET_SYSTEM_TIME=sntp_set_system_time
    -DSNTP_STARTUP_DELAY=0
    )
  • lwipopts.h

  • #ifndef __LWIPOPTS_H__
    #define __LWIPOPTS_H__
    // Common settings used in most of the pico_w examples
    // (see https://www.nongnu.org/lwip/2_1_x/group__lwip__opts.html for details)
    // allow override in some examples
    #ifndef NO_SYS
    #define NO_SYS 1
    #endif
    // allow override in some examples
    #ifndef LWIP_SOCKET
    #define LWIP_SOCKET 0
    #endif
    #if PICO_CYW43_ARCH_POLL
    #define MEM_LIBC_MALLOC 1
    #else
    // MEM_LIBC_MALLOC is incompatible with non polling versions
    #define MEM_LIBC_MALLOC 0
    #endif
    #define MEM_ALIGNMENT 4
    #define MEM_SIZE 4000
    #define MEMP_NUM_TCP_SEG 32
    #define MEMP_NUM_ARP_QUEUE 10
    #define PBUF_POOL_SIZE 24
    #define LWIP_ARP 1
    #define LWIP_ETHERNET 1
    #define LWIP_ICMP 1
    #define LWIP_RAW 1
    #define TCP_WND (8 * TCP_MSS)
    #define TCP_MSS 1460
    #define TCP_SND_BUF (8 * TCP_MSS)
    #define TCP_SND_QUEUELEN ((4 * (TCP_SND_BUF) + (TCP_MSS - 1)) / (TCP_MSS))
    #define LWIP_NETIF_STATUS_CALLBACK 1
    #define LWIP_NETIF_LINK_CALLBACK 1
    #define LWIP_NETIF_HOSTNAME 1
    #define LWIP_NETCONN 0
    #define MEM_STATS 0
    #define SYS_STATS 0
    #define MEMP_STATS 0
    #define LINK_STATS 0
    // #define ETH_PAD_SIZE 2
    #define LWIP_CHKSUM_ALGORITHM 3
    #define LWIP_DHCP 1
    #define LWIP_IPV4 1
    #define LWIP_TCP 1
    #define LWIP_UDP 1
    #define LWIP_DNS 1
    #define LWIP_TCP_KEEPALIVE 1
    #define LWIP_NETIF_TX_SINGLE_PBUF 1
    #define DHCP_DOES_ARP_CHECK 0
    #define LWIP_DHCP_DOES_ACD_CHECK 0
    #ifndef NDEBUG
    #define LWIP_DEBUG 1
    #define LWIP_STATS 1
    #define LWIP_STATS_DISPLAY 1
    #endif
    #define ETHARP_DEBUG LWIP_DBG_OFF
    #define NETIF_DEBUG LWIP_DBG_OFF
    #define PBUF_DEBUG LWIP_DBG_OFF
    #define API_LIB_DEBUG LWIP_DBG_OFF
    #define API_MSG_DEBUG LWIP_DBG_OFF
    #define SOCKETS_DEBUG LWIP_DBG_OFF
    #define ICMP_DEBUG LWIP_DBG_OFF
    #define INET_DEBUG LWIP_DBG_OFF
    #define IP_DEBUG LWIP_DBG_OFF
    #define IP_REASS_DEBUG LWIP_DBG_OFF
    #define RAW_DEBUG LWIP_DBG_OFF
    #define MEM_DEBUG LWIP_DBG_OFF
    #define MEMP_DEBUG LWIP_DBG_OFF
    #define SYS_DEBUG LWIP_DBG_OFF
    #define TCP_DEBUG LWIP_DBG_OFF
    #define TCP_INPUT_DEBUG LWIP_DBG_OFF
    #define TCP_OUTPUT_DEBUG LWIP_DBG_OFF
    #define TCP_RTO_DEBUG LWIP_DBG_OFF
    #define TCP_CWND_DEBUG LWIP_DBG_OFF
    #define TCP_WND_DEBUG LWIP_DBG_OFF
    #define TCP_FR_DEBUG LWIP_DBG_OFF
    #define TCP_QLEN_DEBUG LWIP_DBG_OFF
    #define TCP_RST_DEBUG LWIP_DBG_OFF
    #define UDP_DEBUG LWIP_DBG_OFF
    #define TCPIP_DEBUG LWIP_DBG_OFF
    #define PPP_DEBUG LWIP_DBG_OFF
    #define SLIP_DEBUG LWIP_DBG_OFF
    #define DHCP_DEBUG LWIP_DBG_OFF
    /* in Pico-SDK above 1.5.0 */
    #define MEMP_NUM_SYS_TIMEOUT (LWIP_NUM_SYS_TIMEOUT_INTERNAL+1)
    /* TCP WND must be at least 16 kb to match TLS record size
    or you will get a warning "altcp_tls: TCP_WND is smaller than the RX decrypion buffer, connection RX might stall!" */
    // tls
    #undef TCP_WND
    #define TCP_WND 16384
    #define LWIP_ALTCP 1
    #define LWIP_ALTCP_TLS 1
    #define LWIP_ALTCP_TLS_MBEDTLS 1
    //#define LWIP_DEBUG 1
    #define ALTCP_MBEDTLS_DEBUG LWIP_DBG_ON
    /* set authmode */
    #define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_NONE
    //#define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_REQUIRED
    // httpd
    #define HTTPD_ENABLE_HTTPS 1 // enable https
    #define LWIP_HTTPD 1
    #define LWIP_HTTPD_SSI 1
    #define LWIP_HTTPD_CGI 1
    #define LWIP_HTTPD_SSI_MULTIPART 1
    #define LWIP_HTTPD_SUPPORT_POST 1
    #define LWIP_HTTPD_SSI_INCLUDE_TAG 0
    // extern wbe page files
    #define LWIP_HTTPD_CUSTOM_FILES 1
    #define LWIP_HTTPD_DYNAMIC_HEADERS 1
    #define LWIP_HTTPD_FS_ASYNC_READ 1
    #define LWIP_HTTPD_DYNAMIC_FILE_READ 1
    #define LWIP_HTTPD_FILE_EXTENSION 1
    #define LWIP_HTTPD_SUPPORT_COOKIES 1
    #endif /* __LWIPOPTS_H__ */
  • mbedtls_config.h

  • /* Workaround for some mbedtls source files using INT_MAX without including limits.h */
    #include <limits.h>
    #define MBEDTLS_NO_PLATFORM_ENTROPY
    #define MBEDTLS_ENTROPY_HARDWARE_ALT
    #define MBEDTLS_SSL_OUT_CONTENT_LEN 2048
    #define MBEDTLS_ALLOW_PRIVATE_ACCESS
    #define MBEDTLS_HAVE_TIME
    #define MBEDTLS_HAVE_TIME_DATE
    #define MBEDTLS_CIPHER_MODE_CBC
    #define MBEDTLS_ECP_DP_SECP192R1_ENABLED
    #define MBEDTLS_ECP_DP_SECP224R1_ENABLED
    #define MBEDTLS_ECP_DP_SECP256R1_ENABLED
    #define MBEDTLS_ECP_DP_SECP384R1_ENABLED
    #define MBEDTLS_ECP_DP_SECP521R1_ENABLED
    #define MBEDTLS_ECP_DP_SECP192K1_ENABLED
    #define MBEDTLS_ECP_DP_SECP224K1_ENABLED
    #define MBEDTLS_ECP_DP_SECP256K1_ENABLED
    #define MBEDTLS_ECP_DP_BP256R1_ENABLED
    #define MBEDTLS_ECP_DP_BP384R1_ENABLED
    #define MBEDTLS_ECP_DP_BP512R1_ENABLED
    #define MBEDTLS_ECP_DP_CURVE25519_ENABLED
    #define MBEDTLS_KEY_EXCHANGE_RSA_ENABLED
    #define MBEDTLS_PKCS1_V15
    #define MBEDTLS_SHA256_SMALLER
    #define MBEDTLS_SSL_SERVER_NAME_INDICATION
    #define MBEDTLS_AES_C
    #define MBEDTLS_ASN1_PARSE_C
    #define MBEDTLS_BIGNUM_C
    #define MBEDTLS_CIPHER_C
    #define MBEDTLS_CTR_DRBG_C
    #define MBEDTLS_ENTROPY_C
    #define MBEDTLS_ERROR_C
    #define MBEDTLS_MD_C
    #define MBEDTLS_MD5_C
    #define MBEDTLS_OID_C
    #define MBEDTLS_PKCS5_C
    #define MBEDTLS_PK_C
    #define MBEDTLS_PK_PARSE_C
    #define MBEDTLS_PLATFORM_C
    #define MBEDTLS_RSA_C
    #define MBEDTLS_SHA1_C
    #define MBEDTLS_SHA224_C
    #define MBEDTLS_SHA256_C
    #define MBEDTLS_SHA512_C
    #define MBEDTLS_SSL_CLI_C
    #define MBEDTLS_SSL_SRV_C
    #define MBEDTLS_SSL_TLS_C
    #define MBEDTLS_X509_CRT_PARSE_C
    #define MBEDTLS_X509_USE_C
    #define MBEDTLS_AES_FEWER_TABLES
    /* TLS 1.2 */
    #define MBEDTLS_SSL_PROTO_TLS1_2
    #define MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED
    #define MBEDTLS_GCM_C
    #define MBEDTLS_ECDH_C
    #define MBEDTLS_ECP_C
    #define MBEDTLS_ECDSA_C
    #define MBEDTLS_ASN1_WRITE_C
    // The following is needed to parse a certificate
    #define MBEDTLS_PEM_PARSE_C
    #define MBEDTLS_BASE64_C
    // for large file size
    #define MBEDTLS_SSL_MAX_FRAGMENT_LENGTH

  • index.shtml

  • <html lang="en">
    <head><title>Raspberry Pi Pico W - HTTPS Session</title></head>
    <meta http-equiv="content-type" content="text/html; charset=utf-8">
    <Script>
    function logout() {
    const cookies = document.cookie.split(";");
    for (let i = 0; i < cookies.length; i++) {
    const cookie = cookies[i];
    const eqPos = cookie.indexOf("=");
    const name = eqPos > -1 ? cookie.substr(0, eqPos) : cookie;
    if (name=="sess_id"){
    document.cookie = name + "=;expires=Thu, 01 Jan 1970 00:00:00 GMT";
    }
    }
    }
    </Script>
    <body bgcolor="white" text="black">
    <h1><strong>Raspberry Pi Pico W - </strong>HTTPS Session</h1>
    <table style="border-collapse: collapse; width: 1000; height: 127px;" border="1">
    <tbody>
    <tr style="height: 18px;">
    <td style="width: 36.5145%; height: 18px;">
    <p><a title="System Info" href="index.shtml">Sysem Info</a></p>
    <p><button onClick="location.href='login.html'">Login</button>
    <p>Login Require:</p>
    <ul>
    <li><a title="Light Control" href="light.shtml">Light control</a></li>
    <li><a title="Demo Page 1" href="demo1.shtml">Demo page 1</a></li>
    <li><a title="Demo Page 2" href="demo2.shtml">Demo Page2</a></li>
    </ul>
    <p>Login not required:</p>
    <ul>
    <li><a title="Demo Page 3" href="demo3.html">Demo page 3</a></li>
    <li><a title="Demo page 4" href="demo4.html">Demo Page 4</a></li>
    </ul>
    <!--#logout-->
    </td>
    <td style="width: 63.4855%; height: 18px;">
    <p style="text-align: center;"><span style="color: #0000ff; font-size: 24px;"><strong>System information</strong></span></p>
    <div style="margin-left:20px;">
    <p>Onboard Temp.:<font color=red><!--#temp--></font></p>
    <p>Free RAM Size:<font color=red><!--#freeram--></font></p>
    <p>Free SD Size:<font color=red><!--#freesd--></font></p>
    </div>
    </td>
    </tr>
    </tbody>
    </table>
    </body></html>
  • login.html

  • <html lang="en">
    <head><title>Raspberry Pi Pico W - HTTPS Session</title></head>
    <meta http-equiv="content-type" content="text/html; charset=utf-8">
    <Script>
    function logout() {
    const cookies = document.cookie.split(";");
    for (let i = 0; i < cookies.length; i++) {
    const cookie = cookies[i];
    const eqPos = cookie.indexOf("=");
    const name = eqPos > -1 ? cookie.substr(0, eqPos) : cookie;
    if (name=="sess_id"){
    document.cookie = name + "=;expires=Thu, 01 Jan 1970 00:00:00 GMT";
    }
    document.location.href="index.shtml";
    }
    }
    </Script>
    <body bgcolor="white" text="black">
    <h1><strong>Raspberry Pi Pico W - </strong>HTTPS Session</h1>
    <table style="border-collapse: collapse; width: 1000; height: 127px;" border="1">
    <tbody>
    <tr style="height: 18px;">
    <td style="width: 400; height: 18px;">
    <p><a title="System Info" href="index.shtml">Sysem Info</a></p>
    <p><button onClick="location.href='login.html'">Login</button>
    <p>Login Require:</p>
    <ul>
    <li><a title="Light Control" href="light.shtml">Light control</a></li>
    <li><a title="Demo Page 1" href="demo1.shtml">Demo page 1</a></li>
    <li><a title="Demo Page 2" href="demo2.shtml">Demo Page2</a></li>
    </ul>
    <p>Login not required:</p>
    <ul>
    <li><a title="Demo Page 3" href="demo3.html">Demo page 3</a></li>
    <li><a title="Demo page 4" href="demo4.html">Demo Page 4</a></li>
    </ul>
    <button onClick='javascript:logout();'>Logout</button>
    </td>
    <td style="width: 600; height: 18px;">
    <h1>Login</h1>
    <form name="login" action="login.cgi" method="post">
    <div align="center">
    <label><b>Username:</b></label>
    <input type="text" placeholder="Enter Username" name="user" required><br><br>
    <label><b>Password:</b></label>
    <input type="password" placeholder="Enter Password" name="pass" required><br><br>
    <button type="submit">Login</button>
    </div>
    </form>
    </td>
    </tr>
    </table>
    </body>
    </html>

2023年12月7日 星期四

Raspberry Pi Pico W || LwIP || MbedTLS: EP.1 HTTPS server self-signed Certificate setup & web page files in SD

 本篇文章介紹使用Pico-SDK 結合LwIP and MbedTLS 在Raspberry Pi Pico W上建立HTTPS(Secure HTTP) server。本專案會使用LwIP 中SNTP and HTTP Application。詳細步驟如下說明:

一、連線WiFi並設定static IP:

cyw43_arch_enable_sta_mode();
netif_set_status_callback(netif_default, netif_status_cb);
cyw43_arch_wifi_connect_timeout_ms(WIFI_SSID, WIFI_PASSWORD, CYW43_AUTH_WPA2_AES_PSK, 30000); 

使用netif_set_status_callback當成功連上WiFi時呼叫netif_status_cb設定Static IP。

二、以LwIP SNTP application設定系統時間。
在CMakeLists.txt加入:

add_definitions(
        -DSNTP_SERVER_DNS=1
        -DSNTP_SERVER_ADDRESS="pool.ntp.org"
        -DSNTP_SET_SYSTEM_TIME=sntp_set_system_time
        -DSNTP_STARTUP_DELAY=0
)

sntp_set_system_time為SNTP client取得資料時呼叫的callback function。

在程式中呼叫

sntp_setoperatingmode(SNTP_OPMODE_POLL);
sntp_init();

以啟用SNTP client。

三、建立self-signed Server Side Certificates:

  1. openssl genrsa -des3 -out ca.key 2048:
    建立Root CA key。
  2. openssl req -new -x509 -days 3650 -key ca.key -out ca.cert:
    建立self-signed root Certificate。
  3. openssl genrsa -out server.key 2048:
    建立server side private key。
  4. openssl req -new -out server.csr -key server.key
    使用server key製作server憑證需求(certificate request)
  5. openssl x509 -req -in server.csr -CA ca.cert -CAkey ca.key -CAcreateserial -out server.cert -days 365
    使用root CA ca.cert簽署server side certificate。
四、設定MbedTLS的TLS connection參數:
  1. 將ca.cert, server.cert與server.key加入在專案picow_https_cert.h中。
  2. CMakeLists.txt加入:
    target_link_libraries(picow_https_server
            pico_cyw43_arch_lwip_poll
            pico_lwip_sntp
            pico_lwip_http
            pico_mbedtls
            pico_lwip_mbedtls
            )
  3. 在lwipopts.conf加入:
    /* in Pico-SDK above 1.5.0 */
    #define MEMP_NUM_SYS_TIMEOUT (LWIP_NUM_SYS_TIMEOUT_INTERNAL+1)
    /* TCP WND must be at least 16 kb to match TLS record size
    or you will get a warning "altcp_tls: TCP_WND is smaller than the RX decrypion buffer, connection RX might stall!" */
    // tls
    #undef TCP_WND
    #define TCP_WND 16384
    #define LWIP_ALTCP 1
    #define LWIP_ALTCP_TLS 1
    #define LWIP_ALTCP_TLS_MBEDTLS 1
    #define LWIP_DEBUG 1
    #define ALTCP_MBEDTLS_DEBUG LWIP_DBG_ON
    /* set authmode */
    #define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_NONE
    //#define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_REQUIRED

  4. 設定tls config:
    加入self-signed root CA certificate, server certificate and server private key
    tls_config = altcp_tls_create_config_server_privkey_cert(SERVER_KEY, sizeof(SERVER_KEY), NULL, 0, SERVER_CERT, sizeof(SERVER_CERT));
    mbedtls_x509_crt_parse(tls_config->cert, CA_CERT, sizeof(CA_CERT));

五、設定https server:

這個專案將web page files放在SD卡上,先mount SD在設定root path
有關pico SD storage driver請參閱前面文章
https server如何讀取page file,修改自LwIP的fs_example.c範例程式。
  1. mount SD:
    if (f_mount(&fs, SDMMC_PATH, 1) != FR_OK) {
          printf("mount error\n");
          return 0;
        }
  2. 設定web server root path:
        fs_ex_init(SDMMC_PATH"/");
  3. 啟用https:
    httpd_inits(tls_config);
使用broswer, curl與wget使令測試,過程如下列影片。

六、成果影片



七、程式碼

  • CMakeLists.txt

# Generated Cmake Pico project file
cmake_minimum_required(VERSION 3.13)
set(CMAKE_C_STANDARD 11)
set(CMAKE_CXX_STANDARD 17)
# Initialise pico_sdk from installed location
# (note this can come from environment, CMake cache etc)
set(PICO_SDK_PATH "/home/duser/pico/pico-sdk")
set(PICO_BOARD pico_w CACHE STRING "Board type")
# Pull in Raspberry Pi Pico SDK (must be before project)
include(pico_sdk_import.cmake)
if (PICO_SDK_VERSION_STRING VERSION_LESS "1.4.0")
message(FATAL_ERROR "Raspberry Pi Pico SDK version 1.4.0 (or later) required. Your version is ${PICO_SDK_VERSION_STRING}")
endif()
project(picow_https_server C CXX ASM)
# Initialise the Raspberry Pi Pico SDK
pico_sdk_init()
# Add executable. Default name is the project name, version 0.1
add_executable(picow_https_server
picow_https_server.c
fs_example.c)
add_definitions(
-DSNTP_SERVER_DNS=1
-DSNTP_SERVER_ADDRESS="pool.ntp.org"
-DSNTP_SET_SYSTEM_TIME=sntp_set_system_time
-DSNTP_STARTUP_DELAY=0
)
pico_set_program_name(picow_https_server "picow_https_server")
pico_set_program_version(picow_https_server "0.1")
pico_enable_stdio_uart(picow_https_server 1)
pico_enable_stdio_usb(picow_https_server 0)
# Add the standard library to the build
target_link_libraries(picow_https_server
pico_stdlib)
# Add the standard include files to the build
target_include_directories(picow_https_server PRIVATE
${CMAKE_CURRENT_LIST_DIR}
${CMAKE_CURRENT_LIST_DIR}/.. # for our common lwipopts or any other standard includes, if required
)
# Add any user requested libraries
target_link_libraries(picow_https_server
pico_cyw43_arch_lwip_poll
pico_lwip_sntp
pico_lwip_http
pico_mbedtls
pico_lwip_mbedtls
)
add_subdirectory(pico_storage_drv)
target_link_libraries(picow_https_server
pico_storage_drv
)
pico_add_extra_outputs(picow_https_server)
  • picow_https_server.c
#include < stdio.h >
#include "pico/stdlib.h"
#include "pico/cyw43_arch.h"
#include "lwip/altcp_tls.h"
#include "lwip/altcp_tcp.h"
#include "lwip/altcp.h"
#include "lwip/apps/httpd.h"
#include "lwip/apps/sntp.h"
#include "picow_https_cert.h"
#include "ff.h"
#include "pico_storage.h"
#define WIFI_SSID "your-ssid"
#define WIFI_PASSWORD "your-password"
struct altcp_tls_config {
mbedtls_ssl_config conf;
mbedtls_x509_crt *cert;
mbedtls_pk_context *pkey;
u8_t cert_count;
u8_t cert_max;
u8_t pkey_count;
u8_t pkey_max;
mbedtls_x509_crt *ca;
#if defined(MBEDTLS_SSL_CACHE_C) && ALTCP_MBEDTLS_USE_SESSION_CACHE
/** Inter-connection cache for fast connection startup */
struct mbedtls_ssl_cache_context cache;
#endif
#if defined(MBEDTLS_SSL_SESSION_TICKETS) && ALTCP_MBEDTLS_USE_SESSION_TICKETS
mbedtls_ssl_ticket_context ticket_ctx;
#endif
};
struct altcp_tls_config *tls_config = NULL;
void fs_ex_init(const char *httpd_root_dir);
void netif_status_cb(struct netif *netif) {
if (netif_is_link_up(netif)) {
ip_addr_t ipaddr;
ip_addr_t netmask;
ip_addr_t gateway;
ipaddr_aton("192.168.1.70", &ipaddr);
ipaddr_aton("255.255.255.0", &netmask);
ipaddr_aton("192.168.1.1", &gateway);
netif_set_addr(netif, &ipaddr, &netmask, &gateway);
}
}
void sntp_set_system_time(uint32_t sec, uint32_t us)
{
struct tm info;
time_t tim = sec;
struct timeval tmv;
tmv.tv_sec=sec;
tmv.tv_usec=us;
settimeofday(&tmv, 0);
printf("set time :%s\n", ctime(&tim));
sntp_stop();
}
int main()
{
stdio_init_all();
if (cyw43_arch_init()) {
printf("cyw43 arch init error\n");
return 0;
}
//1. connect to WiFi network and set static IP
cyw43_arch_enable_sta_mode();
netif_set_status_callback(netif_default, netif_status_cb);
int wifi_stat = cyw43_arch_wifi_connect_timeout_ms(WIFI_SSID, WIFI_PASSWORD, CYW43_AUTH_WPA2_AES_PSK, 30000);
if (wifi_stat) {
printf("wifi connect error:%d\n", wifi_stat);
return 0;
}
printf("Wifi connected\n");
printf("get ip addr:%s\n",ipaddr_ntoa(&(cyw43_state.netif[0].ip_addr)));
// 2. set SNTP
sntp_setoperatingmode(SNTP_OPMODE_POLL);
sntp_init();
// 3. set TLS connection
tls_config = altcp_tls_create_config_server_privkey_cert(SERVER_KEY, sizeof(SERVER_KEY), NULL, 0, SERVER_CERT, sizeof(SERVER_CERT));
mbedtls_x509_crt_parse(tls_config->cert, CA_CERT, sizeof(CA_CERT));
//4. mount FS in SD card and start https server
FATFS fs;
FIL fil;
FRESULT res;
if (f_mount(&fs, SDMMC_PATH, 1) != FR_OK) {
printf("mount error\n");
return 0;
}
fs_ex_init(SDMMC_PATH"/");
httpd_inits(tls_config);
time_t tm;
while(1) {
cyw43_arch_poll();
sleep_ms(50);
}
return 0;
}

  • fs_example.c
/**
* @file
* HTTPD custom file system example
*
* This file demonstrates how to add support for an external file system to httpd.
* It provides access to the specified root directory and uses stdio.h file functions
* to read files.
*
* ATTENTION: This implementation is *not* secure: no checks are added to ensure
* files are only read below the specified root directory!
*/
/*
* Copyright (c) 2017 Simon Goldschmidt
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT
* SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
* OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
* OF SUCH DAMAGE.
*
* This file is part of the lwIP TCP/IP stack.
*
* Author: Simon Goldschmidt < goldsimon@gmx.de >
*
*/
#include "lwip/opt.h"
#include "lwip/apps/fs.h"
#include "lwip/def.h"
#include "lwip/mem.h"
#include "stdio.h"
#include "string.h"
#include "ff.h"
#include "pico_storage.h"
/** define LWIP_HTTPD_EXAMPLE_CUSTOMFILES to 1 to enable this file system */
#ifndef LWIP_HTTPD_EXAMPLE_CUSTOMFILES
#define LWIP_HTTPD_EXAMPLE_CUSTOMFILES 1
#endif
/** define LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED to 1 to delay open and read
* as if e.g. reading from external SPI flash */
#ifndef LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
#define LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED 0
#endif
/** define LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ to the number of bytes
* to read to emulate limited transfer buffers and don't read whole files in
* one chunk.
* WARNING: lowering this slows down the connection!
*/
#ifndef LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ
#define LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ 0
#endif
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES
#if !LWIP_HTTPD_CUSTOM_FILES
#error This needs LWIP_HTTPD_CUSTOM_FILES
#endif
#if !LWIP_HTTPD_DYNAMIC_HEADERS
#error This needs LWIP_HTTPD_DYNAMIC_HEADERS
#endif
#if !LWIP_HTTPD_DYNAMIC_FILE_READ
#error This wants to demonstrate read-after-open, so LWIP_HTTPD_DYNAMIC_FILE_READ is required!
#endif
#if !LWIP_HTTPD_FS_ASYNC_READ
#error This needs LWIP_HTTPD_FS_ASYNC_READ
#endif
#if !LWIP_HTTPD_FILE_EXTENSION
#error This needs LWIP_HTTPD_FILE_EXTENSION
#endif
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
#include "lwip/tcpip.h"
#endif
struct fs_custom_data {
//FILE *f;
FIL *f;
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
int delay_read;
fs_wait_cb callback_fn;
void *callback_arg;
#endif
};
const char* fs_ex_root_dir;
void
fs_ex_init(const char *httpd_root_dir)
{
fs_ex_root_dir = strdup(httpd_root_dir);
}
#if LWIP_HTTPD_CUSTOM_FILES
int
fs_open_custom(struct fs_file *file, const char *name)
{
char full_filename[256];
//FILE *f;
FIL f;
FRESULT res;
snprintf(full_filename, 255, "%s%s", fs_ex_root_dir, name);
full_filename[255] = 0;
//f = fopen(full_filename, "rb");
res =f_open(&f, full_filename, FA_READ);
//if (f != NULL) {
if (res == FR_OK) {
//if (!fseek(f, 0, SEEK_END)) {
if (f_lseek(&f, 0) == FR_OK) {
//int len = (int)ftell(f);
int len = (int)f_size(&f);
//if(!fseek(f, 0, SEEK_SET)) {
struct fs_custom_data *data = (struct fs_custom_data *)mem_malloc(sizeof(struct fs_custom_data));
LWIP_ASSERT("out of memory?", data != NULL);
memset(file, 0, sizeof(struct fs_file));
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
file->len = 0; /* read size delayed */
data->delay_read = 3;
LWIP_UNUSED_ARG(len);
#else
file->len = len;
#endif
file->flags = FS_FILE_FLAGS_HEADER_PERSISTENT;
data->f = &f;
file->pextension = data;
return 1;
//}
}
//fclose(f);
f_close(&f);
}
return 0;
}
void
fs_close_custom(struct fs_file *file)
{
if (file && file->pextension) {
struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
if (data->f != NULL) {
//fclose(data->f);
f_close(data->f);
data->f = NULL;
}
mem_free(data);
}
}
#if LWIP_HTTPD_FS_ASYNC_READ
u8_t
fs_canread_custom(struct fs_file *file)
{
/* This function is only necessary for asynchronous I/O:
If reading would block, return 0 and implement fs_wait_read_custom() to call the
supplied callback if reading works. */
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
struct fs_custom_data *data;
LWIP_ASSERT("file != NULL", file != NULL);
data = (struct fs_custom_data *)file->pextension;
if (data == NULL) {
/* file transfer has been completed already */
LWIP_ASSERT("transfer complete", file->index == file->len);
return 1;
}
LWIP_ASSERT("data != NULL", data != NULL);
/* This just simulates a simple delay. This delay would normally come e.g. from SPI transfer */
if (data->delay_read == 3) {
/* delayed file size mode */
data->delay_read = 1;
LWIP_ASSERT("", file->len == 0);
if (!fseek(data->f, 0, SEEK_END)) {
int len = (int)ftell(data->f);
if(!fseek(data->f, 0, SEEK_SET)) {
file->len = len; /* read size delayed */
data->delay_read = 1;
return 0;
}
}
/* if we come here, something is wrong with the file */
LWIP_ASSERT("file error", 0);
}
if (data->delay_read == 1) {
/* tell read function to delay further */
}
#endif
LWIP_UNUSED_ARG(file);
return 1;
}
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
static void
fs_example_read_cb(void *arg)
{
struct fs_custom_data *data = (struct fs_custom_data *)arg;
fs_wait_cb callback_fn = data->callback_fn;
void *callback_arg = data->callback_arg;
data->callback_fn = NULL;
data->callback_arg = NULL;
LWIP_ASSERT("no callback_fn", callback_fn != NULL);
callback_fn(callback_arg);
}
#endif
u8_t
fs_wait_read_custom(struct fs_file *file, fs_wait_cb callback_fn, void *callback_arg)
{
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
err_t err;
struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
LWIP_ASSERT("data not set", data != NULL);
data->callback_fn = callback_fn;
data->callback_arg = callback_arg;
err = tcpip_try_callback(fs_example_read_cb, data);
LWIP_ASSERT("out of queue elements?", err == ERR_OK);
LWIP_UNUSED_ARG(err);
#else
LWIP_ASSERT("not implemented in this example configuration", 0);
#endif
LWIP_UNUSED_ARG(file);
LWIP_UNUSED_ARG(callback_fn);
LWIP_UNUSED_ARG(callback_arg);
/* Return
- 0 if ready to read (at least one byte)
- 1 if reading should be delayed (call 'tcpip_callback(callback_fn, callback_arg)' when ready) */
return 1;
}
int
fs_read_async_custom(struct fs_file *file, char *buffer, int count, fs_wait_cb callback_fn, void *callback_arg)
{
struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
//FILE *f;
FIL *f;
int len;
int read_count = count;
LWIP_ASSERT("data not set", data != NULL);
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_DELAYED
/* This just simulates a delay. This delay would normally come e.g. from SPI transfer */
LWIP_ASSERT("invalid state", data->delay_read >= 0 && data->delay_read <= 2);
if (data->delay_read == 2) {
/* no delay next time */
data->delay_read = 0;
return FS_READ_DELAYED;
} else if (data->delay_read == 1) {
err_t err;
/* execute requested delay */
data->delay_read = 2;
LWIP_ASSERT("duplicate callback request", data->callback_fn == NULL);
data->callback_fn = callback_fn;
data->callback_arg = callback_arg;
err = tcpip_try_callback(fs_example_read_cb, data);
LWIP_ASSERT("out of queue elements?", err == ERR_OK);
LWIP_UNUSED_ARG(err);
return FS_READ_DELAYED;
}
/* execute this read but delay the next one */
data->delay_read = 1;
#endif
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ
read_count = LWIP_MIN(read_count, LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ);
#endif
f = data->f;
//len = (int)fread(buffer, 1, read_count, f);
f_read(f, buffer,read_count, &len);
LWIP_UNUSED_ARG(callback_fn);
LWIP_UNUSED_ARG(callback_arg);
file->index += len;
/* Return
- FS_READ_EOF if all bytes have been read
- FS_READ_DELAYED if reading is delayed (call 'tcpip_callback(callback_fn, callback_arg)' when done) */
if (len == 0) {
/* all bytes read already */
return FS_READ_EOF;
}
return len;
}
#else /* LWIP_HTTPD_FS_ASYNC_READ */
int
fs_read_custom(struct fs_file *file, char *buffer, int count)
{
struct fs_custom_data *data = (struct fs_custom_data *)file->pextension;
FILE *f;
int len;
int read_count = count;
LWIP_ASSERT("data not set", data != NULL);
#if LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ
read_count = LWIP_MIN(read_count, LWIP_HTTPD_EXAMPLE_CUSTOMFILES_LIMIT_READ);
#endif
f = data->f;
len = (int)fread(buffer, 1, read_count, f);
file->index += len;
/* Return FS_READ_EOF if all bytes have been read */
return len;
}
#endif /* LWIP_HTTPD_FS_ASYNC_READ */
#endif /* LWIP_HTTPD_CUSTOM_FILES */
#endif /* LWIP_HTTPD_EXAMPLE_CUSTOMFILES */

  • lwipopts.h
#ifndef __LWIPOPTS_H__
#define __LWIPOPTS_H__
// Common settings used in most of the pico_w examples
// (see https://www.nongnu.org/lwip/2_1_x/group__lwip__opts.html for details)
// allow override in some examples
#ifndef NO_SYS
#define NO_SYS 1
#endif
// allow override in some examples
#ifndef LWIP_SOCKET
#define LWIP_SOCKET 0
#endif
#if PICO_CYW43_ARCH_POLL
#define MEM_LIBC_MALLOC 1
#else
// MEM_LIBC_MALLOC is incompatible with non polling versions
#define MEM_LIBC_MALLOC 0
#endif
#define MEM_ALIGNMENT 4
#define MEM_SIZE 4000
#define MEMP_NUM_TCP_SEG 32
#define MEMP_NUM_ARP_QUEUE 10
#define PBUF_POOL_SIZE 24
#define LWIP_ARP 1
#define LWIP_ETHERNET 1
#define LWIP_ICMP 1
#define LWIP_RAW 1
#define TCP_WND (8 * TCP_MSS)
#define TCP_MSS 1460
#define TCP_SND_BUF (8 * TCP_MSS)
#define TCP_SND_QUEUELEN ((4 * (TCP_SND_BUF) + (TCP_MSS - 1)) / (TCP_MSS))
#define LWIP_NETIF_STATUS_CALLBACK 1
#define LWIP_NETIF_LINK_CALLBACK 1
#define LWIP_NETIF_HOSTNAME 1
#define LWIP_NETCONN 0
#define MEM_STATS 0
#define SYS_STATS 0
#define MEMP_STATS 0
#define LINK_STATS 0
// #define ETH_PAD_SIZE 2
#define LWIP_CHKSUM_ALGORITHM 3
#define LWIP_DHCP 1
#define LWIP_IPV4 1
#define LWIP_TCP 1
#define LWIP_UDP 1
#define LWIP_DNS 1
#define LWIP_TCP_KEEPALIVE 1
#define LWIP_NETIF_TX_SINGLE_PBUF 1
#define DHCP_DOES_ARP_CHECK 0
#define LWIP_DHCP_DOES_ACD_CHECK 0
#ifndef NDEBUG
#define LWIP_DEBUG 1
#define LWIP_STATS 1
#define LWIP_STATS_DISPLAY 1
#endif
#define ETHARP_DEBUG LWIP_DBG_OFF
#define NETIF_DEBUG LWIP_DBG_OFF
#define PBUF_DEBUG LWIP_DBG_OFF
#define API_LIB_DEBUG LWIP_DBG_OFF
#define API_MSG_DEBUG LWIP_DBG_OFF
#define SOCKETS_DEBUG LWIP_DBG_OFF
#define ICMP_DEBUG LWIP_DBG_OFF
#define INET_DEBUG LWIP_DBG_OFF
#define IP_DEBUG LWIP_DBG_OFF
#define IP_REASS_DEBUG LWIP_DBG_OFF
#define RAW_DEBUG LWIP_DBG_OFF
#define MEM_DEBUG LWIP_DBG_OFF
#define MEMP_DEBUG LWIP_DBG_OFF
#define SYS_DEBUG LWIP_DBG_OFF
#define TCP_DEBUG LWIP_DBG_OFF
#define TCP_INPUT_DEBUG LWIP_DBG_OFF
#define TCP_OUTPUT_DEBUG LWIP_DBG_OFF
#define TCP_RTO_DEBUG LWIP_DBG_OFF
#define TCP_CWND_DEBUG LWIP_DBG_OFF
#define TCP_WND_DEBUG LWIP_DBG_OFF
#define TCP_FR_DEBUG LWIP_DBG_OFF
#define TCP_QLEN_DEBUG LWIP_DBG_OFF
#define TCP_RST_DEBUG LWIP_DBG_OFF
#define UDP_DEBUG LWIP_DBG_OFF
#define TCPIP_DEBUG LWIP_DBG_OFF
#define PPP_DEBUG LWIP_DBG_OFF
#define SLIP_DEBUG LWIP_DBG_OFF
#define DHCP_DEBUG LWIP_DBG_OFF
/* in Pico-SDK above 1.5.0 */
#define MEMP_NUM_SYS_TIMEOUT (LWIP_NUM_SYS_TIMEOUT_INTERNAL+1)
/* TCP WND must be at least 16 kb to match TLS record size
or you will get a warning "altcp_tls: TCP_WND is smaller than the RX decrypion buffer, connection RX might stall!" */
// tls
#undef TCP_WND
#define TCP_WND 16384
#define LWIP_ALTCP 1
#define LWIP_ALTCP_TLS 1
#define LWIP_ALTCP_TLS_MBEDTLS 1
#define LWIP_DEBUG 1
#define ALTCP_MBEDTLS_DEBUG LWIP_DBG_ON
/* set authmode */
#define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_NONE
//#define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_REQUIRED
// httpd
#define HTTPD_ENABLE_HTTPS 1 // enable https
#define LWIP_HTTPD 1
//#define LWIP_HTTPD_SSI 1
//#define LWIP_HTTPD_CGI 1
//#define LWIP_HTTPD_SSI_MULTIPART 1
//#define LWIP_HTTPD_SUPPORT_POST 1
//#define LWIP_HTTPD_SSI_INCLUDE_TAG 0
// extern wbe page files
#define LWIP_HTTPD_CUSTOM_FILES 1
#define LWIP_HTTPD_DYNAMIC_HEADERS 1
#define LWIP_HTTPD_FS_ASYNC_READ 1
#define LWIP_HTTPD_DYNAMIC_FILE_READ 1
#define LWIP_HTTPD_FILE_EXTENSION 1
#endif /* __LWIPOPTS_H__ */

  • mbedtls_config.h

/* Workaround for some mbedtls source files using INT_MAX without including limits.h */
#include <limits.h>
#define MBEDTLS_NO_PLATFORM_ENTROPY
#define MBEDTLS_ENTROPY_HARDWARE_ALT
#define MBEDTLS_SSL_OUT_CONTENT_LEN 2048
#define MBEDTLS_ALLOW_PRIVATE_ACCESS
#define MBEDTLS_HAVE_TIME
#define MBEDTLS_HAVE_TIME_DATE
#define MBEDTLS_CIPHER_MODE_CBC
#define MBEDTLS_ECP_DP_SECP192R1_ENABLED
#define MBEDTLS_ECP_DP_SECP224R1_ENABLED
#define MBEDTLS_ECP_DP_SECP256R1_ENABLED
#define MBEDTLS_ECP_DP_SECP384R1_ENABLED
#define MBEDTLS_ECP_DP_SECP521R1_ENABLED
#define MBEDTLS_ECP_DP_SECP192K1_ENABLED
#define MBEDTLS_ECP_DP_SECP224K1_ENABLED
#define MBEDTLS_ECP_DP_SECP256K1_ENABLED
#define MBEDTLS_ECP_DP_BP256R1_ENABLED
#define MBEDTLS_ECP_DP_BP384R1_ENABLED
#define MBEDTLS_ECP_DP_BP512R1_ENABLED
#define MBEDTLS_ECP_DP_CURVE25519_ENABLED
#define MBEDTLS_KEY_EXCHANGE_RSA_ENABLED
#define MBEDTLS_PKCS1_V15
#define MBEDTLS_SHA256_SMALLER
#define MBEDTLS_SSL_SERVER_NAME_INDICATION
#define MBEDTLS_AES_C
#define MBEDTLS_ASN1_PARSE_C
#define MBEDTLS_BIGNUM_C
#define MBEDTLS_CIPHER_C
#define MBEDTLS_CTR_DRBG_C
#define MBEDTLS_ENTROPY_C
#define MBEDTLS_ERROR_C
#define MBEDTLS_MD_C
#define MBEDTLS_MD5_C
#define MBEDTLS_OID_C
#define MBEDTLS_PKCS5_C
#define MBEDTLS_PK_C
#define MBEDTLS_PK_PARSE_C
#define MBEDTLS_PLATFORM_C
#define MBEDTLS_RSA_C
#define MBEDTLS_SHA1_C
#define MBEDTLS_SHA224_C
#define MBEDTLS_SHA256_C
#define MBEDTLS_SHA512_C
#define MBEDTLS_SSL_CLI_C
#define MBEDTLS_SSL_SRV_C
#define MBEDTLS_SSL_TLS_C
#define MBEDTLS_X509_CRT_PARSE_C
#define MBEDTLS_X509_USE_C
#define MBEDTLS_AES_FEWER_TABLES
/* TLS 1.2 */
#define MBEDTLS_SSL_PROTO_TLS1_2
#define MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED
#define MBEDTLS_GCM_C
#define MBEDTLS_ECDH_C
#define MBEDTLS_ECP_C
#define MBEDTLS_ECDSA_C
#define MBEDTLS_ASN1_WRITE_C
// The following is needed to parse a certificate
#define MBEDTLS_PEM_PARSE_C
#define MBEDTLS_BASE64_C

2023年11月22日 星期三

Raspberry Pi || Node-RED || Pico W || TLS: Ep2. MQTT (mosquitto) 2wayauth

 本文章介紹在Raspberry Pi 3B+上建立Mosquitto MQTT Broker。並在Node-RED 與Pico W上互相Publish 與subscribe message。mosquitto 設定需要user name與password。並啟用TLS 2 way authentication功能。

有關Mosquitto MQTT Broker詳細設定步驟請參閱「MQTT 簡介:使用Mosquitto Broker」文章介紹。

本專案詳細設定步驟請參閱下列影片。

成果影片:





程式碼:

主程式:

  • picow_tls_mqtt.c

#include "stdio.h"
#include "pico/stdlib.h"
#include "pico/cyw43_arch.h"
#include "picow_tls_mqtt.h"
#include "picow_tls_cert.h"
#include "lwip/apps/mqtt.h"
#include "lwip/altcp.h"
#include "pio_rotary_encoder.h"
#include "ws2812.h"
#define WS2812_TOTAL_PIXELS 12
#define WIFI_SSID "your-ssid"
#define WIFI_PASSWORD "your-password"
#define SERVER_IP "your-mqtt-broker-ip"
static mqtt_client_instance_t mqtt_client;
static int8_t rotary_value=0;
static uint32_t ws2812_color=0;
uint8_t pub_buff[1024];
void update_ws2812_color() {
for (int i=0; i < rotary_value; i++) {
ws2812_put_pixel_rgb(ws2812_color);
}
for (int i=rotary_value; i < WS2812_TOTAL_PIXELS; i++) {
ws2812_put_pixel_rgb(0x0);
}
}
void mqtt_incoming_data_cb(void *arg, const u8_t *data, u16_t len, u8_t flags)
{
mqtt_client_instance_t* client_inst = (mqtt_client_instance_t*)arg;
LWIP_UNUSED_ARG(data);
memcpy(client_inst->sub_message+client_inst->sub_message_offset, data, len);
client_inst->sub_message_offset += len;
if (flags == MQTT_DATA_FLAG_LAST) {
client_inst->sub_message_recved = true;
client_inst->sub_message[client_inst->sub_message_offset]=0;
ws2812_color = strtol(client_inst->sub_message, NULL, 16);
update_ws2812_color();
}
}
void mqtt_incoming_publish_cb(void *arg, const char *topic, u32_t tot_len)
{
mqtt_client_instance_t* client_inst = (mqtt_client_instance_t*)arg;
client_inst->sub_message_offset=0;
client_inst->sub_message_recved=false;
strcpy(client_inst->sub_topic, topic);
}
void mqtt_request_cb(void *arg, err_t err)
{
mqtt_client_instance_t* client_inst = (mqtt_client_instance_t*)arg;
client_inst->connected=true;
//LWIP_PLATFORM_DIAG(("MQTT client \"%s\" request cb: err %d\n", client_inst->mqtt_connect_client_info.client_id, (int)err));
}
void mqtt_pub_request_cb(void *arg, err_t err)
{
mqtt_client_instance_t* client_inst = (mqtt_client_instance_t*)arg;
//LWIP_PLATFORM_DIAG(("MQTT client \"%s\" pub request cb: err %d\n", client_inst->mqtt_connect_client_info.client_id, (int)err));
}
void mqtt_connection_cb(mqtt_client_t *client, void *arg, mqtt_connection_status_t status)
{
mqtt_client_instance_t* client_inst = (mqtt_client_instance_t*)arg;
if (status == MQTT_CONNECT_ACCEPTED) {
mqtt_sub_unsub(client,
"ws2812_color", 1,
mqtt_request_cb, LWIP_CONST_CAST(void*, client_inst),
1);
}
}
void rotary_encoder_cb(uint8_t action) {
if (action == PIO_RE_CLOCKWISE || action == PIO_RE_COUNTERCLOCKWISE) {
if (action == PIO_RE_CLOCKWISE) {
rotary_value -= 1;
if (rotary_value < 0) rotary_value=0;
}
if (action == PIO_RE_COUNTERCLOCKWISE) {
rotary_value += 1;
if (rotary_value > 12) rotary_value=12;
}
}
}
int main()
{
int wifi_stat;
err_t mqtt_stat;
stdio_init_all();
if (cyw43_arch_init()) {
printf("cyw43 init error\n");
return 0;
}
pio_rotary_encoder_default_init();
pio_rotary_encoder_set_callback(rotary_encoder_cb);
ws2812_default_pio_init();
for (int i=0; i < WS2812_TOTAL_PIXELS; i++) ws2812_put_pixel(0x0);
sleep_ms(50);
//led: red
ws2812_put_pixel_rgb(0xff0000);
// connect to WiFi
cyw43_arch_enable_sta_mode();
wifi_stat = cyw43_arch_wifi_connect_timeout_ms(WIFI_SSID, WIFI_PASSWORD, CYW43_AUTH_WPA2_AES_PSK, 30000);
if (wifi_stat) {
printf("wifi connect error:%d\n", wifi_stat);
return 0;
}
printf("Wifi connected\n");
//led: blue
ws2812_put_pixel_rgb(0xff);
//mqtt_client.tls_config = altcp_tls_create_config_client(ca_cert, sizeof(ca_cert));
mqtt_client.tls_config = altcp_tls_create_config_client_2wayauth(ca_cert, sizeof(ca_cert), mqtt_client_key, sizeof(mqtt_client_key), NULL, 0, mqtt_client_cert, sizeof(mqtt_client_cert));
assert(mqtt_client.tls_config);
/*
// if DNS query
mbedtls_ssl_set_hostname(altcp_tls_context(tls_client->tpcb), SERVER_NAME);
tls_client->dns_found=false;
cyw43_arch_lwip_begin();
tcp_stat = dns_gethostbyname(SERVER_NAME, &tls_client->server_addr, tls_client_dns_found, tls_client);
if (tcp_stat == ERR_OK) printf("dns err ok\n");
if (tcp_stat == ERR_INPROGRESS) printf("dns inprogress...\n");
while(!tls_client->dns_found && tcp_stat != ERR_OK) {
cyw43_arch_poll();
sleep_ms(10);
}
*/
// if not by dns
ipaddr_aton(SERVER_IP, &mqtt_client.ipaddr);
// connect to mqtt server;
mqtt_client.mqtt_connect_client_info = (struct mqtt_connect_client_info_t){
"clientid", //client_id;
"username", //client_user;
"password", //client_pass;
100,//keep_alive;
NULL, // will_topic;
NULL, // will_msg;
0, //will_qos;
0,// will_retain;
#if LWIP_ALTCP && LWIP_ALTCP_TLS
mqtt_client.tls_config //tls_config;
#endif
};
mqtt_client.client = mqtt_client_new();
//mbedtls_ssl_set_hostname(altcp_tls_context(conn)), SERVER_NAME);
mqtt_set_inpub_callback(mqtt_client.client,
mqtt_incoming_publish_cb,
mqtt_incoming_data_cb,
LWIP_CONST_CAST(void*, &mqtt_client.mqtt_connect_client_info));
mqtt_client.connected=false;
cyw43_arch_lwip_begin();
mqtt_stat = mqtt_client_connect(mqtt_client.client,
&mqtt_client.ipaddr, LWIP_IANA_PORT_SECURE_MQTT,
mqtt_connection_cb, LWIP_CONST_CAST(void*, &mqtt_client),
&mqtt_client.mqtt_connect_client_info);
cyw43_arch_lwip_end();
absolute_time_t timeout = make_timeout_time_ms(10000);
while(!mqtt_client.connected && absolute_time_diff_us(get_absolute_time(), timeout) > 0) {
sleep_ms(10);
cyw43_arch_poll();
printf(".");
}
printf("\n");
if (!mqtt_client.connected) {
printf("connect mqtt server:timeout\n");
return 0;
}
printf("mqtt connected\n");
// led : green
ws2812_put_pixel_rgb(0x00ff00);
ws2812_color = 0x00ff00;
rotary_value = 1;
sprintf(pub_buff, "%02d", rotary_value);
mqtt_publish(mqtt_client.client, "ROTARY", pub_buff, 2, 1, 0, mqtt_pub_request_cb, &mqtt_client);
uint8_t pre_rotary_value=rotary_value;
uint32_t pre_ws2812_color=ws2812_color;
while(1) {
cyw43_arch_poll();
sleep_ms(10);
if (pre_rotary_value != rotary_value) {
sprintf(pub_buff, "%02d", rotary_value);
mqtt_publish(mqtt_client.client, "ROTARY", pub_buff, 2, 1, 0, mqtt_pub_request_cb, &mqtt_client);
update_ws2812_color();
pre_rotary_value = rotary_value;
}
}
cyw43_arch_deinit();
return 0;
}

  • picow_tls_mqtt.h

#ifndef __PICOW_TLS_CLIENT_H__
#define __PICOW_TLS_CLIENT_H__
#include "lwip/pbuf.h"
#include "lwip/altcp_tcp.h"
#include "lwip/altcp_tcp.h"
#include "lwip/altcp_tls.h"
#include "lwip/dns.h"
#include "lwip/apps/mqtt.h"
#define RECV_BUFF_MAX_LEN 1024
typedef struct {
mqtt_client_t *client;
struct altcp_tls_config *tls_config;
ip_addr_t ipaddr;
struct mqtt_connect_client_info_t mqtt_connect_client_info;
bool connected;
uint8_t sub_topic[256];
uint8_t sub_message[RECV_BUFF_MAX_LEN];
uint16_t sub_message_offset;
bool sub_message_recved;
} mqtt_client_instance_t;
#endif

  • picow_tls_cert.h

#define ca_cert "-----BEGIN CERTIFICATE-----\n\
...
-----END CERTIFICATE-----\n"
#define mqtt_client_cert "-----BEGIN CERTIFICATE-----\n\
...
-----END CERTIFICATE-----"
#define mqtt_client_key "-----BEGIN RSA PRIVATE KEY-----\n\
...
-----END RSA PRIVATE KEY-----"

  • CMakeLists.txt

# Generated Cmake Pico project file
cmake_minimum_required(VERSION 3.13)
set(CMAKE_C_STANDARD 11)
set(CMAKE_CXX_STANDARD 17)
# Initialise pico_sdk from installed location
# (note this can come from environment, CMake cache etc)
set(PICO_SDK_PATH "/home/duser/pico/pico-sdk")
set(PICO_BOARD pico_w CACHE STRING "Board type")
# Pull in Raspberry Pi Pico SDK (must be before project)
include(pico_sdk_import.cmake)
if (PICO_SDK_VERSION_STRING VERSION_LESS "1.4.0")
message(FATAL_ERROR "Raspberry Pi Pico SDK version 1.4.0 (or later) required. Your version is ${PICO_SDK_VERSION_STRING}")
endif()
project(picow_tls_mqtt C CXX ASM)
# Initialise the Raspberry Pi Pico SDK
pico_sdk_init()
# Add executable. Default name is the project name, version 0.1
add_executable(picow_tls_mqtt picow_tls_mqtt.c)
pico_set_program_name(picow_tls_mqtt "picow_tls_client")
pico_set_program_version(picow_tls_mqtt "0.1")
pico_enable_stdio_uart(picow_tls_mqtt 1)
pico_enable_stdio_usb(picow_tls_mqtt 0)
# Add the standard library to the build
target_link_libraries(picow_tls_mqtt
pico_stdlib)
# Add the standard include files to the build
target_include_directories(picow_tls_mqtt PRIVATE
${CMAKE_CURRENT_LIST_DIR}
${CMAKE_CURRENT_LIST_DIR}/.. # for our common lwipopts or any other standard includes, if required
)
# Add any user requested libraries
target_link_libraries(picow_tls_mqtt
pico_cyw43_arch_lwip_poll
pico_lwip_mqtt
pico_lwip_mbedtls
pico_mbedtls
)
add_subdirectory(rotary_encoder)
target_link_libraries(picow_tls_mqtt
rotary_encoder
)
add_subdirectory(ws2812)
target_link_libraries(picow_tls_mqtt
ws2812
)
pico_add_extra_outputs(picow_tls_mqtt)

  • lwipopts.h

#ifndef __LWIPOPTS_H__
#define __LWIPOPTS_H__
// Common settings used in most of the pico_w examples
// (see https://www.nongnu.org/lwip/2_1_x/group__lwip__opts.html for details)
// allow override in some examples
#ifndef NO_SYS
#define NO_SYS 1
#endif
// allow override in some examples
#ifndef LWIP_SOCKET
#define LWIP_SOCKET 0
#endif
#if PICO_CYW43_ARCH_POLL
#define MEM_LIBC_MALLOC 1
#else
// MEM_LIBC_MALLOC is incompatible with non polling versions
#define MEM_LIBC_MALLOC 0
#endif
#define MEM_ALIGNMENT 4
#define MEM_SIZE 4000
#define MEMP_NUM_TCP_SEG 32
#define MEMP_NUM_ARP_QUEUE 10
#define PBUF_POOL_SIZE 24
#define LWIP_ARP 1
#define LWIP_ETHERNET 1
#define LWIP_ICMP 1
#define LWIP_RAW 1
#define TCP_WND (8 * TCP_MSS)
#define TCP_MSS 1460
#define TCP_SND_BUF (8 * TCP_MSS)
#define TCP_SND_QUEUELEN ((4 * (TCP_SND_BUF) + (TCP_MSS - 1)) / (TCP_MSS))
#define LWIP_NETIF_STATUS_CALLBACK 1
#define LWIP_NETIF_LINK_CALLBACK 1
#define LWIP_NETIF_HOSTNAME 1
#define LWIP_NETCONN 0
#define MEM_STATS 0
#define SYS_STATS 0
#define MEMP_STATS 0
#define LINK_STATS 0
// #define ETH_PAD_SIZE 2
#define LWIP_CHKSUM_ALGORITHM 3
#define LWIP_DHCP 1
#define LWIP_IPV4 1
#define LWIP_TCP 1
#define LWIP_UDP 1
#define LWIP_DNS 1
#define LWIP_TCP_KEEPALIVE 1
#define LWIP_NETIF_TX_SINGLE_PBUF 1
#define DHCP_DOES_ARP_CHECK 0
#define LWIP_DHCP_DOES_ACD_CHECK 0
#ifndef NDEBUG
#define LWIP_DEBUG 1
#define LWIP_STATS 1
#define LWIP_STATS_DISPLAY 1
#endif
#define ETHARP_DEBUG LWIP_DBG_OFF
#define NETIF_DEBUG LWIP_DBG_OFF
#define PBUF_DEBUG LWIP_DBG_OFF
#define API_LIB_DEBUG LWIP_DBG_OFF
#define API_MSG_DEBUG LWIP_DBG_OFF
#define SOCKETS_DEBUG LWIP_DBG_OFF
#define ICMP_DEBUG LWIP_DBG_OFF
#define INET_DEBUG LWIP_DBG_OFF
#define IP_DEBUG LWIP_DBG_OFF
#define IP_REASS_DEBUG LWIP_DBG_OFF
#define RAW_DEBUG LWIP_DBG_OFF
#define MEM_DEBUG LWIP_DBG_OFF
#define MEMP_DEBUG LWIP_DBG_OFF
#define SYS_DEBUG LWIP_DBG_OFF
#define TCP_DEBUG LWIP_DBG_OFF
#define TCP_INPUT_DEBUG LWIP_DBG_OFF
#define TCP_OUTPUT_DEBUG LWIP_DBG_OFF
#define TCP_RTO_DEBUG LWIP_DBG_OFF
#define TCP_CWND_DEBUG LWIP_DBG_OFF
#define TCP_WND_DEBUG LWIP_DBG_OFF
#define TCP_FR_DEBUG LWIP_DBG_OFF
#define TCP_QLEN_DEBUG LWIP_DBG_OFF
#define TCP_RST_DEBUG LWIP_DBG_OFF
#define UDP_DEBUG LWIP_DBG_OFF
#define TCPIP_DEBUG LWIP_DBG_OFF
#define PPP_DEBUG LWIP_DBG_OFF
#define SLIP_DEBUG LWIP_DBG_OFF
#define DHCP_DEBUG LWIP_DBG_OFF
/* in Pico-SDK above 1.5.0 */
#define MEMP_NUM_SYS_TIMEOUT (LWIP_NUM_SYS_TIMEOUT_INTERNAL+1)
/* TCP WND must be at least 16 kb to match TLS record size
or you will get a warning "altcp_tls: TCP_WND is smaller than the RX decrypion buffer, connection RX might stall!" */
// tls
#undef TCP_WND
#define TCP_WND 16384
#define LWIP_ALTCP 1
#define LWIP_ALTCP_TLS 1
#define LWIP_ALTCP_TLS_MBEDTLS 1
#define LWIP_DEBUG 1
#define ALTCP_MBEDTLS_DEBUG LWIP_DBG_ON
/* set authmode */
#define ALTCP_MBEDTLS_AUTHMODE MBEDTLS_SSL_VERIFY_REQUIRED
#endif /* __LWIPOPTS_H__ */

  • mbedtls_config.h

/* Workaround for some mbedtls source files using INT_MAX without including limits.h */
#include < limits.h >
#define MBEDTLS_NO_PLATFORM_ENTROPY
#define MBEDTLS_ENTROPY_HARDWARE_ALT
#define MBEDTLS_SSL_OUT_CONTENT_LEN 2048
#define MBEDTLS_ALLOW_PRIVATE_ACCESS
#define MBEDTLS_HAVE_TIME
#define MBEDTLS_CIPHER_MODE_CBC
#define MBEDTLS_ECP_DP_SECP192R1_ENABLED
#define MBEDTLS_ECP_DP_SECP224R1_ENABLED
#define MBEDTLS_ECP_DP_SECP256R1_ENABLED
#define MBEDTLS_ECP_DP_SECP384R1_ENABLED
#define MBEDTLS_ECP_DP_SECP521R1_ENABLED
#define MBEDTLS_ECP_DP_SECP192K1_ENABLED
#define MBEDTLS_ECP_DP_SECP224K1_ENABLED
#define MBEDTLS_ECP_DP_SECP256K1_ENABLED
#define MBEDTLS_ECP_DP_BP256R1_ENABLED
#define MBEDTLS_ECP_DP_BP384R1_ENABLED
#define MBEDTLS_ECP_DP_BP512R1_ENABLED
#define MBEDTLS_ECP_DP_CURVE25519_ENABLED
#define MBEDTLS_KEY_EXCHANGE_RSA_ENABLED
#define MBEDTLS_PKCS1_V15
#define MBEDTLS_SHA256_SMALLER
#define MBEDTLS_SSL_SERVER_NAME_INDICATION
#define MBEDTLS_AES_C
#define MBEDTLS_ASN1_PARSE_C
#define MBEDTLS_BIGNUM_C
#define MBEDTLS_CIPHER_C
#define MBEDTLS_CTR_DRBG_C
#define MBEDTLS_ENTROPY_C
#define MBEDTLS_ERROR_C
#define MBEDTLS_MD_C
#define MBEDTLS_MD5_C
#define MBEDTLS_OID_C
#define MBEDTLS_PKCS5_C
#define MBEDTLS_PK_C
#define MBEDTLS_PK_PARSE_C
#define MBEDTLS_PLATFORM_C
#define MBEDTLS_RSA_C
#define MBEDTLS_SHA1_C
#define MBEDTLS_SHA224_C
#define MBEDTLS_SHA256_C
#define MBEDTLS_SHA512_C
#define MBEDTLS_SSL_CLI_C
#define MBEDTLS_SSL_SRV_C
#define MBEDTLS_SSL_TLS_C
#define MBEDTLS_X509_CRT_PARSE_C
#define MBEDTLS_X509_USE_C
#define MBEDTLS_AES_FEWER_TABLES
/* TLS 1.2 */
#define MBEDTLS_SSL_PROTO_TLS1_2
#define MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED
#define MBEDTLS_GCM_C
#define MBEDTLS_ECDH_C
#define MBEDTLS_ECP_C
#define MBEDTLS_ECDSA_C
#define MBEDTLS_ASN1_WRITE_C
// The following is needed to parse a certificate
#define MBEDTLS_PEM_PARSE_C
#define MBEDTLS_BASE64_C

Rotary Encoder:

  • pio_rotary_encoder.c

#include "pico/stdio.h"
#include "pico/stdlib.h"
#include "hardware/pio.h"
#include "hardware/gpio.h"
#include "pio_rotary_encoder.h"
#include "rotary_encoder.pio.h"
#include "hardware/clocks.h"
void (*pio_callback)(uint8_t action);
void rotary_encoder_irq_handler() {
if (pio_interrupt_get(RE_PIO, ROTARY_INT_NUM)) { // 1: clockwise, 2: countclockwise
pio_interrupt_clear(RE_PIO, ROTARY_INT_NUM);
uint32_t act = pio_sm_get_blocking(RE_PIO, RE_SM_ROTARY);
if (act == PIO_RE_CLOCKWISE || act == PIO_RE_COUNTERCLOCKWISE) { //PIO_RE_CLOCKWISE;
pio_callback((uint8_t)act);
} else {
pio_callback(PIO_RE_UNKNOW);
}
}
if (pio_interrupt_get(RE_PIO, SWITCH_INT_NUM)) {
pio_interrupt_clear(RE_PIO, SWITCH_INT_NUM);
uint8_t action = PIO_RE_SWPRESS;
pio_callback(action);
}
}
void pio_rotary_switch_init(PIO pio, uint sm, uint in_base, uint freq) {
gpio_pull_up(in_base); // software pull-up
pio_sm_config c;
uint offset = pio_add_program(pio, &rotary_switch_program);
c = rotary_switch_program_get_default_config(offset);
pio_gpio_init(pio, in_base);
pio_sm_set_consecutive_pindirs(pio, sm, in_base, 1,false);
sm_config_set_in_pins(&c, in_base);
sm_config_set_in_shift(&c, false, false, 1);
float div = clock_get_hz(clk_sys)/freq;
sm_config_set_clkdiv(&c, div);
uint pio_irq_num = (pio==pio0) ? PIO0_IRQ_0: PIO1_IRQ_0;
pio_set_irq0_source_enabled(pio, pis_interrupt0, true);
pio_set_irq0_source_enabled(pio, pis_interrupt1, true);
irq_add_shared_handler(pio_irq_num, rotary_encoder_irq_handler, PICO_SHARED_IRQ_HANDLER_DEFAULT_ORDER_PRIORITY);
irq_set_enabled(pio_irq_num, true);
pio_sm_init(pio, sm, offset, &c);
pio_sm_set_enabled(pio, sm, true);
}
void pio_rotary_encoder_init(PIO pio, uint sm, uint in_base, uint freq) {
pio_sm_config c;
uint offset = pio_add_program(pio, &rotary_encoder_program);
c = rotary_encoder_program_get_default_config(offset);
for (int i=0; i < 2; i++) pio_gpio_init(pio, in_base+i);
pio_sm_set_consecutive_pindirs(pio, sm, in_base, 2,false);
sm_config_set_in_pins(&c, in_base);
sm_config_set_in_shift(&c, false, false, 2);
float div = clock_get_hz(clk_sys)/freq;
sm_config_set_clkdiv(&c, div);
uint pio_irq_num = (pio==pio0) ? PIO0_IRQ_0: PIO1_IRQ_0;
pio_set_irq0_source_enabled(pio, pis_interrupt0, true);
irq_add_shared_handler(pio_irq_num, rotary_encoder_irq_handler, PICO_SHARED_IRQ_HANDLER_DEFAULT_ORDER_PRIORITY);
irq_set_enabled(pio_irq_num, true);
pio_sm_init(pio, sm, offset, &c);
pio_sm_set_enabled(pio, sm, true);
}
void pio_rotary_encoder_default_init() {
pio_rotary_switch_init(RE_PIO, RE_SM_SWITCH, RE_SW_PIN, 2000);
pio_rotary_encoder_init(RE_PIO, RE_SM_ROTARY, RE_CLK_DT_PIN, RE_SM_FREQ);
}
void pio_rotary_encoder_set_callback(void (*cb)(uint8_t action)) {
pio_callback=cb;
}

  • pio_rotary_encoder.h

#ifndef _PIO_ROTARY_ENCODER_H_
#define _PIO_ROTARY_ENCODER_H_
#include "hardware/pio.h"
#define RE_PIO pio0
#define RE_CLK_DT_PIN 16
#define RE_SW_PIN 18
#define RE_PIO pio0
#define RE_SM_ROTARY (0)
#define RE_SM_SWITCH (1)
#define RE_SM_FREQ 100000
enum PIO_RE_ACTION{
PIO_RE_CLOCKWISE=1,
PIO_RE_COUNTERCLOCKWISE=2,
PIO_RE_SWPRESS=3,
PIO_RE_UNKNOW=4
};
void pio_rotary_switch_init(PIO pio, uint sm, uint in_base, uint freq);
void pio_rotary_encoder_init(PIO pio, uint sm, uint in_base, uint freq);
void pio_rotary_encoder_default_init();
void pio_rotary_encoder_set_callback(void (*cb)(uint8_t action));
#endif

  • rotary_encoder.pio

.define public ROTARY_INT_NUM 0
.define public SWITCH_INT_NUM 1
.program rotary_encoder
start:
in pins, 2 [31]
mov x, isr ; CLK, DT present state
mov y, osr ; CLK, DT previous state
jmp x!=y, rotary ; not eaqual, rotary occure
jmp saveoldvalue ; clear ISR
rotary:
set y, 0b01 ;
jmp x!=y, pos10
jmp check_dir
pos10:
set y, 0b10
jmp x!=y, saveoldvalue
jmp check_dir
saveoldvalue:
mov osr, isr
in NULL, 32 ;; imporant to clear ISR
jmp start
check_dir:
out y,1 ; previous state DT bit
mov osr,x ; save oldvalue
in NULL, 32 ; clear ISR
in x, 1 ; shift in present state DT bit
mov x, isr ; present state DT bit
jmp x!=y counterclockwise
jmp clockwise
clockwise:
set y,1 ; clockwise: value = 1
jmp output
counterclockwise:
set y,2 ; counterclockwise: value = 2
output:
mov isr, y
push
irq wait ROTARY_INT_NUM
jmp start
.program rotary_switch
wait 0 pin 0 [31] ; Switch PIN is initially pulled up
irq wait SWITCH_INT_NUM ; [31] wait for button bounce
wait 1 pin 0 [31]

  • CMakeLists.txt

add_library(rotary_encoder INTERFACE)
pico_generate_pio_header(rotary_encoder ${CMAKE_CURRENT_LIST_DIR}/rotary_encoder.pio)
target_sources(rotary_encoder INTERFACE
${CMAKE_CURRENT_LIST_DIR}/pio_rotary_encoder.c
)
target_include_directories(rotary_encoder INTERFACE
${CMAKE_CURRENT_LIST_DIR}
)
target_link_libraries(rotary_encoder INTERFACE
hardware_pio
)

ws2812:

  • ws2812.c

#include "pico/stdio.h"
#include "pico/stdlib.h"
#include "hardware/pio.h"
#include "hardware/clocks.h"
#include "ws2812.pio.h"
#include "ws2812.h"
void ws2812_pio_init(PIO pio, uint sm, uint pin, float freq) {
pio_gpio_init(pio, pin);
pio_sm_set_consecutive_pindirs(pio, sm, pin, 1, true);
uint offset = pio_add_program(pio, &ws2812_program);
pio_sm_config c = ws2812_program_get_default_config(offset);
sm_config_set_sideset_pins(&c, pin);
sm_config_set_out_shift(&c, false, true, 24);
sm_config_set_fifo_join(&c, PIO_FIFO_JOIN_TX);
int cycles_per_bit = 6;
float div = clock_get_hz(clk_sys) / (freq * cycles_per_bit);
sm_config_set_clkdiv(&c, div);
pio_sm_init(pio, sm, offset, &c);
pio_sm_set_enabled(pio, sm, true);
}
void ws2812_put_pixel(uint32_t pixel_grb) {
pio_sm_put_blocking(WS2812_PIO, WS2812_SM, pixel_grb << 8u);
}
void ws2812_put_pixel_rgb(uint32_t pixel_rgb) {
uint32_t grb = ((pixel_rgb & 0xff00) >> 8) << 16 | ((pixel_rgb & 0xff0000) >> 16) << 8 | pixel_rgb &0xff;
ws2812_put_pixel(grb);
}
void ws2812_default_pio_init() {
ws2812_pio_init(WS2812_PIO, WS2812_SM, WS2812_PIN, WS2812_PIO_FREQ);
}

  • ws2812.h

#ifndef _WS2812_H_
#define _WS2812_H_
#include "hardware/pio.h"
#define WS2812_PIO pio1
#define WS2812_SM (1)
#define WS2812_PIN (19)
#define WS2812_PIO_FREQ (800000)
void ws2812_pio_init(PIO pio, uint sm, uint pin, float freq);
void ws2812_default_pio_init();
void ws2812_put_pixel(uint32_t pixel_grb);
void ws2812_put_pixel_rgb(uint32_t pixel_rgb);
#endif

  • ws2812.pio

.program ws2812
.side_set 1
.wrap_target
bitloop:
out x, 1 side 0 [1]
jmp !x do_zero side 1 [1]
do_one:
jmp bitloop side 1 [1]
do_zero:
nop side 0 [1]
.wrap

  • CMakeLists.txt

add_library(ws2812 INTERFACE)
pico_generate_pio_header(ws2812 ${CMAKE_CURRENT_LIST_DIR}/ws2812.pio)
target_sources(ws2812 INTERFACE
${CMAKE_CURRENT_LIST_DIR}/ws2812.c
)
target_include_directories(ws2812 INTERFACE
${CMAKE_CURRENT_LIST_DIR}
)
target_link_libraries(ws2812 INTERFACE
hardware_pio
)