mirror of
https://github.com/RustPython/RustPython.git
synced 2026-06-02 19:39:49 +09:00
125 lines
3.5 KiB
Rust
125 lines
3.5 KiB
Rust
/*! Infamous code object. The python class `code`
|
|
|
|
*/
|
|
|
|
use std::fmt;
|
|
|
|
use crate::bytecode;
|
|
use crate::function::PyFuncArgs;
|
|
use crate::obj::objtype::PyClassRef;
|
|
use crate::pyobject::{IdProtocol, PyContext, PyObjectRef, PyRef, PyResult, PyValue, TypeProtocol};
|
|
use crate::vm::VirtualMachine;
|
|
|
|
pub type PyCodeRef = PyRef<PyCode>;
|
|
|
|
pub struct PyCode {
|
|
code: bytecode::CodeObject,
|
|
}
|
|
|
|
impl PyCode {
|
|
pub fn new(code: bytecode::CodeObject) -> PyCode {
|
|
PyCode { code }
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for PyCode {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "code: {:?}", self.code)
|
|
}
|
|
}
|
|
|
|
impl PyValue for PyCode {
|
|
fn class(vm: &VirtualMachine) -> PyClassRef {
|
|
vm.ctx.code_type()
|
|
}
|
|
}
|
|
|
|
pub fn init(context: &PyContext) {
|
|
let code_type = context.code_type.as_object();
|
|
extend_class!(context, code_type, {
|
|
"__new__" => context.new_rustfunc(code_new),
|
|
"__repr__" => context.new_rustfunc(code_repr)
|
|
});
|
|
|
|
for (name, f) in &[
|
|
(
|
|
"co_argcount",
|
|
code_co_argcount as fn(&VirtualMachine, PyFuncArgs) -> PyResult,
|
|
),
|
|
("co_consts", code_co_consts),
|
|
("co_filename", code_co_filename),
|
|
("co_firstlineno", code_co_firstlineno),
|
|
("co_kwonlyargcount", code_co_kwonlyargcount),
|
|
("co_name", code_co_name),
|
|
] {
|
|
context.set_attr(code_type, name, context.new_property(f))
|
|
}
|
|
}
|
|
|
|
pub fn get_value(obj: &PyObjectRef) -> bytecode::CodeObject {
|
|
if let Some(code) = obj.payload::<PyCode>() {
|
|
code.code.clone()
|
|
} else {
|
|
panic!("Inner error getting code {:?}", obj)
|
|
}
|
|
}
|
|
|
|
fn code_new(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(vm, args, required = [(_cls, None)]);
|
|
Err(vm.new_type_error("Cannot directly create code object".to_string()))
|
|
}
|
|
|
|
fn code_repr(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
arg_check!(vm, args, required = [(o, Some(vm.ctx.code_type()))]);
|
|
|
|
let code = get_value(o);
|
|
let repr = format!(
|
|
"<code object {} at 0x{:x} file {:?}, line {}>",
|
|
code.obj_name,
|
|
o.get_id(),
|
|
code.source_path,
|
|
code.first_line_number
|
|
);
|
|
Ok(vm.new_str(repr))
|
|
}
|
|
|
|
fn member_code_obj(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult<bytecode::CodeObject> {
|
|
arg_check!(vm, args, required = [(zelf, Some(vm.ctx.code_type()))]);
|
|
Ok(get_value(zelf))
|
|
}
|
|
|
|
fn code_co_argcount(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
let code_obj = member_code_obj(vm, args)?;
|
|
Ok(vm.ctx.new_int(code_obj.arg_names.len()))
|
|
}
|
|
|
|
fn code_co_filename(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
let code_obj = member_code_obj(vm, args)?;
|
|
let source_path = code_obj.source_path;
|
|
Ok(vm.new_str(source_path))
|
|
}
|
|
|
|
fn code_co_firstlineno(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
let code_obj = member_code_obj(vm, args)?;
|
|
Ok(vm.ctx.new_int(code_obj.first_line_number))
|
|
}
|
|
|
|
fn code_co_kwonlyargcount(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
let code_obj = member_code_obj(vm, args)?;
|
|
Ok(vm.ctx.new_int(code_obj.kwonlyarg_names.len()))
|
|
}
|
|
|
|
fn code_co_consts(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
let code_obj = member_code_obj(vm, args)?;
|
|
let consts = code_obj
|
|
.get_constants()
|
|
.map(|x| vm.ctx.unwrap_constant(x))
|
|
.collect();
|
|
Ok(vm.ctx.new_tuple(consts))
|
|
}
|
|
|
|
fn code_co_name(vm: &VirtualMachine, args: PyFuncArgs) -> PyResult {
|
|
let code_obj = member_code_obj(vm, args)?;
|
|
Ok(vm.new_str(code_obj.obj_name))
|
|
}
|