diff options
| -rw-r--r-- | .gitmodules | 3 | ||||
| m--------- | ext/pybind11 | 0 | ||||
| -rw-r--r-- | py/CMakeLists.txt | 23 | ||||
| -rwxr-xr-x | py/alpha-test.py | 2 | ||||
| -rwxr-xr-x | py/db.py | 10 | ||||
| -rw-r--r-- | py/helpers.py | 11 | ||||
| -rwxr-xr-x | py/iact.py | 18 | ||||
| -rw-r--r-- | py/pykms.cpp | 155 | ||||
| -rw-r--r-- | py/pykms.i | 41 | ||||
| -rwxr-xr-x | py/run.sh | 3 | ||||
| -rwxr-xr-x | py/trans-test.py | 2 | 
11 files changed, 187 insertions, 81 deletions
| diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 0000000..c6d1083 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,3 @@ +[submodule "ext/pybind11"] +	path = ext/pybind11 +	url = https://github.com/pybind/pybind11.git diff --git a/ext/pybind11 b/ext/pybind11 new file mode 160000 +Subproject e70b2abb6dee27a2889b01f245a2a28e6fcd4b0 diff --git a/py/CMakeLists.txt b/py/CMakeLists.txt index c6341d8..ff68939 100644 --- a/py/CMakeLists.txt +++ b/py/CMakeLists.txt @@ -1,27 +1,20 @@ -set(SWIG_EXECUTABLE "swig3.0") -find_package(SWIG 3.0 REQUIRED) -include(${SWIG_USE_FILE}) -  include_directories(${LIBDRM_INCLUDE_DIRS})  link_directories(${LIBDRM_LIBRARY_DIRS})  pkg_check_modules(PYTHON python3 REQUIRED)  include_directories(${PYTHON_INCLUDE_DIRS}) -# XXX how to add these inc dirs in a proper way? -include_directories(../libkms++ ../libkmstest) - -#set(CMAKE_SWIG_FLAGS "-I../../libkms") -set(CMAKE_SWIG_FLAGS "-builtin") +if (NOT ${U_CMAKE_BUILD_TYPE} MATCHES DEBUG) +    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fvisibility=hidden") +endif() -set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wno-missing-field-initializers") +include_directories(${PROJECT_SOURCE_DIR}/ext/pybind11/include) -set_source_files_properties(pykms.i PROPERTIES CPLUSPLUS ON) -swig_add_module(pykms python pykms.i) -swig_link_libraries(pykms kms++ kmstest ${LIBDRM_LIBRARIES} ${PYTHON_LIBRARIES}) +add_library(pykms SHARED pykms.cpp) +target_link_libraries(pykms kms++ kmstest ${LIBDRM_LIBRARIES}) -# We get some "maybe-uninitialized" warnings from the generated code. I hope they are harmless. -set_source_files_properties(${swig_generated_file_fullname} PROPERTIES COMPILE_FLAGS "-Wno-maybe-uninitialized") +# Don't add a 'lib' prefix to the shared library +set_target_properties(pykms PROPERTIES PREFIX "")  file(GLOB PY_SRCS "*.py")  add_custom_target(pyextras SOURCES ${PY_SRCS}) diff --git a/py/alpha-test.py b/py/alpha-test.py index 695cd1b..99c84fb 100755 --- a/py/alpha-test.py +++ b/py/alpha-test.py @@ -15,7 +15,7 @@ mode = conn.get_default_mode()  crtc = conn.get_current_crtc()  planes = [] -for p in card.get_planes(): +for p in card.planes:      if p.supports_crtc(crtc) == False:          continue      planes.append(p) @@ -24,16 +24,16 @@ class FlipHandler(pykms.PageFlipHandlerBase):          self.front_buf = self.front_buf ^ 1          current_xpos = self.bar_xpos; -        old_xpos = (current_xpos + (fb.width() - bar_width - bar_speed)) % (fb.width() - bar_width); -        new_xpos = (current_xpos + bar_speed) % (fb.width() - bar_width); +        old_xpos = (current_xpos + (fb.width - bar_width - bar_speed)) % (fb.width - bar_width); +        new_xpos = (current_xpos + bar_speed) % (fb.width - bar_width);          self.bar_xpos = new_xpos          pykms.draw_color_bar(fb, old_xpos, new_xpos, bar_width) -        if card.has_atomic(): +        if card.has_atomic:              ctx = pykms.AtomicReq(card) -            ctx.add(crtc, "FB_ID", fb.id()) +            ctx.add(crtc, "FB_ID", fb.id)              ctx.commit(self)          else:              crtc.page_flip(fb, self) @@ -61,7 +61,7 @@ def readkey(conn, mask):      exit(0)  sel = selectors.DefaultSelector() -sel.register(card.fd(), selectors.EVENT_READ, readdrm) +sel.register(card.fd, selectors.EVENT_READ, readdrm)  sel.register(sys.stdin, selectors.EVENT_READ, readkey)  while True: diff --git a/py/helpers.py b/py/helpers.py index 15f6cfa..acb9098 100644 --- a/py/helpers.py +++ b/py/helpers.py @@ -4,13 +4,12 @@ def add_props(areq, ob, map):      for key, value in map.items():          areq.add(ob, key, value) -  def props(o):      o.refresh_props() -    map = o.get_prop_map() -    for propid in map: -        prop = o.card().get_prop(propid) -        print("%-15s %d (%#x)" % (prop.name(), map[propid], map[propid])) +    map = o.prop_map +    for propid,propval in map.items(): +        prop = o.card.get_prop(propid) +        print("%-15s %d (%#x)" % (prop.name, propval, propval))  def set_props(ob, map):      areq = pykms.AtomicReq(ob.card()) @@ -32,7 +31,7 @@ cyan = pykms.RGB(0, 255, 255)  def disable_planes(card):      areq = pykms.AtomicReq(card) -    for p in card.get_planes(): +    for p in card.planes:          areq.add(p, "FB_ID", 0)          areq.add(p, "CRTC_ID", 0) @@ -6,6 +6,7 @@ import pykms  from time import sleep  from math import sin  from math import cos +from helpers import *  card = pykms.Card() @@ -21,29 +22,22 @@ crtc = conn.get_current_crtc()  #crtc.set_mode(conn, fb, mode)  i = 0 -for p in card.get_planes(): +for p in card.planes:      globals()["plane"+str(i)] = p      i=i+1  i = 0 -for c in card.get_crtcs(): +for c in card.crtcs:      globals()["crtc"+str(i)] = c      i=i+1 -for p in crtc.get_possible_planes(): -    if p.plane_type() == 0: +for p in crtc.possible_planes: +    if p.plane_type == pykms.PlaneType.Overlay:          plane = p          break  def set_plane(x, y): -    crtc.set_plane(plane, fb, x, y, fb.width(), fb.height(), 0, 0, fb.width(), fb.height()) - -def props(o): -    o.refresh_props() -    map = o.get_prop_map() -    for propid in map: -        prop = card.get_prop(propid) -        print("%-15s %d (%#x)" % (prop.name(), map[propid], map[propid])) +    crtc.set_plane(plane, fb, x, y, fb.width, fb.height, 0, 0, fb.width, fb.height)  set_plane(0, 0) diff --git a/py/pykms.cpp b/py/pykms.cpp new file mode 100644 index 0000000..7e42e4f --- /dev/null +++ b/py/pykms.cpp @@ -0,0 +1,155 @@ +#include <pybind11/pybind11.h> +#include <pybind11/stl.h> +#include <kms++.h> +#include "kmstest.h" + +namespace py = pybind11; + +using namespace kms; +using namespace std; + +class PyPageFlipHandlerBase : PageFlipHandlerBase +{ +public: +	using PageFlipHandlerBase::PageFlipHandlerBase; + +	virtual void handle_page_flip(uint32_t frame, double time) +	{ +		PYBIND11_OVERLOAD_PURE( +					void,                /* Return type */ +					PageFlipHandlerBase, /* Parent class */ +					handle_page_flip,    /* Name of function */ +					frame, time +					); +	} +}; + +PYBIND11_PLUGIN(pykms) { +	py::module m("pykms", "kms bindings"); + +	py::class_<Card>(m, "Card") +			.def(py::init<>()) +			.def_property_readonly("fd", &Card::fd) +			.def("get_first_connected_connector", &Card::get_first_connected_connector) +			.def_property_readonly("connectors", &Card::get_connectors) +			.def_property_readonly("crtcs", &Card::get_crtcs) +			.def_property_readonly("encoders", &Card::get_encoders) +			.def_property_readonly("planes", &Card::get_planes) +			.def_property_readonly("has_atomic", &Card::has_atomic) +			.def("call_page_flip_handlers", &Card::call_page_flip_handlers) +			.def("get_prop", (Property* (Card::*)(uint32_t) const)&Card::get_prop) +			.def("get_prop", (Property* (Card::*)(const string&) const)&Card::get_prop) +			; + +	py::class_<DrmObject, DrmObject*>(m, "DrmObject") +			.def_property_readonly("id", &DrmObject::id) +			.def("refresh_props", &DrmObject::refresh_props) +			.def_property_readonly("prop_map", &DrmObject::get_prop_map) +			.def_property_readonly("card", &DrmObject::card) +			; + +	py::class_<Connector, Connector*>(m, "Connector",  py::base<DrmObject>()) +			.def_property_readonly("fullname", &Connector::fullname) +			.def("get_default_mode", &Connector::get_default_mode) +			.def("get_current_crtc", &Connector::get_current_crtc) +			.def("get_modes", &Connector::get_modes) +			.def("__repr__", [](const Connector& o) { return "<pykms.Connector " + to_string(o.id()) + ">"; }) +			; + +	py::class_<Crtc, Crtc*>(m, "Crtc",  py::base<DrmObject>()) +			.def("set_mode", &Crtc::set_mode) +			.def("page_flip", &Crtc::page_flip) +			.def("set_plane", &Crtc::set_plane) +			.def_property_readonly("possible_planes", &Crtc::get_possible_planes) +			.def("__repr__", [](const Crtc& o) { return "<pykms.Crtc " + to_string(o.id()) + ">"; }) +			; + +	py::class_<Encoder, Encoder*>(m, "Encoder",  py::base<DrmObject>()) +			; + +	py::class_<Plane, Plane*>(m, "Plane",  py::base<DrmObject>()) +			.def("supports_crtc", &Plane::supports_crtc) +			.def_property_readonly("plane_type", &Plane::plane_type) +			.def("__repr__", [](const Plane& o) { return "<pykms.Plane " + to_string(o.id()) + ">"; }) +			; + +	py::enum_<PlaneType>(m, "PlaneType") +			.value("Overlay", PlaneType::Overlay) +			.value("Primary", PlaneType::Primary) +			.value("Cursor", PlaneType::Cursor) +			; + +	py::class_<Property, Property*>(m, "Property",  py::base<DrmObject>()) +			.def_property_readonly("name", &Property::name) +			; + +	py::class_<Framebuffer>(m, "Framebuffer",  py::base<DrmObject>()) +			; + +	py::class_<DumbFramebuffer>(m, "DumbFramebuffer",  py::base<Framebuffer>()) +			.def(py::init<Card&, uint32_t, uint32_t, const string&>(), +			     py::keep_alive<1, 2>())	// Keep Card alive until this is destructed +			.def_property_readonly("width", &DumbFramebuffer::width) +			.def_property_readonly("height", &DumbFramebuffer::height) +			; + +	py::class_<Videomode>(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) +			; + +	py::class_<PyPageFlipHandlerBase>(m, "PageFlipHandlerBase") +			.alias<PageFlipHandlerBase>() +			.def(py::init<>()) +			.def("handle_page_flip", &PageFlipHandlerBase::handle_page_flip) +			; + +	py::class_<AtomicReq>(m, "AtomicReq") +			.def(py::init<Card&>(), +			     py::keep_alive<1, 2>())	// Keep Card alive until this is destructed +			.def("add", (void (AtomicReq::*)(DrmObject*, const string&, uint64_t)) &AtomicReq::add) +			.def("test", &AtomicReq::test) +			.def("commit", (int (AtomicReq::*)()) &AtomicReq::commit) +			.def("commit", (int (AtomicReq::*)(void*)) &AtomicReq::commit) +			.def("commit_sync", &AtomicReq::commit_sync) +			; + + + +	/* libkmstest */ + +	py::class_<RGB>(m, "RGB") +			.def(py::init<>()) +			.def(py::init<uint8_t, uint8_t, uint8_t&>()) +			.def(py::init<uint8_t, uint8_t, uint8_t, uint8_t&>()) +			; + +	// Use lambdas to handle IMappedFramebuffer +	m.def("draw_test_pattern", [](DumbFramebuffer& fb) { draw_test_pattern(fb); } ); +	m.def("draw_color_bar", [](DumbFramebuffer& fb, int old_xpos, int xpos, int width) { +		draw_color_bar(fb, old_xpos, xpos, width); +	} ); +	m.def("draw_rect", [](DumbFramebuffer& fb, uint32_t x, uint32_t y, uint32_t w, uint32_t h, RGB color) { +		draw_rect(fb, x, y, w, h, color); +	} ); + +	return m.ptr(); +} diff --git a/py/pykms.i b/py/pykms.i deleted file mode 100644 index cefdce7..0000000 --- a/py/pykms.i +++ /dev/null @@ -1,41 +0,0 @@ -%module(directors="1") pykms -%{ -#include "kms++.h" - -#include "kmstest.h" - -using namespace kms; -%} - -%include "std_string.i" -%include "stdint.i" -%include "std_vector.i" -%include "std_map.i" - -%feature("director") PageFlipHandlerBase; - -%include "decls.h" -%include "drmobject.h" -%include "atomicreq.h" -%include "crtc.h" -%include "card.h" -%include "property.h" -%include "framebuffer.h" -%include "dumbframebuffer.h" -%include "plane.h" -%include "connector.h" -%include "encoder.h" -%include "pagefliphandler.h" -%include "videomode.h" - -%include "color.h" -%include "kmstest.h" - -%template(ConnectorVector) std::vector<kms::Connector*>; -%template(CrtcVector) std::vector<kms::Crtc*>; -%template(EncoderVector) std::vector<kms::Encoder*>; -%template(PlaneVector) std::vector<kms::Plane*>; -%template(VideoModeVector) std::vector<kms::Videomode>; -/* for some reason uint64_t doesn't compile on 64 bit pc */ -/* %template(map_u32_u64) std::map<uint32_t, uint64_t>; */ -%template(map_u32_u64) std::map<uint32_t, unsigned long long>; diff --git a/py/run.sh b/py/run.sh new file mode 100755 index 0000000..f0ead78 --- /dev/null +++ b/py/run.sh @@ -0,0 +1,3 @@ +#!/bin/sh +PYTHONPATH=build/py/ python3 $* + diff --git a/py/trans-test.py b/py/trans-test.py index d2e6050..57855ff 100755 --- a/py/trans-test.py +++ b/py/trans-test.py @@ -15,7 +15,7 @@ mode = conn.get_default_mode()  crtc = conn.get_current_crtc()  planes = [] -for p in card.get_planes(): +for p in card.planes:      if p.supports_crtc(crtc) == False:          continue      planes.append(p) | 
