1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
|
from .pykms import *
from enum import Enum
import os
import struct
#
# Common RGB colours
#
red = RGB(255, 0, 0)
green = RGB(0, 255, 0)
blue = RGB(0, 0, 255)
yellow = RGB(255, 255, 0)
purple = RGB(255, 0, 255)
white = RGB(255, 255, 255)
cyan = RGB(0, 255, 255)
#
# Rotation enum
#
class Rotation(int, Enum):
ROTATE_0 = 1 << 0
ROTATE_90 = 1 << 1
ROTATE_180 = 1 << 2
ROTATE_270 = 1 << 3
ROTATE_MASK = ROTATE_0 | ROTATE_90 | ROTATE_180 | ROTATE_270
REFLECT_X = 1 << 4
REFLECT_Y = 1 << 5
REFLECT_MASK = REFLECT_X | REFLECT_Y
#
# DrmObject API extensions
#
def __obj_set_prop(self, prop, value):
if self.card.has_atomic:
areq = AtomicReq(self.card)
areq.add(self, prop, value)
if areq.commit_sync() != 0:
print("commit failed")
else:
if self.set_prop_value(prop, value) != 0:
print("setting property failed")
def __obj_set_props(self, map):
if self.card.has_atomic:
areq = AtomicReq(self.card)
for key, value in map.items():
areq.add(self, key, value)
if areq.commit_sync() != 0:
print("commit failed")
else:
for propid,propval in map.items():
if self.set_prop_value(propid, propval) != 0:
print("setting property failed")
DrmObject.set_prop = __obj_set_prop
DrmObject.set_props = __obj_set_props
#
# Card API extensions
#
def __card_disable_planes(self):
areq = AtomicReq(self)
for p in self.planes:
areq.add(p, "FB_ID", 0)
areq.add(p, "CRTC_ID", 0)
if areq.commit_sync() != 0:
print("disabling planes failed")
Card.disable_planes = __card_disable_planes
class DrmEventType(Enum):
VBLANK = 0x01
FLIP_COMPLETE = 0x02
# struct drm_event {
# __u32 type;
# __u32 length;
#};
#
_drm_ev = struct.Struct("II")
#struct drm_event_vblank {
# struct drm_event base;
# __u64 user_data;
# __u32 tv_sec;
# __u32 tv_usec;
# __u32 sequence;
# __u32 reserved;
#};
_drm_ev_vbl = struct.Struct("QIIII") # Note: doesn't contain drm_event
class DrmEvent:
def __init__(self, type, seq, time, data):
self.type = type
self.seq = seq
self.time = time
self.data = data
# Return DrmEvents. Note: blocks if there's nothing to read
def __card_read_events(self):
buf = os.read(self.fd, _drm_ev_vbl.size * 20)
if len(buf) == 0:
return
if len(buf) < _drm_ev.size:
raise RuntimeError("Partial DRM event")
idx = 0
while idx < len(buf):
ev_tuple = _drm_ev.unpack_from(buf, idx)
type = DrmEventType(ev_tuple[0])
if type != DrmEventType.VBLANK and type != DrmEventType.FLIP_COMPLETE:
raise RuntimeError("Illegal DRM event type")
vbl_tuple = _drm_ev_vbl.unpack_from(buf, idx + _drm_ev.size)
seq = vbl_tuple[3]
time = vbl_tuple[1] + vbl_tuple[2] / 1000000.0;
udata = pykms.__ob_unpack_helper(vbl_tuple[0])
yield DrmEvent(type, seq, time, udata)
idx += ev_tuple[1]
Card.read_events = __card_read_events
|