| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252 |
- #include "pocketpy/io.h"
- #if PK_ENABLE_OS
- #include <filesystem>
- #include <cstdio>
- #endif
- namespace pkpy{
- #if PK_ENABLE_OS
- struct FileIO {
- FILE* fp;
- bool is_text;
- FileIO(VM* vm, const Str& file, const Str& mode);
- void close();
- static void _register(VM* vm, PyObject* mod, PyObject* type);
- };
- static FILE* io_fopen(const char* name, const char* mode){
- #if _MSC_VER
- FILE* fp;
- errno_t err = fopen_s(&fp, name, mode);
- if(err != 0) return nullptr;
- return fp;
- #else
- return fopen(name, mode);
- #endif
- }
- static size_t io_fread(void* buffer, size_t size, size_t count, FILE* fp){
- #if _MSC_VER
- return fread_s(buffer, std::numeric_limits<size_t>::max(), size, count, fp);
- #else
- return fread(buffer, size, count, fp);
- #endif
- }
- unsigned char* _default_import_handler(const char* name_p, int name_size, int* out_size){
- std::string name(name_p, name_size);
- bool exists = std::filesystem::exists(std::filesystem::path(name));
- if(!exists) return nullptr;
- FILE* fp = io_fopen(name.c_str(), "rb");
- if(!fp) return nullptr;
- fseek(fp, 0, SEEK_END);
- int buffer_size = ftell(fp);
- unsigned char* buffer = new unsigned char[buffer_size];
- fseek(fp, 0, SEEK_SET);
- size_t sz = io_fread(buffer, 1, buffer_size, fp);
- (void)sz; // suppress warning
- fclose(fp);
- *out_size = buffer_size;
- return buffer;
- };
- void FileIO::_register(VM* vm, PyObject* mod, PyObject* type){
- vm->bind_constructor<3>(type, [](VM* vm, ArgsView args){
- Type cls = PK_OBJ_GET(Type, args[0]);
- return vm->heap.gcnew<FileIO>(cls, vm,
- py_cast<Str&>(vm, args[1]),
- py_cast<Str&>(vm, args[2]));
- });
- vm->bind(type, "read(self, size=-1)", [](VM* vm, ArgsView args){
- FileIO& io = PK_OBJ_GET(FileIO, args[0]);
- i64 size = CAST(i64, args[1]);
- i64 buffer_size;
- if(size < 0){
- long current = ftell(io.fp);
- fseek(io.fp, 0, SEEK_END);
- buffer_size = ftell(io.fp);
- fseek(io.fp, current, SEEK_SET);
- }else{
- buffer_size = size;
- }
- unsigned char* buffer = new unsigned char[buffer_size];
- i64 actual_size = io_fread(buffer, 1, buffer_size, io.fp);
- PK_ASSERT(actual_size <= buffer_size);
- // in text mode, CR may be dropped, which may cause `actual_size < buffer_size`
- Bytes b(buffer, actual_size);
- if(io.is_text) return VAR(b.str());
- return VAR(std::move(b));
- });
- vm->bind_func(type, "write", 2, [](VM* vm, ArgsView args){
- FileIO& io = PK_OBJ_GET(FileIO, args[0]);
- if(io.is_text){
- Str& s = CAST(Str&, args[1]);
- fwrite(s.data, 1, s.length(), io.fp);
- }else{
- Bytes& buffer = CAST(Bytes&, args[1]);
- fwrite(buffer.data(), 1, buffer.size(), io.fp);
- }
- return vm->None;
- });
- vm->bind_func(type, "tell", 1, [](VM* vm, ArgsView args){
- FileIO& io = PK_OBJ_GET(FileIO, args[0]);
- long pos = ftell(io.fp);
- if(pos == -1) vm->IOError(strerror(errno));
- return VAR(pos);
- });
- vm->bind_func(type, "seek", 3, [](VM* vm, ArgsView args){
- FileIO& io = PK_OBJ_GET(FileIO, args[0]);
- long offset = CAST(long, args[1]);
- int whence = CAST(int, args[2]);
- int ret = fseek(io.fp, offset, whence);
- if(ret != 0) vm->IOError(strerror(errno));
- return vm->None;
- });
- vm->bind_func(type, "close", 1, [](VM* vm, ArgsView args){
- FileIO& io = PK_OBJ_GET(FileIO, args[0]);
- io.close();
- return vm->None;
- });
- vm->bind_func(type, __exit__, 1, [](VM* vm, ArgsView args){
- FileIO& io = PK_OBJ_GET(FileIO, args[0]);
- io.close();
- return vm->None;
- });
- vm->bind_func(type, __enter__, 1, PK_LAMBDA(args[0]));
- }
- FileIO::FileIO(VM* vm, const Str& file, const Str& mode){
- this->is_text = mode.sv().find("b") == std::string::npos;
- fp = io_fopen(file.c_str(), mode.c_str());
- if(!fp) vm->IOError(strerror(errno));
- }
- void FileIO::close(){
- if(fp == nullptr) return;
- fclose(fp);
- fp = nullptr;
- }
- void add_module_io(VM* vm){
- PyObject* mod = vm->new_module("io");
- vm->register_user_class<FileIO>(mod, "FileIO");
- mod->attr().set("SEEK_SET", VAR(SEEK_SET));
- mod->attr().set("SEEK_CUR", VAR(SEEK_CUR));
- mod->attr().set("SEEK_END", VAR(SEEK_END));
- vm->bind(vm->builtins, "open(path, mode='r')", [](VM* vm, ArgsView args){
- PK_LOCAL_STATIC StrName m_io("io");
- PK_LOCAL_STATIC StrName m_FileIO("FileIO");
- return vm->call(vm->_modules[m_io]->attr(m_FileIO), args[0], args[1]);
- });
- }
- void add_module_os(VM* vm){
- PyObject* mod = vm->new_module("os");
- PyObject* path_obj = vm->heap.gcnew<DummyInstance>(vm->tp_object);
- mod->attr().set("path", path_obj);
-
- // Working directory is shared by all VMs!!
- vm->bind_func<0>(mod, "getcwd", [](VM* vm, ArgsView args){
- return VAR(std::filesystem::current_path().string());
- });
- vm->bind_func<1>(mod, "chdir", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- std::filesystem::current_path(path);
- return vm->None;
- });
- vm->bind_func<1>(mod, "listdir", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- std::filesystem::directory_iterator di;
- try{
- di = std::filesystem::directory_iterator(path);
- }catch(std::filesystem::filesystem_error&){
- vm->IOError(path.string());
- }
- List ret;
- for(auto& p: di) ret.push_back(VAR(p.path().filename().string()));
- return VAR(ret);
- });
- vm->bind_func<1>(mod, "remove", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- bool ok = std::filesystem::remove(path);
- if(!ok) vm->IOError("operation failed");
- return vm->None;
- });
- vm->bind_func<1>(mod, "mkdir", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- bool ok = std::filesystem::create_directory(path);
- if(!ok) vm->IOError("operation failed");
- return vm->None;
- });
- vm->bind_func<1>(mod, "rmdir", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- bool ok = std::filesystem::remove(path);
- if(!ok) vm->IOError("operation failed");
- return vm->None;
- });
- vm->bind_func<-1>(path_obj, "join", [](VM* vm, ArgsView args){
- std::filesystem::path path;
- for(int i=0; i<args.size(); i++){
- path /= CAST(Str&, args[i]).sv();
- }
- return VAR(path.string());
- });
- vm->bind_func<1>(path_obj, "exists", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- bool exists = std::filesystem::exists(path);
- return VAR(exists);
- });
- vm->bind_func<1>(path_obj, "basename", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- return VAR(path.filename().string());
- });
- vm->bind_func<1>(path_obj, "isdir", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- bool isdir = std::filesystem::is_directory(path);
- return VAR(isdir);
- });
- vm->bind_func<1>(path_obj, "isfile", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- bool isfile = std::filesystem::is_regular_file(path);
- return VAR(isfile);
- });
- vm->bind_func<1>(path_obj, "abspath", [](VM* vm, ArgsView args){
- std::filesystem::path path(CAST(Str&, args[0]).sv());
- return VAR(std::filesystem::absolute(path).string());
- });
- }
- #else
- void add_module_io(VM* vm){}
- void add_module_os(VM* vm){}
- unsigned char* _default_import_handler(const char* name_p, int name_size, int* out_size){
- return nullptr;
- }
- #endif
- } // namespace pkpy
|