1*f0687c8aSRaman Tenneti #include <pybind11/pybind11.h>
2*f0687c8aSRaman Tenneti #include <pybind11/stl.h>
3*f0687c8aSRaman Tenneti #include <kms++/kms++.h>
4*f0687c8aSRaman Tenneti
5*f0687c8aSRaman Tenneti namespace py = pybind11;
6*f0687c8aSRaman Tenneti
7*f0687c8aSRaman Tenneti using namespace kms;
8*f0687c8aSRaman Tenneti using namespace std;
9*f0687c8aSRaman Tenneti
10*f0687c8aSRaman Tenneti // Helper to convert vector<T*> to vector<unique_ptr<T, py::nodelete>>
11*f0687c8aSRaman Tenneti template<typename T>
convert_vector(const vector<T * > & source)12*f0687c8aSRaman Tenneti static vector<unique_ptr<T, py::nodelete>> convert_vector(const vector<T*>& source)
13*f0687c8aSRaman Tenneti {
14*f0687c8aSRaman Tenneti vector<unique_ptr<T, py::nodelete>> v;
15*f0687c8aSRaman Tenneti for (T* p : source)
16*f0687c8aSRaman Tenneti v.push_back(unique_ptr<T, py::nodelete>(p));
17*f0687c8aSRaman Tenneti return v;
18*f0687c8aSRaman Tenneti }
19*f0687c8aSRaman Tenneti
init_pykmsbase(py::module & m)20*f0687c8aSRaman Tenneti void init_pykmsbase(py::module& m)
21*f0687c8aSRaman Tenneti {
22*f0687c8aSRaman Tenneti py::class_<Card>(m, "Card")
23*f0687c8aSRaman Tenneti .def(py::init<>())
24*f0687c8aSRaman Tenneti .def(py::init<const string&>())
25*f0687c8aSRaman Tenneti .def(py::init<const string&, uint32_t>())
26*f0687c8aSRaman Tenneti .def_property_readonly("fd", &Card::fd)
27*f0687c8aSRaman Tenneti .def_property_readonly("minor", &Card::dev_minor)
28*f0687c8aSRaman Tenneti .def_property_readonly("get_first_connected_connector", &Card::get_first_connected_connector)
29*f0687c8aSRaman Tenneti
30*f0687c8aSRaman Tenneti // XXX pybind11 can't handle vector<T*> where T is non-copyable, and complains:
31*f0687c8aSRaman Tenneti // RuntimeError: return_value_policy = move, but the object is neither movable nor copyable!
32*f0687c8aSRaman Tenneti // So we do this manually.
33*f0687c8aSRaman Tenneti .def_property_readonly("connectors", [](Card* self) {
34*f0687c8aSRaman Tenneti return convert_vector(self->get_connectors());
35*f0687c8aSRaman Tenneti })
36*f0687c8aSRaman Tenneti
37*f0687c8aSRaman Tenneti .def_property_readonly("crtcs", [](Card* self) {
38*f0687c8aSRaman Tenneti return convert_vector(self->get_crtcs());
39*f0687c8aSRaman Tenneti })
40*f0687c8aSRaman Tenneti
41*f0687c8aSRaman Tenneti .def_property_readonly("encoders", [](Card* self) {
42*f0687c8aSRaman Tenneti return convert_vector(self->get_encoders());
43*f0687c8aSRaman Tenneti })
44*f0687c8aSRaman Tenneti
45*f0687c8aSRaman Tenneti .def_property_readonly("planes", [](Card* self) {
46*f0687c8aSRaman Tenneti return convert_vector(self->get_planes());
47*f0687c8aSRaman Tenneti })
48*f0687c8aSRaman Tenneti
49*f0687c8aSRaman Tenneti .def_property_readonly("has_atomic", &Card::has_atomic)
50*f0687c8aSRaman Tenneti .def("get_prop", (Property * (Card::*)(uint32_t) const) & Card::get_prop)
51*f0687c8aSRaman Tenneti
52*f0687c8aSRaman Tenneti .def_property_readonly("version_name", &Card::version_name);
53*f0687c8aSRaman Tenneti ;
54*f0687c8aSRaman Tenneti
55*f0687c8aSRaman Tenneti py::class_<DrmObject, unique_ptr<DrmObject, py::nodelete>>(m, "DrmObject")
56*f0687c8aSRaman Tenneti .def_property_readonly("id", &DrmObject::id)
57*f0687c8aSRaman Tenneti .def_property_readonly("idx", &DrmObject::idx)
58*f0687c8aSRaman Tenneti .def_property_readonly("card", &DrmObject::card);
59*f0687c8aSRaman Tenneti
60*f0687c8aSRaman Tenneti py::class_<DrmPropObject, DrmObject, unique_ptr<DrmPropObject, py::nodelete>>(m, "DrmPropObject")
61*f0687c8aSRaman Tenneti .def("refresh_props", &DrmPropObject::refresh_props)
62*f0687c8aSRaman Tenneti .def_property_readonly("prop_map", &DrmPropObject::get_prop_map)
63*f0687c8aSRaman Tenneti .def("get_prop_value", (uint64_t(DrmPropObject::*)(const string&) const) & DrmPropObject::get_prop_value)
64*f0687c8aSRaman Tenneti .def("set_prop_value", (int (DrmPropObject::*)(const string&, uint64_t)) & DrmPropObject::set_prop_value)
65*f0687c8aSRaman Tenneti .def("get_prop_value_as_blob", &DrmPropObject::get_prop_value_as_blob)
66*f0687c8aSRaman Tenneti .def("get_prop", &DrmPropObject::get_prop)
67*f0687c8aSRaman Tenneti .def("has_prop", &DrmPropObject::has_prop);
68*f0687c8aSRaman Tenneti
69*f0687c8aSRaman Tenneti py::class_<Connector, DrmPropObject, unique_ptr<Connector, py::nodelete>>(m, "Connector")
70*f0687c8aSRaman Tenneti .def_property_readonly("fullname", &Connector::fullname)
71*f0687c8aSRaman Tenneti .def("get_default_mode", &Connector::get_default_mode)
72*f0687c8aSRaman Tenneti .def("get_current_crtc", &Connector::get_current_crtc)
73*f0687c8aSRaman Tenneti .def("get_possible_crtcs", [](Connector* self) {
74*f0687c8aSRaman Tenneti return convert_vector(self->get_possible_crtcs());
75*f0687c8aSRaman Tenneti })
76*f0687c8aSRaman Tenneti .def("get_modes", &Connector::get_modes)
77*f0687c8aSRaman Tenneti .def("get_mode", (Videomode(Connector::*)(const string& mode) const) & Connector::get_mode)
78*f0687c8aSRaman Tenneti .def("get_mode", (Videomode(Connector::*)(unsigned xres, unsigned yres, float refresh, bool ilace) const) & Connector::get_mode)
79*f0687c8aSRaman Tenneti .def("connected", &Connector::connected)
80*f0687c8aSRaman Tenneti .def("__repr__", [](const Connector& o) { return "<pykms.Connector " + to_string(o.id()) + ">"; })
81*f0687c8aSRaman Tenneti .def("refresh", &Connector::refresh);
82*f0687c8aSRaman Tenneti
83*f0687c8aSRaman Tenneti py::class_<Crtc, DrmPropObject, unique_ptr<Crtc, py::nodelete>>(m, "Crtc")
84*f0687c8aSRaman Tenneti .def("set_mode", (int (Crtc::*)(Connector*, const Videomode&)) & Crtc::set_mode)
85*f0687c8aSRaman Tenneti .def("set_mode", (int (Crtc::*)(Connector*, Framebuffer&, const Videomode&)) & Crtc::set_mode)
86*f0687c8aSRaman Tenneti .def("disable_mode", &Crtc::disable_mode)
87*f0687c8aSRaman Tenneti .def(
88*f0687c8aSRaman Tenneti "page_flip",
89*f0687c8aSRaman Tenneti [](Crtc* self, Framebuffer& fb, uint32_t data) {
90*f0687c8aSRaman Tenneti self->page_flip(fb, (void*)(intptr_t)data);
91*f0687c8aSRaman Tenneti },
92*f0687c8aSRaman Tenneti py::arg("fb"), py::arg("data") = 0)
93*f0687c8aSRaman Tenneti .def("set_plane", &Crtc::set_plane)
94*f0687c8aSRaman Tenneti .def_property_readonly("possible_planes", &Crtc::get_possible_planes)
95*f0687c8aSRaman Tenneti .def_property_readonly("primary_plane", &Crtc::get_primary_plane)
96*f0687c8aSRaman Tenneti .def_property_readonly("mode", &Crtc::mode)
97*f0687c8aSRaman Tenneti .def_property_readonly("mode_valid", &Crtc::mode_valid)
98*f0687c8aSRaman Tenneti .def("__repr__", [](const Crtc& o) { return "<pykms.Crtc " + to_string(o.id()) + ">"; })
99*f0687c8aSRaman Tenneti .def("refresh", &Crtc::refresh)
100*f0687c8aSRaman Tenneti .def("legacy_gamma_size", &Crtc::legacy_gamma_size)
101*f0687c8aSRaman Tenneti .def("legacy_gamma_set", &Crtc::legacy_gamma_set);
102*f0687c8aSRaman Tenneti
103*f0687c8aSRaman Tenneti py::class_<Encoder, DrmPropObject, unique_ptr<Encoder, py::nodelete>>(m, "Encoder")
104*f0687c8aSRaman Tenneti .def("refresh", &Encoder::refresh);
105*f0687c8aSRaman Tenneti
106*f0687c8aSRaman Tenneti py::class_<Plane, DrmPropObject, unique_ptr<Plane, py::nodelete>>(m, "Plane")
107*f0687c8aSRaman Tenneti .def("supports_crtc", &Plane::supports_crtc)
108*f0687c8aSRaman Tenneti .def_property_readonly("formats", &Plane::get_formats)
109*f0687c8aSRaman Tenneti .def_property_readonly("plane_type", &Plane::plane_type)
110*f0687c8aSRaman Tenneti .def("__repr__", [](const Plane& o) { return "<pykms.Plane " + to_string(o.id()) + ">"; });
111*f0687c8aSRaman Tenneti
112*f0687c8aSRaman Tenneti py::enum_<PlaneType>(m, "PlaneType")
113*f0687c8aSRaman Tenneti .value("Overlay", PlaneType::Overlay)
114*f0687c8aSRaman Tenneti .value("Primary", PlaneType::Primary)
115*f0687c8aSRaman Tenneti .value("Cursor", PlaneType::Cursor);
116*f0687c8aSRaman Tenneti
117*f0687c8aSRaman Tenneti py::class_<Property, DrmObject, unique_ptr<Property, py::nodelete>>(m, "Property")
118*f0687c8aSRaman Tenneti .def_property_readonly("name", &Property::name)
119*f0687c8aSRaman Tenneti .def_property_readonly("enums", &Property::get_enums);
120*f0687c8aSRaman Tenneti
121*f0687c8aSRaman Tenneti py::class_<Blob>(m, "Blob")
122*f0687c8aSRaman Tenneti .def(py::init([](Card& card, py::buffer buf) {
123*f0687c8aSRaman Tenneti py::buffer_info info = buf.request();
124*f0687c8aSRaman Tenneti if (info.ndim != 1)
125*f0687c8aSRaman Tenneti throw std::runtime_error("Incompatible buffer dimension!");
126*f0687c8aSRaman Tenneti
127*f0687c8aSRaman Tenneti return new Blob(card, info.ptr, info.size * info.itemsize);
128*f0687c8aSRaman Tenneti }),
129*f0687c8aSRaman Tenneti py::keep_alive<1, 2>()) // Keep Card alive until this is destructed
130*f0687c8aSRaman Tenneti
131*f0687c8aSRaman Tenneti .def_property_readonly("data", &Blob::data)
132*f0687c8aSRaman Tenneti
133*f0687c8aSRaman Tenneti // XXX pybind11 doesn't support a base object (DrmObject) with custom holder-type,
134*f0687c8aSRaman Tenneti // and a subclass with standard holder-type.
135*f0687c8aSRaman Tenneti // So we just copy the DrmObject members here.
136*f0687c8aSRaman Tenneti // Note that this means that python thinks we don't derive from DrmObject
137*f0687c8aSRaman Tenneti .def_property_readonly("id", &DrmObject::id)
138*f0687c8aSRaman Tenneti .def_property_readonly("idx", &DrmObject::idx)
139*f0687c8aSRaman Tenneti .def_property_readonly("card", &DrmObject::card);
140*f0687c8aSRaman Tenneti
141*f0687c8aSRaman Tenneti py::class_<Framebuffer>(m, "Framebuffer")
142*f0687c8aSRaman Tenneti .def_property_readonly("width", &Framebuffer::width)
143*f0687c8aSRaman Tenneti .def_property_readonly("height", &Framebuffer::height)
144*f0687c8aSRaman Tenneti .def_property_readonly("format", &Framebuffer::format)
145*f0687c8aSRaman Tenneti .def_property_readonly("num_planes", &Framebuffer::num_planes)
146*f0687c8aSRaman Tenneti .def("stride", &Framebuffer::stride)
147*f0687c8aSRaman Tenneti .def("size", &Framebuffer::size)
148*f0687c8aSRaman Tenneti .def("offset", &Framebuffer::offset)
149*f0687c8aSRaman Tenneti .def("fd", &Framebuffer::prime_fd)
150*f0687c8aSRaman Tenneti
151*f0687c8aSRaman Tenneti // XXX pybind11 doesn't support a base object (DrmObject) with custom holder-type,
152*f0687c8aSRaman Tenneti // and a subclass with standard holder-type.
153*f0687c8aSRaman Tenneti // So we just copy the DrmObject members here.
154*f0687c8aSRaman Tenneti // Note that this means that python thinks we don't derive from DrmObject
155*f0687c8aSRaman Tenneti .def_property_readonly("id", &DrmObject::id)
156*f0687c8aSRaman Tenneti .def_property_readonly("idx", &DrmObject::idx)
157*f0687c8aSRaman Tenneti .def_property_readonly("card", &DrmObject::card);
158*f0687c8aSRaman Tenneti
159*f0687c8aSRaman Tenneti py::class_<DumbFramebuffer, Framebuffer>(m, "DumbFramebuffer")
160*f0687c8aSRaman Tenneti .def(py::init<Card&, uint32_t, uint32_t, const string&>(),
161*f0687c8aSRaman Tenneti py::keep_alive<1, 2>()) // Keep Card alive until this is destructed
162*f0687c8aSRaman Tenneti .def(py::init<Card&, uint32_t, uint32_t, PixelFormat>(),
163*f0687c8aSRaman Tenneti py::keep_alive<1, 2>()) // Keep Card alive until this is destructed
164*f0687c8aSRaman Tenneti ;
165*f0687c8aSRaman Tenneti
166*f0687c8aSRaman Tenneti py::class_<DmabufFramebuffer, Framebuffer>(m, "DmabufFramebuffer")
167*f0687c8aSRaman Tenneti .def(py::init<Card&, uint32_t, uint32_t, const string&, vector<int>, vector<uint32_t>, vector<uint32_t>>(),
168*f0687c8aSRaman Tenneti py::keep_alive<1, 2>()) // Keep Card alive until this is destructed
169*f0687c8aSRaman Tenneti .def(py::init<Card&, uint32_t, uint32_t, PixelFormat, vector<int>, vector<uint32_t>, vector<uint32_t>>(),
170*f0687c8aSRaman Tenneti py::keep_alive<1, 2>()) // Keep Card alive until this is destructed
171*f0687c8aSRaman Tenneti ;
172*f0687c8aSRaman Tenneti
173*f0687c8aSRaman Tenneti py::enum_<PixelFormat>(m, "PixelFormat")
174*f0687c8aSRaman Tenneti .value("Undefined", PixelFormat::Undefined)
175*f0687c8aSRaman Tenneti
176*f0687c8aSRaman Tenneti .value("NV12", PixelFormat::NV12)
177*f0687c8aSRaman Tenneti .value("NV21", PixelFormat::NV21)
178*f0687c8aSRaman Tenneti .value("NV16", PixelFormat::NV16)
179*f0687c8aSRaman Tenneti .value("NV61", PixelFormat::NV61)
180*f0687c8aSRaman Tenneti
181*f0687c8aSRaman Tenneti .value("YUV420", PixelFormat::YUV420)
182*f0687c8aSRaman Tenneti .value("YVU420", PixelFormat::YVU420)
183*f0687c8aSRaman Tenneti .value("YUV422", PixelFormat::YUV422)
184*f0687c8aSRaman Tenneti .value("YVU422", PixelFormat::YVU422)
185*f0687c8aSRaman Tenneti .value("YUV444", PixelFormat::YUV444)
186*f0687c8aSRaman Tenneti .value("YVU444", PixelFormat::YVU444)
187*f0687c8aSRaman Tenneti
188*f0687c8aSRaman Tenneti .value("UYVY", PixelFormat::UYVY)
189*f0687c8aSRaman Tenneti .value("YUYV", PixelFormat::YUYV)
190*f0687c8aSRaman Tenneti .value("YVYU", PixelFormat::YVYU)
191*f0687c8aSRaman Tenneti .value("VYUY", PixelFormat::VYUY)
192*f0687c8aSRaman Tenneti
193*f0687c8aSRaman Tenneti .value("XRGB8888", PixelFormat::XRGB8888)
194*f0687c8aSRaman Tenneti .value("XBGR8888", PixelFormat::XBGR8888)
195*f0687c8aSRaman Tenneti .value("RGBX8888", PixelFormat::RGBX8888)
196*f0687c8aSRaman Tenneti .value("BGRX8888", PixelFormat::BGRX8888)
197*f0687c8aSRaman Tenneti
198*f0687c8aSRaman Tenneti .value("ARGB8888", PixelFormat::ARGB8888)
199*f0687c8aSRaman Tenneti .value("ABGR8888", PixelFormat::ABGR8888)
200*f0687c8aSRaman Tenneti .value("RGBA8888", PixelFormat::RGBA8888)
201*f0687c8aSRaman Tenneti .value("BGRA8888", PixelFormat::BGRA8888)
202*f0687c8aSRaman Tenneti
203*f0687c8aSRaman Tenneti .value("RGB888", PixelFormat::RGB888)
204*f0687c8aSRaman Tenneti .value("BGR888", PixelFormat::BGR888)
205*f0687c8aSRaman Tenneti
206*f0687c8aSRaman Tenneti .value("RGB332", PixelFormat::RGB332)
207*f0687c8aSRaman Tenneti
208*f0687c8aSRaman Tenneti .value("RGB565", PixelFormat::RGB565)
209*f0687c8aSRaman Tenneti .value("BGR565", PixelFormat::BGR565)
210*f0687c8aSRaman Tenneti
211*f0687c8aSRaman Tenneti .value("XRGB4444", PixelFormat::XRGB4444)
212*f0687c8aSRaman Tenneti .value("XRGB1555", PixelFormat::XRGB1555)
213*f0687c8aSRaman Tenneti
214*f0687c8aSRaman Tenneti .value("ARGB4444", PixelFormat::ARGB4444)
215*f0687c8aSRaman Tenneti .value("ARGB1555", PixelFormat::ARGB1555)
216*f0687c8aSRaman Tenneti
217*f0687c8aSRaman Tenneti .value("XRGB2101010", PixelFormat::XRGB2101010)
218*f0687c8aSRaman Tenneti .value("XBGR2101010", PixelFormat::XBGR2101010)
219*f0687c8aSRaman Tenneti .value("RGBX1010102", PixelFormat::RGBX1010102)
220*f0687c8aSRaman Tenneti .value("BGRX1010102", PixelFormat::BGRX1010102)
221*f0687c8aSRaman Tenneti
222*f0687c8aSRaman Tenneti .value("ARGB2101010", PixelFormat::ARGB2101010)
223*f0687c8aSRaman Tenneti .value("ABGR2101010", PixelFormat::ABGR2101010)
224*f0687c8aSRaman Tenneti .value("RGBA1010102", PixelFormat::RGBA1010102)
225*f0687c8aSRaman Tenneti .value("BGRA1010102", PixelFormat::BGRA1010102);
226*f0687c8aSRaman Tenneti
227*f0687c8aSRaman Tenneti py::enum_<SyncPolarity>(m, "SyncPolarity")
228*f0687c8aSRaman Tenneti .value("Undefined", SyncPolarity::Undefined)
229*f0687c8aSRaman Tenneti .value("Positive", SyncPolarity::Positive)
230*f0687c8aSRaman Tenneti .value("Negative", SyncPolarity::Negative);
231*f0687c8aSRaman Tenneti
232*f0687c8aSRaman Tenneti py::class_<Videomode>(m, "Videomode")
233*f0687c8aSRaman Tenneti .def(py::init<>())
234*f0687c8aSRaman Tenneti
235*f0687c8aSRaman Tenneti .def_readwrite("name", &Videomode::name)
236*f0687c8aSRaman Tenneti
237*f0687c8aSRaman Tenneti .def_readwrite("clock", &Videomode::clock)
238*f0687c8aSRaman Tenneti
239*f0687c8aSRaman Tenneti .def_readwrite("hdisplay", &Videomode::hdisplay)
240*f0687c8aSRaman Tenneti .def_readwrite("hsync_start", &Videomode::hsync_start)
241*f0687c8aSRaman Tenneti .def_readwrite("hsync_end", &Videomode::hsync_end)
242*f0687c8aSRaman Tenneti .def_readwrite("htotal", &Videomode::htotal)
243*f0687c8aSRaman Tenneti
244*f0687c8aSRaman Tenneti .def_readwrite("vdisplay", &Videomode::vdisplay)
245*f0687c8aSRaman Tenneti .def_readwrite("vsync_start", &Videomode::vsync_start)
246*f0687c8aSRaman Tenneti .def_readwrite("vsync_end", &Videomode::vsync_end)
247*f0687c8aSRaman Tenneti .def_readwrite("vtotal", &Videomode::vtotal)
248*f0687c8aSRaman Tenneti
249*f0687c8aSRaman Tenneti .def_readwrite("vrefresh", &Videomode::vrefresh)
250*f0687c8aSRaman Tenneti
251*f0687c8aSRaman Tenneti .def_readwrite("flags", &Videomode::flags)
252*f0687c8aSRaman Tenneti .def_readwrite("type", &Videomode::type)
253*f0687c8aSRaman Tenneti
254*f0687c8aSRaman Tenneti .def("__repr__", [](const Videomode& vm) { return "<pykms.Videomode " + to_string(vm.hdisplay) + "x" + to_string(vm.vdisplay) + ">"; })
255*f0687c8aSRaman Tenneti
256*f0687c8aSRaman Tenneti .def("to_blob", &Videomode::to_blob)
257*f0687c8aSRaman Tenneti
258*f0687c8aSRaman Tenneti .def_property("hsync", &Videomode::hsync, &Videomode::set_hsync)
259*f0687c8aSRaman Tenneti .def_property("vsync", &Videomode::vsync, &Videomode::set_vsync)
260*f0687c8aSRaman Tenneti
261*f0687c8aSRaman Tenneti .def("to_string_short", &Videomode::to_string_short)
262*f0687c8aSRaman Tenneti .def("to_string_long", &Videomode::to_string_long);
263*f0687c8aSRaman Tenneti
264*f0687c8aSRaman Tenneti m.def("videomode_from_timings", &videomode_from_timings);
265*f0687c8aSRaman Tenneti
266*f0687c8aSRaman Tenneti py::class_<AtomicReq>(m, "AtomicReq")
267*f0687c8aSRaman Tenneti .def(py::init<Card&>(),
268*f0687c8aSRaman Tenneti py::keep_alive<1, 2>()) // Keep Card alive until this is destructed
269*f0687c8aSRaman Tenneti .def("add", (void (AtomicReq::*)(DrmPropObject*, const string&, uint64_t)) & AtomicReq::add)
270*f0687c8aSRaman Tenneti .def("add", (void (AtomicReq::*)(DrmPropObject*, Property*, uint64_t)) & AtomicReq::add)
271*f0687c8aSRaman Tenneti .def("add", (void (AtomicReq::*)(DrmPropObject*, const map<string, uint64_t>&)) & AtomicReq::add)
272*f0687c8aSRaman Tenneti .def("test", &AtomicReq::test, py::arg("allow_modeset") = false)
273*f0687c8aSRaman Tenneti .def(
274*f0687c8aSRaman Tenneti "commit",
275*f0687c8aSRaman Tenneti [](AtomicReq* self, uint32_t data, bool allow) {
276*f0687c8aSRaman Tenneti return self->commit((void*)(intptr_t)data, allow);
277*f0687c8aSRaman Tenneti },
278*f0687c8aSRaman Tenneti py::arg("data") = 0, py::arg("allow_modeset") = false)
279*f0687c8aSRaman Tenneti .def("commit_sync", &AtomicReq::commit_sync, py::arg("allow_modeset") = false);
280*f0687c8aSRaman Tenneti }
281