Skip to content
Snippets Groups Projects
Select Git revision
  • 7e23c5cc0da705223ad7d1dae1331d0691f1100a
  • master default protected
  • schneider/ir
  • rahix/user-space-ctx
  • schneider/iaq-python
  • schneider/ble-mini-demo
  • schneider/ble-ecg-stream-visu
  • schneider/mp-exception-print
  • schneider/sleep-display
  • schneider/deepsleep4
  • schneider/deepsleep2
  • schneider/deepsleep
  • schneider/ble-central
  • rahix/bluetooth-app-favorite
  • schneider/v1.17-changelog
  • schneider/ancs
  • schneider/png
  • schneider/freertos-list-debug
  • schneider/212-reset-hardware-when-entering-repl
  • schneider/bonding-fail-if-full
  • schneider/ble-fixes-2020-3
  • v1.18
  • v1.17
  • v1.16
  • v1.15
  • v1.14
  • v1.13
  • v1.12
  • v1.11
  • v1.10
  • v1.9
  • v1.8
  • v1.7
  • v1.6
  • v1.5
  • v1.4
  • v1.3
  • v1.2
  • v1.1
  • v1.0
  • release-1
41 results

os.c

Blame
  • os.c 5.73 KiB
    #include "epicardium.h"
    
    #include "py/obj.h"
    #include "py/runtime.h"
    
    #include <string.h>
    #include <strings.h>
    
    #include <stdbool.h>
    
    #include "os.h"
    
    bool pycrd_filename_restricted(const char *path)
    {
    	// files that cannot be opened in write modes
    	const char *const forbidden_files[] = {
    		"card10.bin", "menu.py", "main.py", "card10.cfg"
    	};
    
    	const char *fname = strchr(path, '/');
    	while (fname) {
    		path  = fname + 1;
    		fname = strchr(path, '/');
    	}
    	fname = path;
    
    	for (int i = 0;
    	     i < sizeof(forbidden_files) / sizeof(forbidden_files[0]);
    	     i++) {
    		if (strcasecmp(fname, forbidden_files[i]) == 0) {
    			return true;
    		}
    	}
    	return false;
    }
    
    static mp_obj_t mp_os_exit(size_t n_args, const mp_obj_t *args)
    {
    	int ret = 0;
    	if (n_args == 1) {
    		ret = mp_obj_get_int(args[0]);
    	}
    
    	epic_exit(ret);
    
    	/* unreachable */
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(exit_obj, 0, 1, mp_os_exit);
    
    static mp_obj_t mp_os_exec(mp_obj_t name_in)
    {
    	const char *name_ptr;
    	char name_str[256];
    	size_t len, maxlen;
    
    	name_ptr = mp_obj_str_get_data(name_in, &len);
    
    	/*
    	 * The string retrieved from MicroPython is not NULL-terminated so we
    	 * first need to copy it and add a NULL-byte.
    	 */
    	maxlen = len < (sizeof(name_str) - 1) ? len : (sizeof(name_str) - 1);
    	memcpy(name_str, name_ptr, maxlen);
    	name_str[maxlen] = '\0';
    
    	int ret = epic_exec(name_str);
    
    	/*
    	 * If epic_exec() returns, something went wrong.  We can raise an
    	 * exception in all cases.
    	 */
    	mp_raise_OSError(-ret);
    
    	/* unreachable */
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_1(exec_obj, mp_os_exec);
    
    static mp_obj_t mp_os_reset(void)
    {
    	epic_system_reset();
    
    	/* unreachable */
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_0(reset_obj, mp_os_reset);
    
    static mp_obj_t mp_os_listdir(mp_obj_t py_path)
    {
    	const char *path = mp_obj_str_get_str(py_path);
    	int fd           = epic_file_opendir(path);
    
    	if (fd < 0) {
    		mp_raise_OSError(-fd);
    	}
    	struct epic_stat entry;
    	mp_obj_list_t *list = mp_obj_new_list(0, NULL);
    	for (;;) {
    		int res = epic_file_readdir(fd, &entry);
    		if (res < 0) {
    			m_del_obj(mp_obj_list_t, list);
    			epic_file_close(fd);
    			mp_raise_OSError(-res);
    		}
    		if (entry.type == EPICSTAT_NONE) {
    			break;
    		}
    		mp_obj_list_append(
    			list, mp_obj_new_str(entry.name, strlen(entry.name))
    		);
    	}
    	epic_file_close(fd);
    	return MP_OBJ_FROM_PTR(list);
    }
    static MP_DEFINE_CONST_FUN_OBJ_1(listdir_obj, mp_os_listdir);
    
    static mp_obj_t mp_os_unlink(mp_obj_t py_path)
    {
    	const char *path = mp_obj_str_get_str(py_path);
    	if (pycrd_filename_restricted(path)) {
    		mp_raise_OSError(-EACCES);
    	}
    	int rc = epic_file_unlink(path);
    
    	if (rc < 0) {
    		mp_raise_OSError(-rc);
    	}
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_1(unlink_obj, mp_os_unlink);
    
    static mp_obj_t mp_os_mkdir(mp_obj_t py_path)
    {
    	const char *path = mp_obj_str_get_str(py_path);
    	int rc           = epic_file_mkdir(path);
    
    	if (rc < 0) {
    		mp_raise_OSError(-rc);
    	}
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_1(mkdir_obj, mp_os_mkdir);
    
    static mp_obj_t mp_os_rename(mp_obj_t py_oldp, mp_obj_t py_newp)
    {
    	const char *oldp = mp_obj_str_get_str(py_oldp);
    	const char *newp = mp_obj_str_get_str(py_newp);
    	if (pycrd_filename_restricted(oldp) ||
    	    pycrd_filename_restricted(newp)) {
    		mp_raise_OSError(-EACCES);
    	}
    	int rc = epic_file_rename(oldp, newp);
    
    	if (rc < 0) {
    		mp_raise_OSError(-rc);
    	}
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_2(rename_obj, mp_os_rename);
    
    static mp_obj_t mp_os_read_battery()
    {
    	float result;
    	int res = epic_read_battery_voltage(&result);
    	if (res < 0) {
    		mp_raise_OSError(-res);
    	}
    
    	return mp_obj_new_float(result);
    }
    static MP_DEFINE_CONST_FUN_OBJ_0(read_battery_obj, mp_os_read_battery);
    
    static mp_obj_t mp_os_urandom(mp_obj_t size_in)
    {
    	size_t size = mp_obj_get_int(size_in);
    	vstr_t vstr;
    
    	vstr_init_len(&vstr, size);
    	epic_trng_read((uint8_t *)vstr.buf, size);
    
    	return mp_obj_new_str_from_vstr(&mp_type_bytes, &vstr);
    }
    static MP_DEFINE_CONST_FUN_OBJ_1(urandom_obj, mp_os_urandom);
    
    enum usb_config_device {
    	USB_Device_None,
    	USB_Device_Flash,
    	USB_Device_Serial,
    };
    
    static mp_obj_t mp_os_usbconfig(mp_obj_t dev)
    {
    	int device = mp_obj_get_int(dev);
    	switch (device) {
    	case USB_Device_None:
    		epic_usb_shutdown();
    		break;
    	case USB_Device_Flash:
    		epic_usb_storage();
    		break;
    	case USB_Device_Serial:
    		epic_usb_cdcacm();
    		break;
    	default:
    		mp_raise_ValueError("Invalid parameter");
    	}
    	return mp_const_none;
    }
    static MP_DEFINE_CONST_FUN_OBJ_1(usbconfig_obj, mp_os_usbconfig);
    
    static const mp_rom_map_elem_t os_module_globals_table[] = {
    	{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_os) },
    	{ MP_ROM_QSTR(MP_QSTR_exit), MP_ROM_PTR(&exit_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_reset), MP_ROM_PTR(&reset_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_exec), MP_ROM_PTR(&exec_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_listdir), MP_ROM_PTR(&listdir_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_unlink), MP_ROM_PTR(&unlink_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_mkdir), MP_ROM_PTR(&mkdir_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_rename), MP_ROM_PTR(&rename_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_read_battery), MP_ROM_PTR(&read_battery_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_urandom), MP_ROM_PTR(&urandom_obj) },
    	{ MP_ROM_QSTR(MP_QSTR_usbconfig), MP_ROM_PTR(&usbconfig_obj) },
    
    	{ MP_ROM_QSTR(MP_QSTR_SERIAL), MP_ROM_INT(USB_Device_Serial) },
    	{ MP_ROM_QSTR(MP_QSTR_FLASH), MP_ROM_INT(USB_Device_Flash) },
    	{ MP_ROM_QSTR(MP_QSTR_NONE), MP_ROM_INT(USB_Device_None) },
    };
    
    static MP_DEFINE_CONST_DICT(os_module_globals, os_module_globals_table);
    
    // Define module object.
    const mp_obj_module_t os_module = {
    	.base    = { &mp_type_module },
    	.globals = (mp_obj_dict_t *)&os_module_globals,
    };
    
    /* This is a special macro that will make MicroPython aware of this module */
    /* clang-format off */
    MP_REGISTER_MODULE(MP_QSTR_os, os_module, MODULE_OS_ENABLED);