507 lines
15 KiB
C++
507 lines
15 KiB
C++
/*
|
||
Copyright (c) 2016 Microsoft Corporation. All rights reserved.
|
||
Released under Apache 2.0 license as described in the file LICENSE.
|
||
|
||
Author: Leonardo de Moura
|
||
*/
|
||
#include <string>
|
||
#include <vector>
|
||
#include <cstdio>
|
||
#include <iostream>
|
||
#include "util/sstream.h"
|
||
#include "library/handle.h"
|
||
#include "library/io_state.h"
|
||
#include "library/tactic/tactic_state.h"
|
||
#include "library/vm/vm.h"
|
||
#include "library/vm/vm_array.h"
|
||
#include "library/vm/vm_string.h"
|
||
#include "library/vm/vm_io.h"
|
||
#include "library/process.h"
|
||
|
||
namespace lean {
|
||
vm_obj mk_io_result(vm_obj const & r) {
|
||
return mk_vm_constructor(0, 1, &r);
|
||
}
|
||
|
||
vm_obj mk_io_failure(vm_obj const & e) {
|
||
return mk_vm_constructor(1, 1, &e);
|
||
}
|
||
|
||
vm_obj mk_io_failure(std::string const & s) {
|
||
return mk_io_failure(mk_vm_constructor(0, to_obj(s)));
|
||
}
|
||
|
||
vm_obj mk_io_failure(sstream const & s) {
|
||
return mk_io_failure(mk_vm_constructor(0, to_obj(s.str())));
|
||
}
|
||
|
||
static vm_obj io_put_str(vm_obj const & str, vm_obj const &) {
|
||
get_global_ios().get_regular_stream() << to_string(str);
|
||
return mk_io_result(mk_vm_unit());
|
||
}
|
||
|
||
static vm_obj io_get_line(vm_obj const &) {
|
||
if (get_global_ios().get_options().get_bool("server"))
|
||
throw exception("get_line: cannot read from stdin in server mode");
|
||
std::string str;
|
||
std::getline(std::cin, str);
|
||
return mk_io_result(to_obj(str));
|
||
}
|
||
|
||
static vm_obj cmdline_args_to_obj(std::vector<std::string> const & ss) {
|
||
buffer<vm_obj> objs;
|
||
for (auto & s : ss) objs.push_back(to_obj(s));
|
||
return to_obj(objs);
|
||
}
|
||
|
||
/*
|
||
structure io.terminal (m : Type → Type → Type) :=
|
||
(put_str : string → m io.error unit)
|
||
(get_line : m io.error string)
|
||
(cmdline_args : list string)
|
||
*/
|
||
static vm_obj mk_terminal(std::vector<std::string> const & cmdline_args) {
|
||
constexpr size_t num_fields = 3;
|
||
vm_obj fields[num_fields] = {
|
||
mk_native_closure(io_put_str),
|
||
mk_native_closure(io_get_line),
|
||
cmdline_args_to_obj(cmdline_args),
|
||
};
|
||
return mk_vm_constructor(0, num_fields, fields);
|
||
}
|
||
|
||
struct vm_handle : public vm_external {
|
||
handle_ref m_handle;
|
||
vm_handle(handle_ref const & h):m_handle(h) {}
|
||
virtual ~vm_handle() {}
|
||
virtual void dealloc() override { this->~vm_handle(); get_vm_allocator().deallocate(sizeof(vm_handle), this); }
|
||
virtual vm_external * clone(vm_clone_fn const &) override { return new vm_handle(m_handle); }
|
||
virtual vm_external * ts_clone(vm_clone_fn const &) override { lean_unreachable(); }
|
||
};
|
||
|
||
bool is_handle(vm_obj const & o) {
|
||
return is_external(o) && dynamic_cast<vm_handle*>(to_external(o));
|
||
}
|
||
|
||
handle_ref const & to_handle(vm_obj const & o) {
|
||
lean_vm_check(dynamic_cast<vm_handle*>(to_external(o)));
|
||
return static_cast<vm_handle*>(to_external(o))->m_handle;
|
||
}
|
||
|
||
vm_obj to_obj(handle_ref const & h) {
|
||
return mk_vm_external(new (get_vm_allocator().allocate(sizeof(vm_handle))) vm_handle(h));
|
||
}
|
||
|
||
struct vm_child : public vm_external {
|
||
std::shared_ptr<child> m_child;
|
||
vm_child(std::shared_ptr<child> const & h):m_child(h) {}
|
||
virtual ~vm_child() {}
|
||
virtual void dealloc() override { this->~vm_child(); get_vm_allocator().deallocate(sizeof(vm_child), this); }
|
||
virtual vm_external * clone(vm_clone_fn const &) override { return new vm_child(m_child); }
|
||
virtual vm_external * ts_clone(vm_clone_fn const &) override { lean_unreachable(); }
|
||
};
|
||
|
||
bool is_child(vm_obj const & o) {
|
||
return is_external(o) && dynamic_cast<vm_child*>(to_external(o));
|
||
}
|
||
|
||
std::shared_ptr<child> const & to_child(vm_obj const & o) {
|
||
lean_vm_check(dynamic_cast<vm_child*>(to_external(o)));
|
||
return static_cast<vm_child*>(to_external(o))->m_child;
|
||
}
|
||
|
||
vm_obj to_obj(std::shared_ptr<child> const & h) {
|
||
return mk_vm_external(new (get_vm_allocator().allocate(sizeof(vm_child))) vm_child(h));
|
||
}
|
||
|
||
/*
|
||
inductive io.mode
|
||
| read | write | read_write | append
|
||
*/
|
||
char const * to_c_io_mode(vm_obj const & mode, vm_obj const & bin) {
|
||
bool is_bin = to_bool(bin);
|
||
switch (cidx(mode)) {
|
||
case 0: return is_bin ? "rb" : "r";
|
||
case 1: return is_bin ? "wb" : "w";
|
||
case 2: return is_bin ? "r+b" : "r+";
|
||
case 3: return is_bin ? "ab" : "a";
|
||
}
|
||
lean_vm_check(false);
|
||
lean_unreachable();
|
||
}
|
||
|
||
/* (mk_file_handle : string → io.mode → bool → m io.error handle) */
|
||
static vm_obj fs_mk_file_handle(vm_obj const & fname, vm_obj const & m, vm_obj const & bin, vm_obj const &) {
|
||
FILE * h = fopen(to_string(fname).c_str(), to_c_io_mode(m, bin));
|
||
if (h != nullptr)
|
||
return mk_io_result(to_obj(std::make_shared<handle>(h)));
|
||
else
|
||
return mk_io_failure(sstream() << "failed to open file '" << to_string(fname) << "'");
|
||
}
|
||
|
||
static vm_obj mk_handle_has_been_closed_error() {
|
||
return mk_io_failure("invalid io action, handle has been closed");
|
||
}
|
||
|
||
static vm_obj fs_is_eof(vm_obj const & h, vm_obj const &) {
|
||
handle_ref const & href = to_handle(h);
|
||
if (href->is_closed()) return mk_handle_has_been_closed_error();
|
||
bool r = feof(href->m_file) != 0;
|
||
return mk_io_result(mk_vm_bool(r));
|
||
}
|
||
|
||
static vm_obj fs_flush(vm_obj const & h, vm_obj const &) {
|
||
handle_ref const & href = to_handle(h);
|
||
|
||
if (href->is_closed()) {
|
||
return mk_handle_has_been_closed_error();
|
||
}
|
||
|
||
try {
|
||
href->flush();
|
||
return mk_io_result(mk_vm_unit());
|
||
} catch (handle_exception e) {
|
||
return mk_io_failure("flush failed");
|
||
}
|
||
}
|
||
|
||
static vm_obj fs_close(vm_obj const & h, vm_obj const &) {
|
||
handle_ref const & href = to_handle(h);
|
||
|
||
if (href->is_closed()) {
|
||
return mk_handle_has_been_closed_error();
|
||
}
|
||
|
||
if (href->is_stdin())
|
||
return mk_io_failure("close failed, stdin cannot be closed");
|
||
if (href->is_stdout())
|
||
return mk_io_failure("close failed, stdout cannot be closed");
|
||
if (href->is_stderr())
|
||
return mk_io_failure("close failed, stderr cannot be closed");
|
||
|
||
try {
|
||
href->close();
|
||
return mk_io_result(mk_vm_unit());
|
||
} catch (handle_exception e) {
|
||
return mk_io_failure("close failed");
|
||
}
|
||
}
|
||
|
||
static vm_obj mk_buffer(parray<vm_obj> const & a) {
|
||
return mk_vm_pair(mk_vm_nat(a.size()), to_obj(a));
|
||
}
|
||
|
||
static vm_obj fs_read(vm_obj const & h, vm_obj const & n, vm_obj const &) {
|
||
handle_ref const & href = to_handle(h);
|
||
if (href->is_closed()) return mk_handle_has_been_closed_error();
|
||
buffer<char> tmp;
|
||
unsigned num = force_to_unsigned(n); /* TODO(Leo): handle size_t */
|
||
tmp.resize(num, 0);
|
||
size_t sz = fread(tmp.data(), 1, num, href->m_file);
|
||
if (ferror(href->m_file)) {
|
||
clearerr(href->m_file);
|
||
return mk_io_failure("read failed");
|
||
}
|
||
parray<vm_obj> r;
|
||
for (size_t i = 0; i < sz; i++) {
|
||
r.push_back(mk_vm_simple(static_cast<unsigned char>(tmp[i])));
|
||
}
|
||
return mk_io_result(mk_buffer(r));
|
||
}
|
||
|
||
static vm_obj fs_write(vm_obj const & h, vm_obj const & b, vm_obj const &) {
|
||
handle_ref const & href = to_handle(h);
|
||
|
||
if (href->is_closed()) {
|
||
return mk_handle_has_been_closed_error();
|
||
}
|
||
|
||
buffer<char> tmp;
|
||
parray<vm_obj> const & a = to_array(cfield(b, 1));
|
||
unsigned sz = a.size();
|
||
for (unsigned i = 0; i < sz; i++) {
|
||
tmp.push_back(static_cast<unsigned char>(cidx(a[i])));
|
||
}
|
||
|
||
try {
|
||
href->write(tmp);
|
||
return mk_io_result(mk_vm_unit());
|
||
} catch (handle_exception e) {
|
||
return mk_io_failure("write failed");
|
||
}
|
||
}
|
||
|
||
static vm_obj fs_get_line(vm_obj const & h, vm_obj const &) {
|
||
handle_ref const & href = to_handle(h);
|
||
|
||
if (href->is_closed()) {
|
||
return mk_handle_has_been_closed_error();
|
||
}
|
||
|
||
parray<vm_obj> r;
|
||
while (true) {
|
||
int c = fgetc(href->m_file);
|
||
if (ferror(href->m_file)) {
|
||
clearerr(href->m_file);
|
||
return mk_io_failure("get_line failed");
|
||
}
|
||
if (c == EOF)
|
||
break;
|
||
r.push_back(mk_vm_simple(static_cast<unsigned char>(static_cast<char>(c))));
|
||
if (c == '\n')
|
||
break;
|
||
}
|
||
return mk_io_result(mk_buffer(r));
|
||
}
|
||
|
||
static vm_obj fs_stdin(vm_obj const &) {
|
||
return mk_io_result(to_obj(std::make_shared<handle>(stdin)));
|
||
}
|
||
|
||
static vm_obj fs_stdout(vm_obj const &) {
|
||
return mk_io_result(to_obj(std::make_shared<handle>(stdout)));
|
||
}
|
||
|
||
static vm_obj fs_stderr(vm_obj const &) {
|
||
return mk_io_result(to_obj(std::make_shared<handle>(stderr)));
|
||
}
|
||
|
||
/*
|
||
(mk_file_handle : string → io.mode → bool → m io.error handle)
|
||
(is_eof : handle → m io.error bool)
|
||
(flush : handle → m io.error unit)
|
||
(close : handle → m io.error unit)
|
||
(read : handle → nat → m io.error char_buffer)
|
||
(write : handle → char_buffer → m io.error unit)
|
||
(get_line : handle → m io.error char_buffer)
|
||
(stdin : m io.error handle)
|
||
(stdout : m io.error handle)
|
||
(stderr : m io.error handle)
|
||
*/
|
||
static vm_obj mk_fs() {
|
||
vm_obj fields[11] = {
|
||
mk_native_closure(fs_mk_file_handle),
|
||
mk_native_closure(fs_is_eof),
|
||
mk_native_closure(fs_flush),
|
||
mk_native_closure(fs_close),
|
||
mk_native_closure(fs_read),
|
||
mk_native_closure(fs_write),
|
||
mk_native_closure(fs_get_line),
|
||
mk_native_closure(fs_stdin),
|
||
mk_native_closure(fs_stdout),
|
||
mk_native_closure(fs_stderr)
|
||
};
|
||
return mk_vm_constructor(0, 11, fields);
|
||
}
|
||
|
||
stdio to_stdio(vm_obj const & o) {
|
||
switch (cidx(o)) {
|
||
case 0:
|
||
return stdio::PIPED;
|
||
case 1:
|
||
return stdio::INHERIT;
|
||
case 2:
|
||
return stdio::NUL;
|
||
default:
|
||
lean_unreachable()
|
||
}
|
||
}
|
||
|
||
/*
|
||
structure spawn_args :=
|
||
(cmd : string)
|
||
/- Add an argument to pass to the process. -/
|
||
(args : list string)
|
||
/- Configuration for the process's stdin handle. -/
|
||
(stdin := stdio.inherit)
|
||
/- Configuration for the process's stdout handle. -/
|
||
(stdout := stdio.inherit)
|
||
/- Configuration for the process's stderr handle. -/
|
||
(stderr := stdio.inherit)
|
||
(cwd := option string)
|
||
*/
|
||
static vm_obj io_process_spawn(vm_obj const & process_obj, vm_obj const &) {
|
||
std::string cmd = to_string(cfield(process_obj, 0));
|
||
|
||
list<std::string> args = to_list<std::string>(cfield(process_obj, 1), [&] (vm_obj const & o) -> std::string {
|
||
return to_string(o);
|
||
});
|
||
auto stdin_stdio = to_stdio(cfield(process_obj, 2));
|
||
auto stdout_stdio = to_stdio(cfield(process_obj, 3));
|
||
auto stderr_stdio = to_stdio(cfield(process_obj, 4));
|
||
|
||
optional<std::string> cwd;
|
||
if (!is_none(cfield(process_obj, 5)))
|
||
cwd = to_string(get_some_value(cfield(process_obj, 5)));
|
||
|
||
lean::process proc(cmd);
|
||
|
||
for (auto arg : args) {
|
||
proc.arg(arg);
|
||
}
|
||
|
||
proc.set_stdin(stdin_stdio);
|
||
proc.set_stdout(stdout_stdio);
|
||
proc.set_stderr(stderr_stdio);
|
||
|
||
if (cwd) proc.set_cwd(*cwd);
|
||
|
||
auto ch = proc.spawn();
|
||
return mk_io_result(to_obj(ch));
|
||
}
|
||
|
||
static vm_obj io_process_wait(vm_obj const & ch, vm_obj const &) {
|
||
return mk_io_result(mk_vm_nat(to_child(ch)->wait()));
|
||
}
|
||
|
||
/*
|
||
structure io.process (Err : Type) (handle : Type) (m : Type → Type → Type) :=
|
||
(child : Type)
|
||
(stdin : child -> handle)
|
||
(stdout : child -> handle)
|
||
(stderr : child -> handle)
|
||
(spawn : process → m Err child)
|
||
(wait : child -> m Err nat)
|
||
*/
|
||
static vm_obj mk_process() {
|
||
return mk_vm_constructor(0, {
|
||
mk_native_closure([] (vm_obj const & c) { return to_obj(to_child(c)->get_stdin()); }),
|
||
mk_native_closure([] (vm_obj const & c) { return to_obj(to_child(c)->get_stdout()); }),
|
||
mk_native_closure([] (vm_obj const & c) { return to_obj(to_child(c)->get_stderr()); }),
|
||
mk_native_closure(io_process_spawn),
|
||
mk_native_closure(io_process_wait),
|
||
});
|
||
}
|
||
|
||
static vm_obj io_return(vm_obj const &, vm_obj const & a, vm_obj const &) {
|
||
return mk_io_result(a);
|
||
}
|
||
|
||
static vm_obj io_bind(vm_obj const & /* α */, vm_obj const & /* β */, vm_obj const & a, vm_obj const & b, vm_obj const &) {
|
||
vm_obj r = invoke(a, mk_vm_unit());
|
||
if (cidx(r) == 0) {
|
||
vm_obj v = cfield(r, 0);
|
||
return invoke(b, v, mk_vm_unit());
|
||
} else {
|
||
return r;
|
||
}
|
||
}
|
||
|
||
static vm_obj io_monad(vm_obj const &) {
|
||
vm_state & S = get_vm_state();
|
||
vm_obj const & mk_unsafe_monad = S.get_constant(get_unsafe_monad_from_pure_bind_name());
|
||
return invoke(mk_unsafe_monad, mk_vm_simple(0), mk_native_closure(io_return), mk_native_closure(io_bind));
|
||
}
|
||
|
||
static vm_obj io_catch(vm_obj const &, vm_obj const &, vm_obj const &, vm_obj const & a, vm_obj const & b, vm_obj const &) {
|
||
vm_obj r = invoke(a, mk_vm_unit());
|
||
if (cidx(r) == 1) {
|
||
vm_obj e = cfield(r, 0);
|
||
return invoke(b, e, mk_vm_unit());
|
||
} else {
|
||
return r;
|
||
}
|
||
}
|
||
|
||
static vm_obj io_fail(vm_obj const &, vm_obj const &, vm_obj const & e, vm_obj const &) {
|
||
return mk_io_failure(e);
|
||
}
|
||
|
||
/* (iterate : Π e α, α → (α → m e (option α)) → m e α) */
|
||
static vm_obj io_iterate(vm_obj const &, vm_obj const &, vm_obj const & a, vm_obj const & fn, vm_obj const &) {
|
||
vm_obj r = a;
|
||
while (true) {
|
||
vm_obj p = invoke(fn, r, mk_vm_unit());
|
||
if (cidx(p) == 1) {
|
||
return p;
|
||
} else {
|
||
vm_obj v = cfield(p, 0);
|
||
if (is_none(v)) {
|
||
return mk_io_result(r);
|
||
} else {
|
||
r = get_some_value(v);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
static vm_obj mk_io_env() {
|
||
return mk_vm_constructor(0, {
|
||
// get_env
|
||
mk_native_closure([] (vm_obj const & k, vm_obj const &) {
|
||
if (auto v = getenv(to_string(k).c_str())) {
|
||
return mk_io_result(mk_vm_some(to_obj(std::string(v))));
|
||
} else {
|
||
return mk_io_result(mk_vm_none());
|
||
}
|
||
}),
|
||
});
|
||
}
|
||
|
||
/*
|
||
class io.interface :=
|
||
(m : Type → Type → Type)
|
||
(monad : Π e, monad (m e))
|
||
(catch : Π e₁ e₂ α, m e₁ α → (e₁ → m e₂ α) → m e₂ α)
|
||
(fail : Π e α, e → m e α)
|
||
(iterate : Π e α, α → (α → m e (option α)) → m e α)
|
||
-- Primitive Types
|
||
(handle : Type)
|
||
-- Interface Extensions
|
||
(term : io.terminal m)
|
||
(fs : io.file_system handle m)
|
||
(process : io.process io.error handle m)
|
||
(env : io.environment _)
|
||
*/
|
||
vm_obj mk_io_interface(std::vector<std::string> const & cmdline_args) {
|
||
return mk_vm_constructor(0, {
|
||
mk_native_closure(io_monad),
|
||
mk_native_closure(io_catch),
|
||
mk_native_closure(io_fail),
|
||
mk_native_closure(io_iterate),
|
||
mk_terminal(cmdline_args),
|
||
mk_fs(),
|
||
mk_process(),
|
||
mk_io_env(),
|
||
});
|
||
}
|
||
|
||
vm_obj mk_io_interface() {
|
||
return mk_io_interface({});
|
||
}
|
||
|
||
optional<vm_obj> is_io_result(vm_obj const & o) {
|
||
if (cidx(o) == 0)
|
||
return some(cfield(o, 0));
|
||
else
|
||
return optional<vm_obj>();
|
||
}
|
||
|
||
optional<vm_obj> is_io_error(vm_obj const & o) {
|
||
if (cidx(o) == 1)
|
||
return some(cfield(o, 0));
|
||
else
|
||
return optional<vm_obj>();
|
||
}
|
||
|
||
/*
|
||
inductive io.error
|
||
| other : string → io.error
|
||
| sys : nat → io.error
|
||
*/
|
||
std::string io_error_to_string(vm_obj const & o) {
|
||
if (cidx(o) == 0) {
|
||
return to_string(cfield(o, 0));
|
||
} else if (cidx(o) == 1) {
|
||
return (sstream() << "system error #" << to_unsigned(cfield(o, 0))).str();
|
||
}
|
||
lean_vm_check(false);
|
||
lean_unreachable();
|
||
}
|
||
|
||
void initialize_vm_io() {
|
||
}
|
||
|
||
void finalize_vm_io() {
|
||
}
|
||
}
|