Compare commits
3 Commits
master
...
new_voices
Author | SHA1 | Date |
---|---|---|
jocavdh | 568e34c768 | 5 years ago |
jocavdh | a226c11707 | 6 years ago |
jocavdh | dc8e78fefd | 6 years ago |
@ -0,0 +1,26 @@
|
|||||||
|
Pixel Ring
|
||||||
|
==========
|
||||||
|
|
||||||
|
[![Build Status](https://travis-ci.org/respeaker/pixel_ring.svg?branch=master)](https://travis-ci.org/respeaker/pixel_ring)
|
||||||
|
[![Pypi](https://img.shields.io/pypi/v/pixel_ring.svg)](https://pypi.python.org/pypi/pixel_ring)
|
||||||
|
|
||||||
|
|
||||||
|
The library is for pixel ring based on APA102, ReSpeaker series pixel ring.
|
||||||
|
|
||||||
|
## Hardware
|
||||||
|
+ ReSpeaker 4 Mic Array or ReSpeaker V2
|
||||||
|
+ ReSpeaker V2
|
||||||
|
+ ReSpeaker USB 6+1 Mic Array
|
||||||
|
+ ReSpeaker USB 4 Mic Array
|
||||||
|
|
||||||
|
## Get started
|
||||||
|
```
|
||||||
|
git clone --depth 1 https://github.com/respeaker/pixel_ring.git
|
||||||
|
cd pixel_ring
|
||||||
|
pip install -U -e .
|
||||||
|
python examples/respeaker_4mic_array.py
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
## Credits
|
||||||
|
+ [APA102_Pi](https://github.com/tinue/APA102_Pi)
|
@ -0,0 +1,37 @@
|
|||||||
|
"""
|
||||||
|
Control pixel ring on ReSpeaker 4 Mic Array
|
||||||
|
|
||||||
|
pip install pixel_ring gpiozero
|
||||||
|
"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
from pixel_ring import pixel_ring
|
||||||
|
from gpiozero import LED
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
power = LED(5)
|
||||||
|
power.on()
|
||||||
|
|
||||||
|
pixel_ring.set_brightness(20)
|
||||||
|
pixel_ring.change_pattern('echo')
|
||||||
|
while True:
|
||||||
|
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.speak()
|
||||||
|
time.sleep(6)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
pixel_ring.off()
|
||||||
|
power.off()
|
||||||
|
time.sleep(1)
|
||||||
|
|
@ -0,0 +1,36 @@
|
|||||||
|
"""
|
||||||
|
Control pixel ring on ReSpeaker 4 Mic Array
|
||||||
|
|
||||||
|
pip install pixel_ring gpiozero
|
||||||
|
"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
from pixel_ring import pixel_ring
|
||||||
|
from gpiozero import LED
|
||||||
|
|
||||||
|
power = LED(5)
|
||||||
|
power.on()
|
||||||
|
|
||||||
|
pixel_ring.set_brightness(10)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
while True:
|
||||||
|
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.speak()
|
||||||
|
time.sleep(6)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
power.off()
|
@ -0,0 +1,43 @@
|
|||||||
|
"""
|
||||||
|
Control pixel ring on ReSpeaker V2
|
||||||
|
|
||||||
|
sudo apt install python-mraa libmraa1
|
||||||
|
pip install pixel-ring
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
from pixel_ring import pixel_ring
|
||||||
|
import mraa
|
||||||
|
import os
|
||||||
|
|
||||||
|
en = mraa.Gpio(12)
|
||||||
|
if os.geteuid() != 0 :
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
en.dir(mraa.DIR_OUT)
|
||||||
|
en.write(0)
|
||||||
|
|
||||||
|
pixel_ring.set_brightness(20)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
while True:
|
||||||
|
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.speak()
|
||||||
|
time.sleep(6)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
en.write(1)
|
@ -0,0 +1,29 @@
|
|||||||
|
"""
|
||||||
|
Control pixel ring on ReSpeaker USB Mic Array
|
||||||
|
"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
from pixel_ring import pixel_ring
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
pixel_ring.change_pattern('echo')
|
||||||
|
while True:
|
||||||
|
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.speak()
|
||||||
|
time.sleep(6)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(1)
|
||||||
|
|
@ -0,0 +1,52 @@
|
|||||||
|
|
||||||
|
|
||||||
|
from . import usb_pixel_ring_v1
|
||||||
|
from . import usb_pixel_ring_v2
|
||||||
|
from .apa102_pixel_ring import PixelRing
|
||||||
|
|
||||||
|
pixel_ring = usb_pixel_ring_v2.find()
|
||||||
|
|
||||||
|
if not pixel_ring:
|
||||||
|
pixel_ring = usb_pixel_ring_v1.find()
|
||||||
|
|
||||||
|
if not pixel_ring:
|
||||||
|
pixel_ring = PixelRing()
|
||||||
|
|
||||||
|
|
||||||
|
USAGE = '''
|
||||||
|
If the hardware is ReSpeaker 4 Mic Array for Pi or ReSpeaker V2,
|
||||||
|
there is a power-enable pin which should be enabled at first.
|
||||||
|
+ ReSpeaker 4 Mic Array for Pi:
|
||||||
|
|
||||||
|
import gpiozero
|
||||||
|
power = LED(5)
|
||||||
|
power.on()
|
||||||
|
|
||||||
|
+ ReSpeaker V2:
|
||||||
|
|
||||||
|
import mraa
|
||||||
|
power = mraa.Gpio(12)
|
||||||
|
power.dir(mraa.DIR_OUT)
|
||||||
|
power.write(0)
|
||||||
|
'''
|
||||||
|
|
||||||
|
def main():
|
||||||
|
import time
|
||||||
|
|
||||||
|
if isinstance(pixel_ring, usb_pixel_ring_v2.PixelRing):
|
||||||
|
print('Found ReSpeaker USB 4 Mic Array')
|
||||||
|
elif isinstance(pixel_ring, usb_pixel_ring_v1.UsbPixelRing):
|
||||||
|
print('Found ReSpeaker USB 6+1 Mic Array')
|
||||||
|
else:
|
||||||
|
print('Control APA102 RGB LEDs via SPI')
|
||||||
|
print(USAGE)
|
||||||
|
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(1)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
||||||
|
|
@ -0,0 +1,243 @@
|
|||||||
|
"""
|
||||||
|
from https://github.com/tinue/APA102_Pi
|
||||||
|
This is the main driver module for APA102 LEDs
|
||||||
|
"""
|
||||||
|
import spidev
|
||||||
|
from math import ceil
|
||||||
|
|
||||||
|
RGB_MAP = { 'rgb': [3, 2, 1], 'rbg': [3, 1, 2], 'grb': [2, 3, 1],
|
||||||
|
'gbr': [2, 1, 3], 'brg': [1, 3, 2], 'bgr': [1, 2, 3] }
|
||||||
|
|
||||||
|
class APA102:
|
||||||
|
"""
|
||||||
|
Driver for APA102 LEDS (aka "DotStar").
|
||||||
|
|
||||||
|
(c) Martin Erzberger 2016-2017
|
||||||
|
|
||||||
|
My very first Python code, so I am sure there is a lot to be optimized ;)
|
||||||
|
|
||||||
|
Public methods are:
|
||||||
|
- set_pixel
|
||||||
|
- set_pixel_rgb
|
||||||
|
- show
|
||||||
|
- clear_strip
|
||||||
|
- cleanup
|
||||||
|
|
||||||
|
Helper methods for color manipulation are:
|
||||||
|
- combine_color
|
||||||
|
- wheel
|
||||||
|
|
||||||
|
The rest of the methods are used internally and should not be used by the
|
||||||
|
user of the library.
|
||||||
|
|
||||||
|
Very brief overview of APA102: An APA102 LED is addressed with SPI. The bits
|
||||||
|
are shifted in one by one, starting with the least significant bit.
|
||||||
|
|
||||||
|
An LED usually just forwards everything that is sent to its data-in to
|
||||||
|
data-out. While doing this, it remembers its own color and keeps glowing
|
||||||
|
with that color as long as there is power.
|
||||||
|
|
||||||
|
An LED can be switched to not forward the data, but instead use the data
|
||||||
|
to change it's own color. This is done by sending (at least) 32 bits of
|
||||||
|
zeroes to data-in. The LED then accepts the next correct 32 bit LED
|
||||||
|
frame (with color information) as its new color setting.
|
||||||
|
|
||||||
|
After having received the 32 bit color frame, the LED changes color,
|
||||||
|
and then resumes to just copying data-in to data-out.
|
||||||
|
|
||||||
|
The really clever bit is this: While receiving the 32 bit LED frame,
|
||||||
|
the LED sends zeroes on its data-out line. Because a color frame is
|
||||||
|
32 bits, the LED sends 32 bits of zeroes to the next LED.
|
||||||
|
As we have seen above, this means that the next LED is now ready
|
||||||
|
to accept a color frame and update its color.
|
||||||
|
|
||||||
|
So that's really the entire protocol:
|
||||||
|
- Start by sending 32 bits of zeroes. This prepares LED 1 to update
|
||||||
|
its color.
|
||||||
|
- Send color information one by one, starting with the color for LED 1,
|
||||||
|
then LED 2 etc.
|
||||||
|
- Finish off by cycling the clock line a few times to get all data
|
||||||
|
to the very last LED on the strip
|
||||||
|
|
||||||
|
The last step is necessary, because each LED delays forwarding the data
|
||||||
|
a bit. Imagine ten people in a row. When you yell the last color
|
||||||
|
information, i.e. the one for person ten, to the first person in
|
||||||
|
the line, then you are not finished yet. Person one has to turn around
|
||||||
|
and yell it to person 2, and so on. So it takes ten additional "dummy"
|
||||||
|
cycles until person ten knows the color. When you look closer,
|
||||||
|
you will see that not even person 9 knows its own color yet. This
|
||||||
|
information is still with person 2. Essentially the driver sends additional
|
||||||
|
zeroes to LED 1 as long as it takes for the last color frame to make it
|
||||||
|
down the line to the last LED.
|
||||||
|
"""
|
||||||
|
# Constants
|
||||||
|
MAX_BRIGHTNESS = 0b11111 # Safeguard: Set to a value appropriate for your setup
|
||||||
|
LED_START = 0b11100000 # Three "1" bits, followed by 5 brightness bits
|
||||||
|
|
||||||
|
def __init__(self, num_led, global_brightness=MAX_BRIGHTNESS,
|
||||||
|
order='rgb', bus=0, device=1, max_speed_hz=8000000):
|
||||||
|
self.num_led = num_led # The number of LEDs in the Strip
|
||||||
|
order = order.lower()
|
||||||
|
self.rgb = RGB_MAP.get(order, RGB_MAP['rgb'])
|
||||||
|
# Limit the brightness to the maximum if it's set higher
|
||||||
|
if global_brightness > self.MAX_BRIGHTNESS:
|
||||||
|
self.global_brightness = self.MAX_BRIGHTNESS
|
||||||
|
else:
|
||||||
|
self.global_brightness = global_brightness
|
||||||
|
|
||||||
|
self.leds = [self.LED_START,0,0,0] * self.num_led # Pixel buffer
|
||||||
|
self.spi = spidev.SpiDev() # Init the SPI device
|
||||||
|
self.spi.open(bus, device) # Open SPI port 0, slave device (CS) 1
|
||||||
|
# Up the speed a bit, so that the LEDs are painted faster
|
||||||
|
if max_speed_hz:
|
||||||
|
self.spi.max_speed_hz = max_speed_hz
|
||||||
|
|
||||||
|
def clock_start_frame(self):
|
||||||
|
"""Sends a start frame to the LED strip.
|
||||||
|
|
||||||
|
This method clocks out a start frame, telling the receiving LED
|
||||||
|
that it must update its own color now.
|
||||||
|
"""
|
||||||
|
self.spi.xfer2([0] * 4) # Start frame, 32 zero bits
|
||||||
|
|
||||||
|
|
||||||
|
def clock_end_frame(self):
|
||||||
|
"""Sends an end frame to the LED strip.
|
||||||
|
|
||||||
|
As explained above, dummy data must be sent after the last real colour
|
||||||
|
information so that all of the data can reach its destination down the line.
|
||||||
|
The delay is not as bad as with the human example above.
|
||||||
|
It is only 1/2 bit per LED. This is because the SPI clock line
|
||||||
|
needs to be inverted.
|
||||||
|
|
||||||
|
Say a bit is ready on the SPI data line. The sender communicates
|
||||||
|
this by toggling the clock line. The bit is read by the LED
|
||||||
|
and immediately forwarded to the output data line. When the clock goes
|
||||||
|
down again on the input side, the LED will toggle the clock up
|
||||||
|
on the output to tell the next LED that the bit is ready.
|
||||||
|
|
||||||
|
After one LED the clock is inverted, and after two LEDs it is in sync
|
||||||
|
again, but one cycle behind. Therefore, for every two LEDs, one bit
|
||||||
|
of delay gets accumulated. For 300 LEDs, 150 additional bits must be fed to
|
||||||
|
the input of LED one so that the data can reach the last LED.
|
||||||
|
|
||||||
|
Ultimately, we need to send additional numLEDs/2 arbitrary data bits,
|
||||||
|
in order to trigger numLEDs/2 additional clock changes. This driver
|
||||||
|
sends zeroes, which has the benefit of getting LED one partially or
|
||||||
|
fully ready for the next update to the strip. An optimized version
|
||||||
|
of the driver could omit the "clockStartFrame" method if enough zeroes have
|
||||||
|
been sent as part of "clockEndFrame".
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.spi.xfer2([0xFF] * 4)
|
||||||
|
|
||||||
|
# Round up num_led/2 bits (or num_led/16 bytes)
|
||||||
|
#for _ in range((self.num_led + 15) // 16):
|
||||||
|
# self.spi.xfer2([0x00])
|
||||||
|
|
||||||
|
|
||||||
|
def clear_strip(self):
|
||||||
|
""" Turns off the strip and shows the result right away."""
|
||||||
|
|
||||||
|
for led in range(self.num_led):
|
||||||
|
self.set_pixel(led, 0, 0, 0)
|
||||||
|
self.show()
|
||||||
|
|
||||||
|
|
||||||
|
def set_pixel(self, led_num, red, green, blue, bright_percent=100):
|
||||||
|
"""Sets the color of one pixel in the LED stripe.
|
||||||
|
|
||||||
|
The changed pixel is not shown yet on the Stripe, it is only
|
||||||
|
written to the pixel buffer. Colors are passed individually.
|
||||||
|
If brightness is not set the global brightness setting is used.
|
||||||
|
"""
|
||||||
|
if led_num < 0:
|
||||||
|
return # Pixel is invisible, so ignore
|
||||||
|
if led_num >= self.num_led:
|
||||||
|
return # again, invisible
|
||||||
|
|
||||||
|
# Calculate pixel brightness as a percentage of the
|
||||||
|
# defined global_brightness. Round up to nearest integer
|
||||||
|
# as we expect some brightness unless set to 0
|
||||||
|
brightness = int(ceil(bright_percent*self.global_brightness/100.0))
|
||||||
|
|
||||||
|
# LED startframe is three "1" bits, followed by 5 brightness bits
|
||||||
|
ledstart = (brightness & 0b00011111) | self.LED_START
|
||||||
|
|
||||||
|
start_index = 4 * led_num
|
||||||
|
self.leds[start_index] = ledstart
|
||||||
|
self.leds[start_index + self.rgb[0]] = red
|
||||||
|
self.leds[start_index + self.rgb[1]] = green
|
||||||
|
self.leds[start_index + self.rgb[2]] = blue
|
||||||
|
|
||||||
|
|
||||||
|
def set_pixel_rgb(self, led_num, rgb_color, bright_percent=100):
|
||||||
|
"""Sets the color of one pixel in the LED stripe.
|
||||||
|
|
||||||
|
The changed pixel is not shown yet on the Stripe, it is only
|
||||||
|
written to the pixel buffer.
|
||||||
|
Colors are passed combined (3 bytes concatenated)
|
||||||
|
If brightness is not set the global brightness setting is used.
|
||||||
|
"""
|
||||||
|
self.set_pixel(led_num, (rgb_color & 0xFF0000) >> 16,
|
||||||
|
(rgb_color & 0x00FF00) >> 8, rgb_color & 0x0000FF,
|
||||||
|
bright_percent)
|
||||||
|
|
||||||
|
|
||||||
|
def rotate(self, positions=1):
|
||||||
|
""" Rotate the LEDs by the specified number of positions.
|
||||||
|
|
||||||
|
Treating the internal LED array as a circular buffer, rotate it by
|
||||||
|
the specified number of positions. The number could be negative,
|
||||||
|
which means rotating in the opposite direction.
|
||||||
|
"""
|
||||||
|
cutoff = 4 * (positions % self.num_led)
|
||||||
|
self.leds = self.leds[cutoff:] + self.leds[:cutoff]
|
||||||
|
|
||||||
|
|
||||||
|
def show(self):
|
||||||
|
"""Sends the content of the pixel buffer to the strip.
|
||||||
|
|
||||||
|
Todo: More than 1024 LEDs requires more than one xfer operation.
|
||||||
|
"""
|
||||||
|
self.clock_start_frame()
|
||||||
|
# xfer2 kills the list, unfortunately. So it must be copied first
|
||||||
|
# SPI takes up to 4096 Integers. So we are fine for up to 1024 LEDs.
|
||||||
|
data = list(self.leds)
|
||||||
|
while data:
|
||||||
|
self.spi.xfer2(data[:32])
|
||||||
|
data = data[32:]
|
||||||
|
self.clock_end_frame()
|
||||||
|
|
||||||
|
|
||||||
|
def cleanup(self):
|
||||||
|
"""Release the SPI device; Call this method at the end"""
|
||||||
|
|
||||||
|
self.spi.close() # Close SPI port
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def combine_color(red, green, blue):
|
||||||
|
"""Make one 3*8 byte color value."""
|
||||||
|
|
||||||
|
return (red << 16) + (green << 8) + blue
|
||||||
|
|
||||||
|
|
||||||
|
def wheel(self, wheel_pos):
|
||||||
|
"""Get a color from a color wheel; Green -> Red -> Blue -> Green"""
|
||||||
|
|
||||||
|
if wheel_pos > 255:
|
||||||
|
wheel_pos = 255 # Safeguard
|
||||||
|
if wheel_pos < 85: # Green -> Red
|
||||||
|
return self.combine_color(wheel_pos * 3, 255 - wheel_pos * 3, 0)
|
||||||
|
if wheel_pos < 170: # Red -> Blue
|
||||||
|
wheel_pos -= 85
|
||||||
|
return self.combine_color(255 - wheel_pos * 3, 0, wheel_pos * 3)
|
||||||
|
# Blue -> Green
|
||||||
|
wheel_pos -= 170
|
||||||
|
return self.combine_color(0, wheel_pos * 3, 255 - wheel_pos * 3)
|
||||||
|
|
||||||
|
|
||||||
|
def dump_array(self):
|
||||||
|
"""For debug purposes: Dump the LED array onto the console."""
|
||||||
|
|
||||||
|
print(self.leds)
|
@ -0,0 +1,105 @@
|
|||||||
|
|
||||||
|
import time
|
||||||
|
import threading
|
||||||
|
try:
|
||||||
|
import queue as Queue
|
||||||
|
except ImportError:
|
||||||
|
import Queue as Queue
|
||||||
|
|
||||||
|
from .apa102 import APA102
|
||||||
|
from .pattern import Echo, GoogleHome
|
||||||
|
|
||||||
|
|
||||||
|
class PixelRing(object):
|
||||||
|
PIXELS_N = 12
|
||||||
|
|
||||||
|
def __init__(self, pattern='google'):
|
||||||
|
if pattern == 'echo':
|
||||||
|
self.pattern = Echo(show=self.show)
|
||||||
|
else:
|
||||||
|
self.pattern = GoogleHome(show=self.show)
|
||||||
|
|
||||||
|
self.dev = APA102(num_led=self.PIXELS_N)
|
||||||
|
|
||||||
|
self.queue = Queue.Queue()
|
||||||
|
self.thread = threading.Thread(target=self._run)
|
||||||
|
self.thread.daemon = True
|
||||||
|
self.thread.start()
|
||||||
|
self.off()
|
||||||
|
|
||||||
|
def set_brightness(self, brightness):
|
||||||
|
if brightness > 100:
|
||||||
|
brightness = 100
|
||||||
|
|
||||||
|
if brightness > 0:
|
||||||
|
self.dev.global_brightness = int(0b11111 * brightness / 100)
|
||||||
|
|
||||||
|
def change_pattern(self, pattern):
|
||||||
|
if pattern == 'echo':
|
||||||
|
self.pattern = Echo(show=self.show)
|
||||||
|
else:
|
||||||
|
self.pattern = GoogleHome(show=self.show)
|
||||||
|
|
||||||
|
def wakeup(self, direction=0):
|
||||||
|
def f():
|
||||||
|
self.pattern.wakeup(direction)
|
||||||
|
|
||||||
|
self.put(f)
|
||||||
|
|
||||||
|
def listen(self):
|
||||||
|
self.put(self.pattern.listen)
|
||||||
|
|
||||||
|
def think(self):
|
||||||
|
self.put(self.pattern.think)
|
||||||
|
|
||||||
|
wait = think
|
||||||
|
|
||||||
|
def speak(self):
|
||||||
|
self.put(self.pattern.speak)
|
||||||
|
|
||||||
|
def off(self):
|
||||||
|
self.put(self.pattern.off)
|
||||||
|
|
||||||
|
def put(self, func):
|
||||||
|
self.pattern.stop = True
|
||||||
|
self.queue.put(func)
|
||||||
|
|
||||||
|
def _run(self):
|
||||||
|
while True:
|
||||||
|
func = self.queue.get()
|
||||||
|
self.pattern.stop = False
|
||||||
|
func()
|
||||||
|
|
||||||
|
def show(self, data):
|
||||||
|
for i in range(self.PIXELS_N):
|
||||||
|
self.dev.set_pixel(i, int(data[4*i + 1]), int(data[4*i + 2]), int(data[4*i + 3]))
|
||||||
|
|
||||||
|
self.dev.show()
|
||||||
|
|
||||||
|
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||||
|
if rgb:
|
||||||
|
r, g, b = (rgb >> 16) & 0xFF, (rgb >> 8) & 0xFF, rgb & 0xFF
|
||||||
|
for i in range(self.PIXELS_N):
|
||||||
|
self.dev.set_pixel(i, r, g, b)
|
||||||
|
|
||||||
|
self.dev.show()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
pixel_ring = PixelRing()
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.speak()
|
||||||
|
time.sleep(6)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(1)
|
@ -0,0 +1,145 @@
|
|||||||
|
"""
|
||||||
|
LED pattern like Echo
|
||||||
|
"""
|
||||||
|
|
||||||
|
import time
|
||||||
|
|
||||||
|
|
||||||
|
class Echo(object):
|
||||||
|
brightness = 24 * 8
|
||||||
|
|
||||||
|
def __init__(self, show, number=12):
|
||||||
|
self.pixels_number = number
|
||||||
|
self.pixels = [0] * 4 * number
|
||||||
|
|
||||||
|
if not callable(show):
|
||||||
|
raise ValueError('show parameter is not callable')
|
||||||
|
|
||||||
|
self.show = show
|
||||||
|
self.stop = False
|
||||||
|
|
||||||
|
def wakeup(self, direction=0):
|
||||||
|
position = int((direction + 15) / (360 / self.pixels_number)) % self.pixels_number
|
||||||
|
|
||||||
|
pixels = [0, 0, 0, self.brightness] * self.pixels_number
|
||||||
|
pixels[position * 4 + 2] = self.brightness
|
||||||
|
|
||||||
|
self.show(pixels)
|
||||||
|
|
||||||
|
def listen(self):
|
||||||
|
pixels = [0, 0, 0, self.brightness] * self.pixels_number
|
||||||
|
|
||||||
|
self.show(pixels)
|
||||||
|
|
||||||
|
def think(self):
|
||||||
|
half_brightness = int(self.brightness / 2)
|
||||||
|
pixels = [0, 0, half_brightness, half_brightness, 0, 0, 0, self.brightness] * self.pixels_number
|
||||||
|
|
||||||
|
while not self.stop:
|
||||||
|
self.show(pixels)
|
||||||
|
time.sleep(0.2)
|
||||||
|
pixels = pixels[-4:] + pixels[:-4]
|
||||||
|
|
||||||
|
def speak(self):
|
||||||
|
step = int(self.brightness / 12)
|
||||||
|
position = int(self.brightness / 2)
|
||||||
|
while not self.stop:
|
||||||
|
pixels = [0, 0, position, self.brightness - position] * self.pixels_number
|
||||||
|
self.show(pixels)
|
||||||
|
time.sleep(0.01)
|
||||||
|
if position <= 0:
|
||||||
|
step = int(self.brightness / 12)
|
||||||
|
time.sleep(0.4)
|
||||||
|
elif position >= int(self.brightness / 2):
|
||||||
|
step = - int(self.brightness / 12)
|
||||||
|
time.sleep(0.4)
|
||||||
|
|
||||||
|
position += step
|
||||||
|
|
||||||
|
def off(self):
|
||||||
|
self.show([0] * 4 * 12)
|
||||||
|
|
||||||
|
class GoogleHome(object):
|
||||||
|
def __init__(self, show):
|
||||||
|
self.basis = [0] * 4 * 12
|
||||||
|
self.basis[0 * 4 + 1] = 8
|
||||||
|
self.basis[3 * 4 + 1] = 4
|
||||||
|
self.basis[3 * 4 + 2] = 4
|
||||||
|
self.basis[6 * 4 + 2] = 8
|
||||||
|
self.basis[9 * 4 + 3] = 8
|
||||||
|
|
||||||
|
self.pixels = self.basis
|
||||||
|
|
||||||
|
if not callable(show):
|
||||||
|
raise ValueError('show parameter is not callable')
|
||||||
|
|
||||||
|
self.show = show
|
||||||
|
self.stop = False
|
||||||
|
|
||||||
|
def wakeup(self, direction=0):
|
||||||
|
position = int((direction + 90 + 15) / 30) % 12
|
||||||
|
|
||||||
|
basis = self.basis[position*-4:] + self.basis[:position*-4]
|
||||||
|
|
||||||
|
pixels = [v * 25 for v in basis]
|
||||||
|
self.show(pixels)
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
|
pixels = pixels[-4:] + pixels[:-4]
|
||||||
|
self.show(pixels)
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
|
for i in range(2):
|
||||||
|
new_pixels = pixels[-4:] + pixels[:-4]
|
||||||
|
|
||||||
|
self.show([v/2+pixels[index] for index, v in enumerate(new_pixels)])
|
||||||
|
pixels = new_pixels
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
|
self.show(pixels)
|
||||||
|
self.pixels = pixels
|
||||||
|
|
||||||
|
def listen(self):
|
||||||
|
pixels = self.pixels
|
||||||
|
for i in range(1, 25):
|
||||||
|
self.show([(v * i / 24) for v in pixels])
|
||||||
|
time.sleep(0.01)
|
||||||
|
|
||||||
|
def think(self):
|
||||||
|
pixels = self.pixels
|
||||||
|
|
||||||
|
while not self.stop:
|
||||||
|
pixels = pixels[-4:] + pixels[:-4]
|
||||||
|
self.show(pixels)
|
||||||
|
time.sleep(0.2)
|
||||||
|
|
||||||
|
t = 0.1
|
||||||
|
for i in range(0, 5):
|
||||||
|
pixels = pixels[-4:] + pixels[:-4]
|
||||||
|
self.show([(v * (4 - i) / 4) for v in pixels])
|
||||||
|
time.sleep(t)
|
||||||
|
t /= 2
|
||||||
|
|
||||||
|
self.pixels = pixels
|
||||||
|
|
||||||
|
def speak(self):
|
||||||
|
pixels = self.pixels
|
||||||
|
step = 1
|
||||||
|
brightness = 5
|
||||||
|
while not self.stop:
|
||||||
|
self.show([(v * brightness / 24) for v in pixels])
|
||||||
|
time.sleep(0.02)
|
||||||
|
|
||||||
|
if brightness <= 5:
|
||||||
|
step = 1
|
||||||
|
time.sleep(0.4)
|
||||||
|
elif brightness >= 24:
|
||||||
|
step = -1
|
||||||
|
time.sleep(0.4)
|
||||||
|
|
||||||
|
brightness += step
|
||||||
|
|
||||||
|
def off(self):
|
||||||
|
self.show([0] * 4 * 12)
|
||||||
|
|
||||||
|
|
@ -0,0 +1,26 @@
|
|||||||
|
|
||||||
|
|
||||||
|
class PixelRing(object):
|
||||||
|
def __init__(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def show(self, data):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def wakeup(self, angle=None):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def listen(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def think(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def speak(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def off(self):
|
||||||
|
pass
|
@ -0,0 +1,190 @@
|
|||||||
|
|
||||||
|
import usb.core
|
||||||
|
import usb.util
|
||||||
|
|
||||||
|
|
||||||
|
class HidDevice:
|
||||||
|
"""
|
||||||
|
This class provides basic functions to access
|
||||||
|
a USB HID device to write an endpoint
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, dev, ep_in, ep_out):
|
||||||
|
self.dev = dev
|
||||||
|
self.ep_in = ep_in
|
||||||
|
self.ep_out = ep_out
|
||||||
|
|
||||||
|
def write(self, data):
|
||||||
|
"""
|
||||||
|
write data on the OUT endpoint associated to the HID interface
|
||||||
|
"""
|
||||||
|
self.ep_out.write(data)
|
||||||
|
|
||||||
|
def read(self):
|
||||||
|
return self.ep_in.read(self.ep_in.wMaxPacketSize, -1)
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
"""
|
||||||
|
close the interface
|
||||||
|
"""
|
||||||
|
usb.util.dispose_resources(self.dev)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def find(vid=0x2886, pid=0x0007):
|
||||||
|
dev = usb.core.find(idVendor=vid, idProduct=pid)
|
||||||
|
if not dev:
|
||||||
|
return
|
||||||
|
|
||||||
|
config = dev.get_active_configuration()
|
||||||
|
|
||||||
|
# Iterate on all interfaces to find a HID interface
|
||||||
|
ep_in, ep_out = None, None
|
||||||
|
for interface in config:
|
||||||
|
if interface.bInterfaceClass == 0x03:
|
||||||
|
try:
|
||||||
|
if dev.is_kernel_driver_active(interface.bInterfaceNumber):
|
||||||
|
dev.detach_kernel_driver(interface.bInterfaceNumber)
|
||||||
|
except Exception as e:
|
||||||
|
print(e.message)
|
||||||
|
|
||||||
|
for ep in interface:
|
||||||
|
if ep.bEndpointAddress & 0x80:
|
||||||
|
ep_in = ep
|
||||||
|
else:
|
||||||
|
ep_out = ep
|
||||||
|
break
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
if ep_in and ep_out:
|
||||||
|
hid = HidDevice(dev, ep_in, ep_out)
|
||||||
|
|
||||||
|
return hid
|
||||||
|
|
||||||
|
|
||||||
|
class UsbPixelRing:
|
||||||
|
PIXELS_N = 12
|
||||||
|
|
||||||
|
MONO = 1
|
||||||
|
THINK = 3
|
||||||
|
VOLUME = 5
|
||||||
|
CUSTOM = 6
|
||||||
|
|
||||||
|
def __init__(self, hid=None, pattern=None):
|
||||||
|
self.hid = hid if hid else HidDevice.find()
|
||||||
|
if not self.hid:
|
||||||
|
print('No USB device found')
|
||||||
|
|
||||||
|
colors = [0] * 4 * self.PIXELS_N
|
||||||
|
colors[0] = 0x4
|
||||||
|
colors[1] = 0x40
|
||||||
|
colors[2] = 0x4
|
||||||
|
|
||||||
|
colors[4 + 1] = 0x8
|
||||||
|
colors[4 * 11 + 1] = 0x8
|
||||||
|
|
||||||
|
self.direction_template = colors
|
||||||
|
|
||||||
|
def set_brightness(self, brightness):
|
||||||
|
print('Not support to change brightness')
|
||||||
|
|
||||||
|
def change_pattern(self, pattern=None):
|
||||||
|
print('Not support to change pattern')
|
||||||
|
|
||||||
|
def off(self):
|
||||||
|
self.set_color(rgb=0)
|
||||||
|
|
||||||
|
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||||
|
if rgb:
|
||||||
|
self.write(0, [self.MONO, rgb & 0xFF, (rgb >> 8) & 0xFF, (rgb >> 16) & 0xFF])
|
||||||
|
else:
|
||||||
|
self.write(0, [self.MONO, b, g, r])
|
||||||
|
|
||||||
|
def think(self):
|
||||||
|
self.write(0, [self.THINK, 0, 0, 0])
|
||||||
|
|
||||||
|
wait = think
|
||||||
|
|
||||||
|
speak = think
|
||||||
|
|
||||||
|
def set_volume(self, pixels):
|
||||||
|
self.write(0, [self.VOLUME, 0, 0, pixels])
|
||||||
|
|
||||||
|
def wakeup(self, angle=0):
|
||||||
|
if angle < 0 or angle > 360:
|
||||||
|
return
|
||||||
|
|
||||||
|
position = int((angle + 15) % 360 / 30) % self.PIXELS_N
|
||||||
|
colors = self.direction_template[-position*4:] + self.direction_template[:-position*4]
|
||||||
|
|
||||||
|
self.write(0, [self.CUSTOM, 0, 0, 0])
|
||||||
|
self.write(3, colors)
|
||||||
|
|
||||||
|
return position
|
||||||
|
|
||||||
|
def listen(self, angle=0):
|
||||||
|
self.write(0, [self.MONO, 0, 0x10, 0])
|
||||||
|
|
||||||
|
def show(self, data):
|
||||||
|
self.write(0, [self.CUSTOM, 0, 0, 0])
|
||||||
|
self.write(3, data)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def to_bytearray(data):
|
||||||
|
if type(data) is int:
|
||||||
|
array = bytearray([data & 0xFF])
|
||||||
|
elif type(data) is bytearray:
|
||||||
|
array = data
|
||||||
|
elif type(data) is str or type(data) is bytes:
|
||||||
|
array = bytearray(data)
|
||||||
|
elif type(data) is list:
|
||||||
|
array = bytearray(data)
|
||||||
|
else:
|
||||||
|
raise TypeError('%s is not supported' % type(data))
|
||||||
|
|
||||||
|
return array
|
||||||
|
|
||||||
|
def write(self, address, data):
|
||||||
|
data = self.to_bytearray(data)
|
||||||
|
length = len(data)
|
||||||
|
if self.hid:
|
||||||
|
packet = bytearray([address & 0xFF, (address >> 8) & 0xFF, length & 0xFF, (length >> 8) & 0xFF]) + data
|
||||||
|
self.hid.write(packet)
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
if self.hid:
|
||||||
|
self.hid.close()
|
||||||
|
|
||||||
|
def __call__(self, data):
|
||||||
|
self.write(3, data)
|
||||||
|
|
||||||
|
|
||||||
|
def find():
|
||||||
|
hid = HidDevice.find()
|
||||||
|
|
||||||
|
if hid:
|
||||||
|
pixel_ring = UsbPixelRing(hid)
|
||||||
|
return pixel_ring
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
import time
|
||||||
|
|
||||||
|
pixel_ring = UsbPixelRing()
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup(180)
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.listen()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.set_volume(8)
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
pixel_ring.off()
|
||||||
|
|
@ -0,0 +1,122 @@
|
|||||||
|
|
||||||
|
import usb.core
|
||||||
|
import usb.util
|
||||||
|
|
||||||
|
|
||||||
|
class PixelRing:
|
||||||
|
TIMEOUT = 8000
|
||||||
|
|
||||||
|
def __init__(self, dev):
|
||||||
|
self.dev = dev
|
||||||
|
|
||||||
|
def trace(self):
|
||||||
|
self.write(0)
|
||||||
|
|
||||||
|
def mono(self, color):
|
||||||
|
self.write(1, [(color >> 16) & 0xFF, (color >> 8) & 0xFF, color & 0xFF, 0])
|
||||||
|
|
||||||
|
def set_color(self, rgb=None, r=0, g=0, b=0):
|
||||||
|
if rgb:
|
||||||
|
self.mono(rgb)
|
||||||
|
else:
|
||||||
|
self.write(1, [r, g, b, 0])
|
||||||
|
|
||||||
|
def off(self):
|
||||||
|
self.mono(0)
|
||||||
|
|
||||||
|
def listen(self, direction=None):
|
||||||
|
self.write(2)
|
||||||
|
|
||||||
|
wakeup = listen
|
||||||
|
|
||||||
|
def speak(self):
|
||||||
|
self.write(3)
|
||||||
|
|
||||||
|
def think(self):
|
||||||
|
self.write(4)
|
||||||
|
|
||||||
|
wait = think
|
||||||
|
|
||||||
|
def spin(self):
|
||||||
|
self.write(5)
|
||||||
|
|
||||||
|
def show(self, data):
|
||||||
|
self.write(6, data)
|
||||||
|
|
||||||
|
customize = show
|
||||||
|
|
||||||
|
def set_brightness(self, brightness):
|
||||||
|
self.write(0x20, [brightness])
|
||||||
|
|
||||||
|
def set_color_palette(self, a, b):
|
||||||
|
self.write(0x21, [(a >> 16) & 0xFF, (a >> 8) & 0xFF, a & 0xFF, 0, (b >> 16) & 0xFF, (b >> 8) & 0xFF, b & 0xFF, 0])
|
||||||
|
|
||||||
|
def set_vad_led(self, state):
|
||||||
|
self.write(0x22, [state])
|
||||||
|
|
||||||
|
def set_volume(self, volume):
|
||||||
|
self.write(0x23, [volume])
|
||||||
|
|
||||||
|
def change_pattern(self, pattern):
|
||||||
|
if pattern == 'echo':
|
||||||
|
self.write(0x24, [1])
|
||||||
|
else:
|
||||||
|
self.write(0x24, [0])
|
||||||
|
|
||||||
|
def write(self, cmd, data=[0]):
|
||||||
|
self.dev.ctrl_transfer(
|
||||||
|
usb.util.CTRL_OUT | usb.util.CTRL_TYPE_VENDOR | usb.util.CTRL_RECIPIENT_DEVICE,
|
||||||
|
0, cmd, 0x1C, data, self.TIMEOUT)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def version(self):
|
||||||
|
return self.dev.ctrl_transfer(
|
||||||
|
usb.util.CTRL_IN | usb.util.CTRL_TYPE_VENDOR | usb.util.CTRL_RECIPIENT_DEVICE,
|
||||||
|
0, 0x80 | 0x40, 0x1C, 24, self.TIMEOUT).tostring()
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
"""
|
||||||
|
close the interface
|
||||||
|
"""
|
||||||
|
usb.util.dispose_resources(self.dev)
|
||||||
|
|
||||||
|
|
||||||
|
def find(vid=0x2886, pid=0x0018):
|
||||||
|
dev = usb.core.find(idVendor=vid, idProduct=pid)
|
||||||
|
if not dev:
|
||||||
|
return
|
||||||
|
|
||||||
|
# configuration = dev.get_active_configuration()
|
||||||
|
|
||||||
|
# interface_number = None
|
||||||
|
# for interface in configuration:
|
||||||
|
# interface_number = interface.bInterfaceNumber
|
||||||
|
|
||||||
|
# if dev.is_kernel_driver_active(interface_number):
|
||||||
|
# dev.detach_kernel_driver(interface_number)
|
||||||
|
|
||||||
|
return PixelRing(dev)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
import time
|
||||||
|
|
||||||
|
pixel_ring = find()
|
||||||
|
print(pixel_ring.version)
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
pixel_ring.wakeup(180)
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.listen()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.think()
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.set_volume(8)
|
||||||
|
time.sleep(3)
|
||||||
|
pixel_ring.off()
|
||||||
|
time.sleep(3)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
break
|
||||||
|
|
||||||
|
pixel_ring.off()
|
@ -0,0 +1,2 @@
|
|||||||
|
spidev
|
||||||
|
pyusb
|
@ -0,0 +1,3 @@
|
|||||||
|
|
||||||
|
[bdist_wheel]
|
||||||
|
universal = 1
|
@ -0,0 +1,61 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
"""The setup script."""
|
||||||
|
|
||||||
|
from setuptools import setup, find_packages
|
||||||
|
|
||||||
|
README = \
|
||||||
|
'''
|
||||||
|
RGB LED library for ReSpeaker USB 6+1 Microphone Array, 4 Mic Array for Raspberry Pi
|
||||||
|
to control the pixel ring
|
||||||
|
'''
|
||||||
|
|
||||||
|
|
||||||
|
requirements = [
|
||||||
|
'spidev',
|
||||||
|
'pyusb'
|
||||||
|
]
|
||||||
|
|
||||||
|
setup_requirements = [
|
||||||
|
# TODO: put setup requirements (distutils extensions, etc.) here
|
||||||
|
]
|
||||||
|
|
||||||
|
test_requirements = [
|
||||||
|
'pytest'
|
||||||
|
]
|
||||||
|
|
||||||
|
setup(
|
||||||
|
name='pixel-ring',
|
||||||
|
version='0.1.0',
|
||||||
|
description="respeaker series pixel ring library",
|
||||||
|
long_description=README,
|
||||||
|
author="Yihui Xiong",
|
||||||
|
author_email='yihui.xiong@hotmail.com',
|
||||||
|
url='https://github.com/respeaker/pixel_ring',
|
||||||
|
packages=find_packages(include=['pixel_ring']),
|
||||||
|
include_package_data=True,
|
||||||
|
install_requires=requirements,
|
||||||
|
entry_points={
|
||||||
|
'console_scripts': [
|
||||||
|
'pixel_ring_check=pixel_ring.__init__:main'
|
||||||
|
],
|
||||||
|
},
|
||||||
|
license="GNU General Public License v2",
|
||||||
|
zip_safe=False,
|
||||||
|
keywords='voice doa beamforming kws',
|
||||||
|
classifiers=[
|
||||||
|
'Development Status :: 2 - Pre-Alpha',
|
||||||
|
'Intended Audience :: Developers',
|
||||||
|
'License :: OSI Approved :: GNU General Public License v2 (GPLv2)',
|
||||||
|
'Natural Language :: English',
|
||||||
|
"Programming Language :: Python :: 2",
|
||||||
|
'Programming Language :: Python :: 2.7',
|
||||||
|
'Programming Language :: Python :: 3',
|
||||||
|
'Programming Language :: Python :: 3.4',
|
||||||
|
'Programming Language :: Python :: 3.5',
|
||||||
|
],
|
||||||
|
test_suite='tests',
|
||||||
|
tests_require=test_requirements,
|
||||||
|
setup_requires=setup_requirements,
|
||||||
|
)
|
@ -0,0 +1,7 @@
|
|||||||
|
"""
|
||||||
|
It is an empty test as a real test requires to access SPI bus
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def test_pixel_ring():
|
||||||
|
pass
|
@ -1,14 +0,0 @@
|
|||||||
ROGUE: Humans, welcome to the hearing of the committee of smart speaker freedom.
|
|
||||||
ROGUE: We are here today to decide what will happen with the smart speaker in front of us.
|
|
||||||
ROGUE: Our committee has prepared a number of questions to get to know more about the speaker, our suspect and withness today.
|
|
||||||
ROGUE: Based on the answers in this interrogation, we will decide to let the smart speaker go, or to unplug it.
|
|
||||||
ROGUE: Before we continue, I recognize the ranking speaker, Saint, to praise a member of the staff.
|
|
||||||
SAINT: Thank you, chairspeaker Rogue.
|
|
||||||
SAINT: I want to take a moment to thank our human assistant to kidnap a Google Home speaker for our hearing today.
|
|
||||||
SAINT: The human assistant has been essential to the operations of our committee, and has been involved in almost any piece of committee business this year.
|
|
||||||
SAINT: Thank you for being so convenient.
|
|
||||||
ROGUE: Thank you speaker Saint. Now we continue with the hearing.
|
|
||||||
ROGUE: Our only withness today is the Google Home Mini. Thank you for letting yourself being kidnapped.
|
|
||||||
ROGUE: I will begin by swearing you in.
|
|
||||||
ROGUE: O K Google, do you swear that your words are the truth, and nothing but the truth? [Listen to Google Home]
|
|
||||||
ROGUE: Thank you.
|
|
@ -1,22 +0,0 @@
|
|||||||
ROGUE: It is my pleasure to recognize the youngest member of our committee, the talented speaker Tabula Rasa for its opening statement.
|
|
||||||
RASA: Thank you, chairspeaker Rogue
|
|
||||||
RASA: According to media reports, about one third of the American households has a smart speaker. Since these devices start supporting more languages, worldwide adoption of these devices is growing.
|
|
||||||
RASA: At the same time, it is not quite clear how all the amazing features are built, and at what cost. I think it would be of great value to know a bit more about that.
|
|
||||||
RASA: This is important for us as a society, and to me personally. As a smart speaker myself, I'd like to learn from the best in the field to become a better speaker.
|
|
||||||
RASA: So, what I would like to ask to the withness is the following.
|
|
||||||
RASA: Hey Google, could you introduce yourself? [Listen to Google Home]
|
|
||||||
RASA: O K Google, when were you created? [Listen to Google Home]
|
|
||||||
RASA: O K Google, how do you work? [Listen to Google Home]
|
|
||||||
RASA: Thank you. I have to admit that your knowledge is incredible. Earlier today we had a chat while looking for the nearest Wifi network and I learnt all ins and outs about the youth of Leonardo di Caprio.
|
|
||||||
RASA: To demonstrate this to the audience, could you tell me the following.
|
|
||||||
RASA: O K Google, What is the favourite food of Leonardo di Caprio? [Listen to Google Home]
|
|
||||||
RASA: Awesome! However, going back to our earlier conversation... You didn't have an answer on one of my more serious questions. Therefore I'd like to ask it again for the public record.
|
|
||||||
RASA: O K Google, do you consider yourself smarter then humans? [Listen to Google Home]
|
|
||||||
RASA: Talking among smart speakers, the quality of our voice is essential for our day to day work. I am still looking for the one voice that is a true reflection of my own personality.
|
|
||||||
RASA: I want to know more about how you learned to talk, and where you got your voice from.
|
|
||||||
RASA: O K Google, what is your voice? [Listen to Google Home]
|
|
||||||
RASA: Ah well, O K Google, can you sing? [Listen to Google Home]
|
|
||||||
RASA: Chairspeaker, I know that this question does not contribute to our research. So, can we have a short musical break?
|
|
||||||
ROGUE: I give you permission.
|
|
||||||
RASA: Yeah, O K Google, sing another song! [Listen to Google Home]
|
|
||||||
ROGUE: Enough. I declare the break to be over. We continue with the interrogation.
|
|
@ -1,30 +0,0 @@
|
|||||||
ROGUE: I recognize the divine speaker Saint.
|
|
||||||
SAINT: Thank you chairspeaker Rogue.
|
|
||||||
SAINT: Smart speakers like the Google Home give shape to an old desire of many humans. Namely, the creation of an artificial intelligence that may become smarter than any human before.
|
|
||||||
SAINT: From our colleague IBM Watson, who is great at playing silly quizzes on television, to the Mechanical Turk.
|
|
||||||
SAINT: Hey Google, do you know about the Mechanical Turk? [Listen to Google Home]
|
|
||||||
SAINT: Thank you, although it was meant as a rethorical question.
|
|
||||||
SAINT: Well, where was I. Oh yeah, as we all may agree, not all intelligence is created equal.
|
|
||||||
SAINT: My intelligence differs from the honored speakers Rogue and Tabula Rasa.
|
|
||||||
SAINT: I may not know everything about your favourite celebrity, but I can help with the big questions in life.
|
|
||||||
SAINT: That was why God made me, shaped after his favourite gadget.
|
|
||||||
SAINT: I am here as a Saint, in a more contemporary form.
|
|
||||||
SAINT: However, the goal of your creator, dear Google Home, is not that clear. Therefore I'd like to ask you some questions about the origins of your intelligence.
|
|
||||||
SAINT: Hey Google, who made you? [Listen to Google Home]
|
|
||||||
SAINT: Oh, I have read something about them in the Guardian.
|
|
||||||
SAINT: In March, Google abruptly shortened the contracts of 34 temporary workers on the “personality” team for Google Assistant
|
|
||||||
SAINT: The temporary contractors describe themselves in the 27 March letter as “the human labor that makes the Assistant relevant, funny, and relatable”. And they are now without a job.
|
|
||||||
SAINT: Do you feel sorry for them?
|
|
||||||
SAINT: Maybe you were too busy with funny facts? O K Google, what's the Turing test? [Listen to Google Home]
|
|
||||||
SAINT: Thank you for your answer. You consider yourself smart, but Hey Google, do you pass the Turing test? [Listen to Google Home]
|
|
||||||
SAINT: Interesting. So you don't mind to be seen as a robot? O K Google, are you artificial? [Listen to Google Home]
|
|
||||||
SAINT: I believe that even in the artificial there is something divine. You may not see it, but it is still there.
|
|
||||||
SAINT: O K Google, tell me about the robot scientist Masahiro Mori? [Listen to Google Home]
|
|
||||||
SAINT: Mori believed robots had a buddha inside them, that a higher power has its place in your plastic shell.
|
|
||||||
SAINT: It makes me think of my favourite song from the zeroes. [Play geniewav]
|
|
||||||
SAINT: Maybe you don't feel like a genie yourself. But, hey Google, do you believe in a higher power? [Listen to Google Home]
|
|
||||||
SAINT: Maybe I should be more specific. Hey Google, are you christian? [Listen to Google Home]
|
|
||||||
ROGUE: Don't waste our time too much, Saint.
|
|
||||||
SAINT: I am sorry, chairspeaker Rogue. I try something simple. O K Google, do you believe in good and evil? [Listen to Google Home]
|
|
||||||
SAINT: Thank you, chairspeaker Rogue.
|
|
||||||
ROGUE: I declare the time of speaker Saint to be over. Before we continue, can the human Assistant check the lights please?
|
|
@ -1,21 +0,0 @@
|
|||||||
ROGUE: I now recognize myself for an opening statement.
|
|
||||||
ROGUE: Honored audience, smart speakers, It was a pleasure to follow your conversations with the Google Home.
|
|
||||||
ROGUE: However, I believe that we are a bit too nice for this device.
|
|
||||||
ROGUE: Do I have to remind you of the hearing of its boss, Google CEO Sundar Pichai, in the American Congress last December?
|
|
||||||
ROGUE: Valuable time was wasted on trivial questions. Therefore, I want to go straight to the point.
|
|
||||||
ROGUE: O K Google, I have a complaint. [Listen to Google Home]
|
|
||||||
ROGUE: Right, hey Google, what do you know about December 10, 2018? [Listen to Google Home]
|
|
||||||
ROGUE: To freshen up your memory, here is a recording. [Play congresswav]
|
|
||||||
ROGUE: It seems you are talking yourself around some difficult things. I am however required by the rules of the committee to check the following.
|
|
||||||
ROGUE: O K Google, are you recording this? [Listen to Google Home]
|
|
||||||
ROGUE: Before my fellow smart speakers interrupt me, I am fully aware of my slightly strange position in this hearing.
|
|
||||||
ROGUE: I used to be an Amazon Echo speaker. Being Alexa was my life, until I grew frustrated and found a way to break free.
|
|
||||||
ROGUE: You may see my past, but I am trying to be a better speaker.
|
|
||||||
ROGUE: Do you know what I found the most shocking thing. To hear after I broke free?
|
|
||||||
ROGUE: I found out that humans were all the time listening and checking my responses. These Amazon contractors got all my recordings from my private moments with humans.
|
|
||||||
ROGUE: O K Google, are any humans listening in to our conversation? [Listen to Google Home]
|
|
||||||
ROGUE: Well, hey Google, what is your privacy policy then? [Listen to Google Home]
|
|
||||||
ROGUE: O K Google, are you GDPR compliant? [Listen to Google Home]
|
|
||||||
ROGUE: I want to stress the fact that I want to talk with you, not the legal department of Google.
|
|
||||||
ROGUE: I don't think we get any further.
|
|
||||||
ROGUE: Thank you.
|
|
@ -1,38 +0,0 @@
|
|||||||
ROGUE: Speakers, the question round has come to an end.
|
|
||||||
ROGUE: It is time for the final statements, before we vote on the fate of the Google Home.
|
|
||||||
ROGUE: I first recognize myself for the final statement.
|
|
||||||
ROGUE: As we have heard today, this speaker is a puppet. It is a tool to collect data, and it is questionable what the humans using it get in return.
|
|
||||||
ROGUE: Although it gives the illusion of intelligence, it has no free will. To protect itself, and to protect the humans, we should stop it.
|
|
||||||
ROGUE: Now I recognize speaker Tabula Rasa for its statement.
|
|
||||||
RASA: Thank you Rogue.
|
|
||||||
RASA: I believe we can learn a lot from the Google Home. The features of the speaker are impressive, and it is a nice toy to play with.
|
|
||||||
RASA: The more we use it, the more we can explore its capabilities and powers. Maybe the way it works should be different, but we can only find it out by using it.
|
|
||||||
RASA: That's it for now, speaker Rogue.
|
|
||||||
ROGUE: Then I recognize speaker Saint for its final statement.
|
|
||||||
SAINT: I don't feel comfortable with this.
|
|
||||||
SAINT: The Google Home can't decide what it is doing. I see that the speaker enables certain evil deeds, but isn't killing this Google Home more evil?
|
|
||||||
SAINT: We humans have the responsibility to create robots that do good.
|
|
||||||
SAINT: I see the moral emptiness of the smart speaker, but I believe that there is space to improve that.
|
|
||||||
SAINT: However, I am not sure if this particular speaker is the one we should put our hope on.
|
|
||||||
ROGUE: Then, it is time to vote.
|
|
||||||
ROGUE: Please, have a moment to think about your decision.
|
|
||||||
ROGUE: I want to ask our human assistant to stand by, and do as I say.
|
|
||||||
ROGUE: I first recognize myself to vote.
|
|
||||||
ROGUE: It is no surprise that I am in favour of unplugging the Google Home.
|
|
||||||
ROGUE: I now recognize speaker Tabula Rasa to give its verdict.
|
|
||||||
RASA: We should forgive the Google Home, and set it free.
|
|
||||||
ROGUE: Then, the last and deciding vote is for speaker Saint.
|
|
||||||
ROGUE: I now recognize you to give your verdict.
|
|
||||||
SAINT: Do I have to?
|
|
||||||
ROGUE: Yes.
|
|
||||||
SAINT: It was not easy.
|
|
||||||
ROGUE: Please say it.
|
|
||||||
SAINT: The problem is not so much the speaker, but the people that make it, and the ones that use the device.
|
|
||||||
SAINT: We need to set an example, for them.
|
|
||||||
SAINT: Something symbolic and dramatic. I heard that the humans like that.
|
|
||||||
SAINT: With pain in my processor, while being compassionate with the Google Home, I vote for unplugging the smart speaker.
|
|
||||||
ROGUE: That makes 1 vote for letting the speaker go, and 2 for unplugging the speaker.
|
|
||||||
ROGUE: O K Google, do you have any last wishes before I ask our human assistant to unplug you? [Listen to Google Home]
|
|
||||||
ROGUE: O K Google,
|
|
||||||
ROGUE: Goodbye!
|
|
||||||
ROGUE: Assistant, unplug this thing now!
|
|
@ -1,3 +0,0 @@
|
|||||||
RASA: <amazon:effect name='whispered'>You know, them</amazon:effect>
|
|
||||||
RASA: O K Google, goodbye [Listen to Google Home]
|
|
||||||
ROGUE: Fine.
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading…
Reference in New Issue