2020-10-27 20:55:03 -04:00
|
|
|
/*
|
|
|
|
* This file is part of the MicroPython project, http://micropython.org/
|
|
|
|
*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2020 Scott Shawcroft for Adafruit Industries
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2020-11-19 17:47:12 -05:00
|
|
|
#include <stdbool.h>
|
2020-11-23 22:44:53 -05:00
|
|
|
#include "py/mpconfig.h"
|
2022-08-17 16:52:13 -04:00
|
|
|
#include "py/mpstate.h"
|
2022-08-17 14:31:11 -04:00
|
|
|
#include "py/stackctrl.h"
|
2022-06-06 19:54:02 -04:00
|
|
|
#include "supervisor/background_callback.h"
|
2021-11-09 21:04:34 -05:00
|
|
|
#include "supervisor/workflow.h"
|
2022-06-06 19:54:02 -04:00
|
|
|
#include "supervisor/serial.h"
|
2021-11-09 21:04:34 -05:00
|
|
|
#include "supervisor/shared/workflow.h"
|
2021-08-04 19:27:54 -04:00
|
|
|
|
2022-06-06 19:54:02 -04:00
|
|
|
#if CIRCUITPY_BLEIO
|
|
|
|
#include "shared-bindings/_bleio/__init__.h"
|
|
|
|
#include "supervisor/shared/bluetooth/bluetooth.h"
|
|
|
|
#endif
|
|
|
|
|
2021-08-04 19:27:54 -04:00
|
|
|
#if CIRCUITPY_USB
|
2022-06-06 19:54:02 -04:00
|
|
|
#include "supervisor/usb.h"
|
2020-11-22 19:10:09 -05:00
|
|
|
#include "tusb.h"
|
2021-08-04 19:27:54 -04:00
|
|
|
#endif
|
2020-10-27 20:55:03 -04:00
|
|
|
|
2022-06-06 19:54:02 -04:00
|
|
|
#if CIRCUITPY_WEB_WORKFLOW
|
|
|
|
#include "supervisor/shared/web_workflow/web_workflow.h"
|
|
|
|
#endif
|
|
|
|
static background_callback_t workflow_background_cb;
|
|
|
|
|
2022-08-17 18:44:07 -04:00
|
|
|
static bool workflow_started = false;
|
|
|
|
|
2022-06-06 19:54:02 -04:00
|
|
|
static void workflow_background(void *data) {
|
2022-07-12 12:37:09 -04:00
|
|
|
#if CIRCUITPY_WEB_WORKFLOW
|
|
|
|
supervisor_web_workflow_background();
|
|
|
|
#endif
|
2022-06-06 19:54:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Called during a VM reset. Doesn't actually reset things.
|
2020-11-23 22:44:53 -05:00
|
|
|
void supervisor_workflow_reset(void) {
|
2022-06-06 19:54:02 -04:00
|
|
|
#if CIRCUITPY_BLEIO
|
|
|
|
supervisor_start_bluetooth();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CIRCUITPY_WEB_WORKFLOW
|
|
|
|
supervisor_start_web_workflow();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
workflow_background_cb.fun = workflow_background;
|
|
|
|
workflow_background_cb.data = NULL;
|
|
|
|
supervisor_workflow_request_background();
|
|
|
|
}
|
|
|
|
|
|
|
|
void supervisor_workflow_request_background(void) {
|
2022-08-17 18:44:07 -04:00
|
|
|
if (!workflow_started) {
|
|
|
|
return;
|
|
|
|
}
|
2022-06-06 19:54:02 -04:00
|
|
|
background_callback_add_core(&workflow_background_cb);
|
2020-11-23 22:44:53 -05:00
|
|
|
}
|
|
|
|
|
2020-11-25 17:52:06 -05:00
|
|
|
// Return true as soon as USB communication with host has started,
|
|
|
|
// even before enumeration is done.
|
2020-11-27 16:02:17 -05:00
|
|
|
// Not that some chips don't notice when USB is unplugged after first being plugged in,
|
|
|
|
// so this is not perfect, but tud_suspended() check helps.
|
2020-11-25 17:52:06 -05:00
|
|
|
bool supervisor_workflow_connecting(void) {
|
2021-08-04 19:27:54 -04:00
|
|
|
#if CIRCUITPY_USB
|
2020-11-27 16:02:17 -05:00
|
|
|
return tud_connected() && !tud_suspended();
|
2021-08-04 19:27:54 -04:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2020-11-25 17:52:06 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return true if host has completed connection to us (such as USB enumeration).
|
2020-11-19 17:47:12 -05:00
|
|
|
bool supervisor_workflow_active(void) {
|
2021-08-04 19:27:54 -04:00
|
|
|
#if CIRCUITPY_USB
|
2020-11-22 19:10:09 -05:00
|
|
|
// Eventually there might be other non-USB workflows, such as BLE.
|
|
|
|
// tud_ready() checks for usb mounted and not suspended.
|
|
|
|
return tud_ready();
|
2021-08-04 19:27:54 -04:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2020-10-27 20:55:03 -04:00
|
|
|
}
|
2022-06-06 19:54:02 -04:00
|
|
|
|
|
|
|
void supervisor_workflow_start(void) {
|
|
|
|
// Start USB after giving boot.py a chance to tweak behavior.
|
|
|
|
#if CIRCUITPY_USB
|
|
|
|
// Setup USB connection after heap is available.
|
|
|
|
// It needs the heap to build descriptors.
|
|
|
|
usb_init();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Set up any other serial connection.
|
|
|
|
serial_init();
|
|
|
|
|
|
|
|
#if CIRCUITPY_BLEIO
|
|
|
|
bleio_reset();
|
|
|
|
supervisor_bluetooth_enable_workflow();
|
|
|
|
supervisor_start_bluetooth();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CIRCUITPY_WEB_WORKFLOW
|
|
|
|
supervisor_start_web_workflow();
|
|
|
|
#endif
|
2022-08-17 18:44:07 -04:00
|
|
|
|
|
|
|
workflow_started = true;
|
2022-06-06 19:54:02 -04:00
|
|
|
}
|
2022-08-17 14:31:11 -04:00
|
|
|
|
|
|
|
FRESULT supervisor_workflow_mkdir_parents(FATFS *fs, char *path) {
|
|
|
|
FRESULT result = FR_OK;
|
|
|
|
// Make parent directories.
|
|
|
|
for (size_t j = 1; j < strlen(path); j++) {
|
|
|
|
if (path[j] == '/') {
|
|
|
|
path[j] = '\0';
|
|
|
|
result = f_mkdir(fs, path);
|
|
|
|
path[j] = '/';
|
|
|
|
if (result != FR_OK && result != FR_EXIST) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Make the target directory.
|
|
|
|
return f_mkdir(fs, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
FRESULT supervisor_workflow_delete_directory_contents(FATFS *fs, const TCHAR *path) {
|
|
|
|
FF_DIR dir;
|
|
|
|
FILINFO file_info;
|
|
|
|
// Check the stack since we're putting paths on it.
|
|
|
|
if (mp_stack_usage() >= MP_STATE_THREAD(stack_limit)) {
|
|
|
|
return FR_INT_ERR;
|
|
|
|
}
|
|
|
|
FRESULT res = FR_OK;
|
|
|
|
while (res == FR_OK) {
|
|
|
|
res = f_opendir(fs, &dir, path);
|
|
|
|
if (res != FR_OK) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
res = f_readdir(&dir, &file_info);
|
|
|
|
// We close and reopen the directory every time since we're deleting
|
|
|
|
// entries and it may invalidate the directory handle.
|
|
|
|
f_closedir(&dir);
|
|
|
|
if (res != FR_OK || file_info.fname[0] == '\0') {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
size_t pathlen = strlen(path);
|
|
|
|
size_t fnlen = strlen(file_info.fname);
|
|
|
|
TCHAR full_path[pathlen + 1 + fnlen];
|
|
|
|
memcpy(full_path, path, pathlen);
|
|
|
|
full_path[pathlen] = '/';
|
|
|
|
size_t full_pathlen = pathlen + 1 + fnlen;
|
|
|
|
memcpy(full_path + pathlen + 1, file_info.fname, fnlen);
|
|
|
|
full_path[full_pathlen] = '\0';
|
|
|
|
if ((file_info.fattrib & AM_DIR) != 0) {
|
|
|
|
res = supervisor_workflow_delete_directory_contents(fs, full_path);
|
|
|
|
}
|
|
|
|
if (res != FR_OK) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
res = f_unlink(fs, full_path);
|
|
|
|
}
|
|
|
|
f_closedir(&dir);
|
|
|
|
return res;
|
|
|
|
}
|