#include #include #include namespace py = pybind11; using namespace kms; using namespace std; // Helper to convert vector to vector> template static vector> convert_vector(const vector& source) { vector> v; for (T* p : source) v.push_back(unique_ptr(p)); return v; } void init_pykmsbase(py::module& m) { py::class_(m, "Card") .def(py::init<>()) .def(py::init()) .def(py::init()) .def_property_readonly("fd", &Card::fd) .def_property_readonly("minor", &Card::dev_minor) .def_property_readonly("get_first_connected_connector", &Card::get_first_connected_connector) // XXX pybind11 can't handle vector where T is non-copyable, and complains: // RuntimeError: return_value_policy = move, but the object is neither movable nor copyable! // So we do this manually. .def_property_readonly("connectors", [](Card* self) { return convert_vector(self->get_connectors()); }) .def_property_readonly("crtcs", [](Card* self) { return convert_vector(self->get_crtcs()); }) .def_property_readonly("encoders", [](Card* self) { return convert_vector(self->get_encoders()); }) .def_property_readonly("planes", [](Card* self) { return convert_vector(self->get_planes()); }) .def_property_readonly("has_atomic", &Card::has_atomic) .def("get_prop", (Property * (Card::*)(uint32_t) const) & Card::get_prop) .def_property_readonly("version_name", &Card::version_name); ; py::class_>(m, "DrmObject") .def_property_readonly("id", &DrmObject::id) .def_property_readonly("idx", &DrmObject::idx) .def_property_readonly("card", &DrmObject::card); py::class_>(m, "DrmPropObject") .def("refresh_props", &DrmPropObject::refresh_props) .def_property_readonly("prop_map", &DrmPropObject::get_prop_map) .def("get_prop_value", (uint64_t(DrmPropObject::*)(const string&) const) & DrmPropObject::get_prop_value) .def("set_prop_value", (int (DrmPropObject::*)(const string&, uint64_t)) & DrmPropObject::set_prop_value) .def("get_prop_value_as_blob", &DrmPropObject::get_prop_value_as_blob) .def("get_prop", &DrmPropObject::get_prop) .def("has_prop", &DrmPropObject::has_prop); py::class_>(m, "Connector") .def_property_readonly("fullname", &Connector::fullname) .def("get_default_mode", &Connector::get_default_mode) .def("get_current_crtc", &Connector::get_current_crtc) .def("get_possible_crtcs", [](Connector* self) { return convert_vector(self->get_possible_crtcs()); }) .def("get_modes", &Connector::get_modes) .def("get_mode", (Videomode(Connector::*)(const string& mode) const) & Connector::get_mode) .def("get_mode", (Videomode(Connector::*)(unsigned xres, unsigned yres, float refresh, bool ilace) const) & Connector::get_mode) .def("connected", &Connector::connected) .def("__repr__", [](const Connector& o) { return ""; }) .def("refresh", &Connector::refresh); py::class_>(m, "Crtc") .def("set_mode", (int (Crtc::*)(Connector*, const Videomode&)) & Crtc::set_mode) .def("set_mode", (int (Crtc::*)(Connector*, Framebuffer&, const Videomode&)) & Crtc::set_mode) .def("disable_mode", &Crtc::disable_mode) .def( "page_flip", [](Crtc* self, Framebuffer& fb, uint32_t data) { self->page_flip(fb, (void*)(intptr_t)data); }, py::arg("fb"), py::arg("data") = 0) .def("set_plane", &Crtc::set_plane) .def_property_readonly("possible_planes", &Crtc::get_possible_planes) .def_property_readonly("primary_plane", &Crtc::get_primary_plane) .def_property_readonly("mode", &Crtc::mode) .def_property_readonly("mode_valid", &Crtc::mode_valid) .def("__repr__", [](const Crtc& o) { return ""; }) .def("refresh", &Crtc::refresh) .def("legacy_gamma_size", &Crtc::legacy_gamma_size) .def("legacy_gamma_set", &Crtc::legacy_gamma_set); py::class_>(m, "Encoder") .def("refresh", &Encoder::refresh); py::class_>(m, "Plane") .def("supports_crtc", &Plane::supports_crtc) .def_property_readonly("formats", &Plane::get_formats) .def_property_readonly("plane_type", &Plane::plane_type) .def("__repr__", [](const Plane& o) { return ""; }); py::enum_(m, "PlaneType") .value("Overlay", PlaneType::Overlay) .value("Primary", PlaneType::Primary) .value("Cursor", PlaneType::Cursor); py::class_>(m, "Property") .def_property_readonly("name", &Property::name) .def_property_readonly("enums", &Property::get_enums); py::class_(m, "Blob") .def(py::init([](Card& card, py::buffer buf) { py::buffer_info info = buf.request(); if (info.ndim != 1) throw std::runtime_error("Incompatible buffer dimension!"); return new Blob(card, info.ptr, info.size * info.itemsize); }), py::keep_alive<1, 2>()) // Keep Card alive until this is destructed .def_property_readonly("data", &Blob::data) // XXX pybind11 doesn't support a base object (DrmObject) with custom holder-type, // and a subclass with standard holder-type. // So we just copy the DrmObject members here. // Note that this means that python thinks we don't derive from DrmObject .def_property_readonly("id", &DrmObject::id) .def_property_readonly("idx", &DrmObject::idx) .def_property_readonly("card", &DrmObject::card); py::class_(m, "Framebuffer") .def_property_readonly("width", &Framebuffer::width) .def_property_readonly("height", &Framebuffer::height) .def_property_readonly("format", &Framebuffer::format) .def_property_readonly("num_planes", &Framebuffer::num_planes) .def("stride", &Framebuffer::stride) .def("size", &Framebuffer::size) .def("offset", &Framebuffer::offset) .def("fd", &Framebuffer::prime_fd) // XXX pybind11 doesn't support a base object (DrmObject) with custom holder-type, // and a subclass with standard holder-type. // So we just copy the DrmObject members here. // Note that this means that python thinks we don't derive from DrmObject .def_property_readonly("id", &DrmObject::id) .def_property_readonly("idx", &DrmObject::idx) .def_property_readonly("card", &DrmObject::card); py::class_(m, "DumbFramebuffer") .def(py::init(), py::keep_alive<1, 2>()) // Keep Card alive until this is destructed .def(py::init(), py::keep_alive<1, 2>()) // Keep Card alive until this is destructed ; py::class_(m, "DmabufFramebuffer") .def(py::init, vector, vector>(), py::keep_alive<1, 2>()) // Keep Card alive until this is destructed .def(py::init, vector, vector>(), py::keep_alive<1, 2>()) // Keep Card alive until this is destructed ; py::enum_(m, "PixelFormat") .value("Undefined", PixelFormat::Undefined) .value("NV12", PixelFormat::NV12) .value("NV21", PixelFormat::NV21) .value("NV16", PixelFormat::NV16) .value("NV61", PixelFormat::NV61) .value("YUV420", PixelFormat::YUV420) .value("YVU420", PixelFormat::YVU420) .value("YUV422", PixelFormat::YUV422) .value("YVU422", PixelFormat::YVU422) .value("YUV444", PixelFormat::YUV444) .value("YVU444", PixelFormat::YVU444) .value("UYVY", PixelFormat::UYVY) .value("YUYV", PixelFormat::YUYV) .value("YVYU", PixelFormat::YVYU) .value("VYUY", PixelFormat::VYUY) .value("XRGB8888", PixelFormat::XRGB8888) .value("XBGR8888", PixelFormat::XBGR8888) .value("RGBX8888", PixelFormat::RGBX8888) .value("BGRX8888", PixelFormat::BGRX8888) .value("ARGB8888", PixelFormat::ARGB8888) .value("ABGR8888", PixelFormat::ABGR8888) .value("RGBA8888", PixelFormat::RGBA8888) .value("BGRA8888", PixelFormat::BGRA8888) .value("RGB888", PixelFormat::RGB888) .value("BGR888", PixelFormat::BGR888) .value("RGB332", PixelFormat::RGB332) .value("RGB565", PixelFormat::RGB565) .value("BGR565", PixelFormat::BGR565) .value("XRGB4444", PixelFormat::XRGB4444) .value("XRGB1555", PixelFormat::XRGB1555) .value("ARGB4444", PixelFormat::ARGB4444) .value("ARGB1555", PixelFormat::ARGB1555) .value("XRGB2101010", PixelFormat::XRGB2101010) .value("XBGR2101010", PixelFormat::XBGR2101010) .value("RGBX1010102", PixelFormat::RGBX1010102) .value("BGRX1010102", PixelFormat::BGRX1010102) .value("ARGB2101010", PixelFormat::ARGB2101010) .value("ABGR2101010", PixelFormat::ABGR2101010) .value("RGBA1010102", PixelFormat::RGBA1010102) .value("BGRA1010102", PixelFormat::BGRA1010102); m.def("fourcc_to_pixelformat", &FourCCToPixelFormat); py::enum_(m, "SyncPolarity") .value("Undefined", SyncPolarity::Undefined) .value("Positive", SyncPolarity::Positive) .value("Negative", SyncPolarity::Negative); py::class_(m, "Videomode") .def(py::init<>()) .def_readwrite("name", &Videomode::name) .def_readwrite("clock", &Videomode::clock) .def_readwrite("hdisplay", &Videomode::hdisplay) .def_readwrite("hsync_start", &Videomode::hsync_start) .def_readwrite("hsync_end", &Videomode::hsync_end) .def_readwrite("htotal", &Videomode::htotal) .def_readwrite("vdisplay", &Videomode::vdisplay) .def_readwrite("vsync_start", &Videomode::vsync_start) .def_readwrite("vsync_end", &Videomode::vsync_end) .def_readwrite("vtotal", &Videomode::vtotal) .def_readwrite("vrefresh", &Videomode::vrefresh) .def_readwrite("flags", &Videomode::flags) .def_readwrite("type", &Videomode::type) .def("__repr__", [](const Videomode& vm) { return ""; }) .def("to_blob", &Videomode::to_blob) .def_property("hsync", &Videomode::hsync, &Videomode::set_hsync) .def_property("vsync", &Videomode::vsync, &Videomode::set_vsync) .def("to_string_short", &Videomode::to_string_short) .def("to_string_long", &Videomode::to_string_long); m.def("videomode_from_timings", &videomode_from_timings); py::class_(m, "AtomicReq") .def(py::init(), py::keep_alive<1, 2>()) // Keep Card alive until this is destructed .def("add", (void (AtomicReq::*)(DrmPropObject*, const string&, uint64_t)) & AtomicReq::add) .def("add", (void (AtomicReq::*)(DrmPropObject*, Property*, uint64_t)) & AtomicReq::add) .def("add", (void (AtomicReq::*)(DrmPropObject*, const map&)) & AtomicReq::add) .def("test", &AtomicReq::test, py::arg("allow_modeset") = false) .def( "commit", [](AtomicReq* self, uint32_t data, bool allow) { return self->commit((void*)(intptr_t)data, allow); }, py::arg("data") = 0, py::arg("allow_modeset") = false) .def("commit_sync", &AtomicReq::commit_sync, py::arg("allow_modeset") = false); }