| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230 |
- #include "pocketpy/objects/base.h"
- #include "pocketpy/pocketpy.h"
- #include "pocketpy/interpreter/vm.h"
- #if PK_ENABLE_OS == 1
- #include <errno.h>
- #if PY_SYS_PLATFORM == 0
- #include <direct.h>
- int platform_chdir(const char* path) { return _chdir(path); }
- bool platform_getcwd(char* buf, size_t size) { return _getcwd(buf, size) != NULL; }
- #elif PY_SYS_PLATFORM == 3 || PY_SYS_PLATFORM == 5
- #include <unistd.h>
- int platform_chdir(const char* path) { return chdir(path); }
- bool platform_getcwd(char* buf, size_t size) { return getcwd(buf, size) != NULL; }
- #else
- int platform_chdir(const char* path) { return -1; }
- bool platform_getcwd(char* buf, size_t size) { return false; }
- #endif
- static bool os_chdir(int argc, py_Ref argv) {
- PY_CHECK_ARGC(1);
- PY_CHECK_ARG_TYPE(0, tp_str);
- const char* path = py_tostr(py_arg(0));
- int code = platform_chdir(path);
- if(code != 0) {
- const char* msg = strerror(errno);
- return OSError("[Errno %d] %s: '%s'", errno, msg, path);
- }
- py_newnone(py_retval());
- return true;
- }
- static bool os_getcwd(int argc, py_Ref argv) {
- char buf[1024];
- if(!platform_getcwd(buf, sizeof(buf))) return OSError("getcwd() failed");
- py_newstr(py_retval(), buf);
- return true;
- }
- static bool os_system(int argc, py_Ref argv) {
- PY_CHECK_ARGC(1);
- PY_CHECK_ARG_TYPE(0, tp_str);
- #if PK_IS_DESKTOP_PLATFORM
- const char* cmd = py_tostr(py_arg(0));
- int code = system(cmd);
- py_newint(py_retval(), code);
- return true;
- #else
- return OSError("system() is not supported on this platform");
- #endif
- }
- static bool os_remove(int argc, py_Ref argv) {
- PY_CHECK_ARGC(1);
- PY_CHECK_ARG_TYPE(0, tp_str);
- const char* path = py_tostr(py_arg(0));
- int code = remove(path);
- if(code != 0) {
- const char* msg = strerror(errno);
- return OSError("[Errno %d] %s: '%s'", errno, msg, path);
- }
- py_newnone(py_retval());
- return true;
- }
- void pk__add_module_os() {
- py_Ref mod = py_newmodule("os");
- py_bindfunc(mod, "chdir", os_chdir);
- py_bindfunc(mod, "getcwd", os_getcwd);
- py_bindfunc(mod, "system", os_system);
- py_bindfunc(mod, "remove", os_remove);
- }
- typedef struct {
- const char* path;
- const char* mode;
- FILE* file;
- } io_FileIO;
- static bool io_FileIO__new__(int argc, py_Ref argv) {
- // __new__(cls, file, mode)
- PY_CHECK_ARGC(3);
- PY_CHECK_ARG_TYPE(1, tp_str);
- PY_CHECK_ARG_TYPE(2, tp_str);
- py_Type cls = py_totype(argv);
- io_FileIO* ud = py_newobject(py_retval(), cls, 0, sizeof(io_FileIO));
- ud->path = py_tostr(py_arg(1));
- ud->mode = py_tostr(py_arg(2));
- ud->file = fopen(ud->path, ud->mode);
- if(ud->file == NULL) {
- const char* msg = strerror(errno);
- return OSError("[Errno %d] %s: '%s'", errno, msg, ud->path);
- }
- return true;
- }
- static bool io_FileIO__enter__(int argc, py_Ref argv) {
- py_assign(py_retval(), py_arg(0));
- return true;
- }
- static bool io_FileIO__exit__(int argc, py_Ref argv) {
- io_FileIO* ud = py_touserdata(py_arg(0));
- if(ud->file != NULL) {
- fclose(ud->file);
- ud->file = NULL;
- }
- py_newnone(py_retval());
- return true;
- }
- static bool io_FileIO_read(int argc, py_Ref argv) {
- io_FileIO* ud = py_touserdata(py_arg(0));
- bool is_binary = ud->mode[strlen(ud->mode) - 1] == 'b';
- int size;
- if(argc == 1) {
- long current = ftell(ud->file);
- fseek(ud->file, 0, SEEK_END);
- size = ftell(ud->file);
- fseek(ud->file, current, SEEK_SET);
- } else if(argc == 2) {
- PY_CHECK_ARG_TYPE(1, tp_int);
- size = py_toint(py_arg(1));
- } else {
- return TypeError("read() takes at most 2 arguments (%d given)", argc);
- }
- if(is_binary) {
- void* dst = py_newbytes(py_retval(), size);
- int actual_size = fread(dst, 1, size, ud->file);
- py_bytes_resize(py_retval(), actual_size);
- } else {
- void* dst = malloc(size);
- int actual_size = fread(dst, 1, size, ud->file);
- py_newstrv(py_retval(), (c11_sv){dst, actual_size});
- free(dst);
- }
- return true;
- }
- static bool io_FileIO_tell(int argc, py_Ref argv) {
- io_FileIO* ud = py_touserdata(py_arg(0));
- py_newint(py_retval(), ftell(ud->file));
- return true;
- }
- static bool io_FileIO_seek(int argc, py_Ref argv) {
- PY_CHECK_ARGC(3);
- PY_CHECK_ARG_TYPE(1, tp_int);
- PY_CHECK_ARG_TYPE(2, tp_int);
- io_FileIO* ud = py_touserdata(py_arg(0));
- long cookie = py_toint(py_arg(1));
- int whence = py_toint(py_arg(2));
- py_newint(py_retval(), fseek(ud->file, cookie, whence));
- return true;
- }
- static bool io_FileIO_close(int argc, py_Ref argv) {
- PY_CHECK_ARGC(1);
- io_FileIO* ud = py_touserdata(py_arg(0));
- if(ud->file != NULL) {
- fclose(ud->file);
- ud->file = NULL;
- }
- py_newnone(py_retval());
- return true;
- }
- static bool io_FileIO_write(int argc, py_Ref argv) {
- PY_CHECK_ARGC(2);
- io_FileIO* ud = py_touserdata(py_arg(0));
- size_t written_size;
- if(ud->mode[strlen(ud->mode) - 1] == 'b') {
- PY_CHECK_ARG_TYPE(1, tp_bytes);
- int filesize;
- unsigned char* data = py_tobytes(py_arg(1), &filesize);
- written_size = fwrite(data, 1, filesize, ud->file);
- } else {
- PY_CHECK_ARG_TYPE(1, tp_str);
- c11_sv sv = py_tosv(py_arg(1));
- written_size = fwrite(sv.data, 1, sv.size, ud->file);
- }
- py_newint(py_retval(), written_size);
- return true;
- }
- void pk__add_module_io() {
- py_Ref mod = py_newmodule("io");
- py_Type FileIO = pk_newtype("FileIO", tp_object, mod, NULL, false, true);
- py_bindmagic(FileIO, __new__, io_FileIO__new__);
- py_bindmagic(FileIO, __enter__, io_FileIO__enter__);
- py_bindmagic(FileIO, __exit__, io_FileIO__exit__);
- py_bindmethod(FileIO, "read", io_FileIO_read);
- py_bindmethod(FileIO, "write", io_FileIO_write);
- py_bindmethod(FileIO, "close", io_FileIO_close);
- py_bindmethod(FileIO, "tell", io_FileIO_tell);
- py_bindmethod(FileIO, "seek", io_FileIO_seek);
- py_newint(py_emplacedict(mod, py_name("SEEK_SET")), SEEK_SET);
- py_newint(py_emplacedict(mod, py_name("SEEK_CUR")), SEEK_CUR);
- py_newint(py_emplacedict(mod, py_name("SEEK_END")), SEEK_END);
- py_setdict(&pk_current_vm->builtins, py_name("open"), py_tpobject(FileIO));
- }
- #else
- void pk__add_module_os() {}
- void pk__add_module_io() {}
- #endif
- void pk__add_module_sys() {
- py_Ref mod = py_newmodule("sys");
- py_newstr(py_emplacedict(mod, py_name("platform")), PY_SYS_PLATFORM_STRING);
- py_newstr(py_emplacedict(mod, py_name("version")), PK_VERSION);
- py_newlist(py_emplacedict(mod, py_name("argv")));
- }
|