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
|
#!/usr/bin/env python
import select
import os
import fcntl
import struct
import exceptions
#struct input_event {
# struct timeval time; = {long seconds, long microseconds}
# unsigned short type;
# unsigned short code;
# unsigned int value;
#};
input_event_struct = "@llHHi"
input_event_size = struct.calcsize(input_event_struct)
EVENT_BUTTON_PRESS = 1
EVENT_RELATIVE_MOTION = 2
RELATIVE_AXES_DIAL = 7
BUTTON_MISC = 0x100
def report(x):
sys.stderr.write(x + "\n")
class PowerMate:
def __init__(self, filename = None):
self.handle = -1
if filename:
if not self.OpenDevice(filename):
raise exceptions.RuntimeError, 'Unable to find powermate'
else:
ok = 0
for d in range(0, 16):
if self.OpenDevice("/dev/input/event%d" % d):
ok = 1
break
if not ok:
raise exceptions.RuntimeError, 'Unable to find powermate'
self.poll = select.poll()
self.poll.register(self.handle, select.POLLIN)
self.event_queue = [] # queue used to reduce kernel/userspace context switching
def __del__(self):
if self.handle >= 0:
self.poll.unregister(self.handle)
os.close(self.handle)
self.handle = -1
del self.poll
def OpenDevice(self, filename):
try:
self.handle = os.open(filename, os.O_RDWR)
if self.handle < 0:
return 0
name = fcntl.ioctl(self.handle, 0x80ff4506, chr(0) * 256) # read device name
name = name.replace(chr(0), '')
if name == 'Griffin PowerMate' or name == 'Griffin SoundKnob':
fcntl.fcntl(self.handle, fcntl.F_SETFL, os.O_NDELAY)
return 1
os.close(self.handle)
self.handle = -1
return 0
except exceptions.OSError:
return 0
def WaitForEvent(self, timeout): # timeout in seconds
if len(self.event_queue) > 0:
return self.event_queue.pop(0)
if self.handle < 0:
return None
r = self.poll.poll(int(timeout*100))
if len(r) == 0:
return None
return self.GetEvent()
def GetEvent(self): # only call when descriptor is readable
if self.handle < 0:
return None
try:
data = os.read(self.handle, input_event_size * 32)
while data != '':
self.event_queue.append(struct.unpack(input_event_struct, data[0:input_event_size]))
data = data[input_event_size:]
return self.event_queue.pop(0)
except exceptions.OSError, e: # Errno 11: Resource temporarily unavailable
#if e.errno == 19: # device has been disconnected
# report("PowerMate disconnected! Urgent!");
return None
def SetLEDState(self, static_brightness, pulse_speed, pulse_table, pulse_on_sleep, pulse_on_wake):
static_brightness &= 0xff;
if pulse_speed < 0:
pulse_speed = 0
if pulse_speed > 510:
pulse_speed = 510
if pulse_table < 0:
pulse_table = 0
if pulse_table > 2:
pulse_table = 2
pulse_on_sleep = not not pulse_on_sleep # not not = convert to 0/1
pulse_on_wake = not not pulse_on_wake
magic = static_brightness | (pulse_speed << 8) | (pulse_table << 17) | (pulse_on_sleep << 19) | (pulse_on_wake << 20)
data = struct.pack(input_event_struct, 0, 0, 0x04, 0x01, magic)
os.write(self.handle, data)
|