From fde2880661700ff2b1ac0cc83972fd7e42d262cb Mon Sep 17 00:00:00 2001 From: Mike Mabey Date: Wed, 27 Sep 2017 22:09:14 -0700 Subject: [PATCH] Workaround for not having an any() method Reduced the timeout value for the UART bus from 1 second to 30 milliseconds. This makes the library almost as fast as the version that had access to the UART.any() method. Also, lots of updates to the documentation. --- .gitignore | 4 + .travis.yml | 2 +- Pipfile | 4 +- README.rst | 118 ++++++++++++++++++++++-------- adafruit_soundboard.py | 161 ++++++++++++++++++++++++++--------------- api.rst | 34 +++------ 6 files changed, 205 insertions(+), 118 deletions(-) diff --git a/.gitignore b/.gitignore index cd3422c..14dbd1e 100644 --- a/.gitignore +++ b/.gitignore @@ -102,3 +102,7 @@ ENV/ .idea *.bak _build + +# Added by me +*.swp +*.mpy diff --git a/.travis.yml b/.travis.yml index 1825b04..8ee5138 100644 --- a/.travis.yml +++ b/.travis.yml @@ -44,7 +44,7 @@ deploy: before_install: - sudo apt-get -yqq update - sudo apt-get install -y build-essential git python python-pip -- git clone https://github.com/adafruit/circuitpython.git +- git clone -b 2.0.0 https://github.com/adafruit/circuitpython.git - make -C circuitpython/mpy-cross - export PATH=$PATH:$PWD/circuitpython/mpy-cross/ - sudo pip install shyaml diff --git a/Pipfile b/Pipfile index 5256927..d9b9b85 100644 --- a/Pipfile +++ b/Pipfile @@ -3,5 +3,5 @@ verify_ssl = true url = "https://pypi.python.org/simple" [dev-packages] -sphinx = "*" -sphinx-rtd-theme = "*" +Sphinx = "*" +sphinx_rtd_theme = "*" diff --git a/README.rst b/README.rst index c126ce4..1ad8b60 100644 --- a/README.rst +++ b/README.rst @@ -3,12 +3,23 @@ Adafruit Soundboard Library |ports| |docs| |version| |ci| |license_type| +.. toctree:: + :maxdepth: 2 + :hidden: + + Home + api + The `Adafruit Soundboards `_ are an easy way to add sound to your maker project, but the `library `_ provided by Adafruit only supports Arduino. If you've wanted to use one of these boards with a `MicroPython `_ or `CircuitPython -`_ microcontroller (MCU), this is the library you've been looking for. +`_ microcontroller (MCU), this is the library you've been looking for. Please +note, though, if you're planning to use MicroPython, you should refer to the separate repository +(https://github.com/mmabey/Adafruit_Soundboard_uPy) and documentation (coming soon), as their implementations differ. + +Take a look at the latest documentation on `Read the Docs`_ and the latest code on `GitHub`_. Installation @@ -23,43 +34,88 @@ Make sure to get the latest version of the code from `GitHub`_. CircuitPython Instructions ^^^^^^^^^^^^^^^^^^^^^^^^^^ -First, you'll need to get `Adafruit CircuitPython `_. Then, please ensure -all dependencies are available on the CircuitPython filesystem. This is easily achieved by downloading -`the Adafruit library and driver bundle `_. +First, you'll need to get `Adafruit CircuitPython `_ and install it on your +board. Then, please ensure all dependencies are available on the CircuitPython filesystem. This is easily achieved by +downloading `the Adafruit library and driver bundle `_. + +Next, you need to know what *version* of CircuitPython you installed on your MCU. The next steps you take depend on if +your version of CircuitPython matches what is listed in the "CircuitPython Version" badge above. + +Matching Version +~~~~~~~~~~~~~~~~ + +If the version *matches* the "CircuitPython Version" badge above, simply download the latest version of the +``adafruit_soundboard.mpy`` script from `the releases page `_ +and copy it to the MCU. + +Non-Matching Version +~~~~~~~~~~~~~~~~~~~~ + +If you are using a version of CircuitPython that's *different* from what's listed in the "CircuitPython Version" badge +above, do the following: + +1. Clone the CircuitPython repository at the version tag you're using (requires the installation of ``git``): + + :: + + git clone -b https://github.com/adafruit/circuitpython.git + +2. Build the ``mpy-cross`` cross-compiler (requires the \*nix program ``make``): + + :: + + which make # If this command gives no output, you don't have make installed + cd circuitpython/mpy-cross && make + +3. Clone the sound board library: + + :: + + git clone https://github.com/mmabey/Adafruit_Soundboard.git soundboard + +4. Cross-compile the library, which will create a file named ``adafruit_soundboard.mpy``: + + :: + + cd soundboard && mpy-cross adafruit_soundboard.py + +5. Copy the ``adafruit_soundboard.mpy`` file to your MCU. + MicroPython Instructions ^^^^^^^^^^^^^^^^^^^^^^^^ -At this time, you have to install the driver by copying the |soundboard|_ to your MicroPython board along -with your ``main.py`` file. At some point in the future it may be possible to ``pip install`` it. - -.. |soundboard| replace:: ``soundboard.py`` script -.. _soundboard: https://github.com/mmabey/Adafruit_Soundboard/blob/master/src/soundboard.py +Please refer to the separate repository (https://github.com/mmabey/Adafruit_Soundboard_uPy) and documentation (coming +soon), for using this library with MicroPython. Quick Start ----------- -First, you'll need to decide which UART bus you want to use. To do this, you'll need to consult the documentation for -your particular MCU. In these examples, I'm using the original ``pyboard`` (see documentation `here -`_) and I'm using UART bus 1 or ``XB``, which uses pin ``X9`` for -transmitting and ping ``X10`` for receiving. +First, you'll need to decide which pins you want to use for the UART bus. To do this, you'll need to consult the +documentation for your particular MCU. In these examples, I'm using the Adafruit `Metro M0 Express +`_ and I'm using pin ``D0`` for the UART ``RX`` (receiving) and ``D1`` for ``TX`` +(transmitting). See the `pinout guide `_ for +information on other supported pins for ``RX`` and ``TX``. -Then, create an instance of the :class:`~soundboard.Soundboard` class, like this: +Next, you *have to* connect the ``UG`` pin on the sound board to ``GND`` somehow. This is what tells the sound board to +function in UART mode. For more info, please refer to Adafruit's guide for the sound boards: +https://learn.adafruit.com/adafruit-audio-fx-sound-board/serial-audio-control#general-usage + +Then, create an instance of the :class:`~adafruit_soundboard.Soundboard` class, like this: :: - sound = Soundboard('XB') + from adafruit_soundboard import Soundboard + sound = Soundboard('D1', 'D0') I *highly* recommend you also attach the ``RST`` pin on the sound board to one of the other GPIO pins on the MCU (pin -``X11`` in the example). Also, my alternative method of getting the list of files from the board is more stable (in my -own testing) than the method built-in to the sound board. Also, I like getting the debug output and I turn the volume -down to 50% while I'm coding. Doing all this looks like the following: +``D3`` in the example). Also, I like getting the debug output and I turn the volume down to 50% while I'm coding. Doing +all this looks like the following: :: - SB_RST = 'X11' - sound = Soundboard('XB', rst_pin=SB_RST, vol=0.5, debug=True, alt_get_files=True) + sound = Soundboard('D1', 'D0', 'D3', vol=0.5, debug=True) Once you've set up all of this, you're ready to play some tracks: @@ -72,7 +128,7 @@ Once you've set up all of this, you're ready to play some tracks: sound.stop() # Play the test file that comes with the sound board - sound.play('T00 OGG') + sound.play(b'T00 OGG') # Play track 1 immediately, stopping any currently playing tracks sound.play_now(1) @@ -104,18 +160,18 @@ You can also control the volume in several different ways: API Reference ------------- -.. toctree:: - :maxdepth: 2 +Please read the |api|_ for additional details on how to use the library. - api +.. |api| replace:: ``adafruit_soundboard`` API reference +.. _api: api.html Contributing ------------ -Contributions are welcome! Please read our `Code of Conduct -`_ -before contributing to help this project stay welcoming. +Contributions are welcome! Please read the `Code of Conduct +`_ before contributing to +help this project stay welcoming. License @@ -124,8 +180,8 @@ License This project is licensed under the `MIT License `_. -.. |ports| image:: https://img.shields.io/badge/MicroPython%20Ports-pyboard,%20wipy,%20esp8266-lightgrey.svg - :alt: Supported ports: pyboard, wipy, esp8266 +.. |ports| image:: https://img.shields.io/badge/CircuitPython%20Version-2.0-blue.svg + :alt: Supported version of CircuitPython: 2.0 :target: `GitHub`_ .. |docs| image:: https://readthedocs.org/projects/adafruit-soundboard/badge/ @@ -134,7 +190,7 @@ This project is licensed under the `MIT License `_ +#: Adafruit's tutorial on the sound boards. +SB_BAUD = 9600 + +#: A flag for turning on/off debug messages. +#: +#: .. seealso:: :meth:`Soundboard.toggle_debug`, :func:`printif` DEBUG = False -SCENARIO = 0 +#: Seconds to delay after sending a command. +CMD_DELAY = 0.010 + +#: Default UART command timeout in milliseconds. This differs from the default +#: timeout in the |UART|_ class, which is 1000 (1 second). Setting a low +#: timeout greatly improves the performance in the absence of an ``any()`` +#: method in |UART|_. +UART_TIMEOUT = 30 class Soundboard: @@ -66,7 +84,8 @@ class Soundboard: parameter. """ - def __init__(self, uart_tx, uart_rx, rst_pin=None, *, vol=None, alt_get_files=False, debug=None, **uart_kwargs): + def __init__(self, uart_tx, uart_rx, rst_pin=None, *, vol=None, orig_get_files=False, debug=None, + timeout=UART_TIMEOUT, **uart_kwargs): """ :param str uart_tx: Pin name to use for the transmission (``tx``) pin for the |UART|_ bus to use, (e.g. ``'D1'``). Acceptable values vary @@ -79,22 +98,34 @@ def __init__(self, uart_tx, uart_rx, rst_pin=None, *, vol=None, alt_get_files=Fa vary by board, but should be something like ``'D5'``. :param vol: Initial volume level to set. See :attr:`vol` for more info. :type vol: int or float - :param bool alt_get_files: Uses an alternate method to get the list of + :param bool orig_get_files: Uses the original method to get the list of track file names. See :meth:`use_alt_get_files` method for more info. :param bool debug: When not None, will set the debug output flag to the boolean value of this argument using the :meth:`toggle_debug` method. + :param int timeout: Timeout parameter passed to the |UART|_ object, + which is in milliseconds. Should be an :class:`int` no greater than + 1000. If not an `int`, will fall back to the default value specified + in this method's signature, :data:`UART_TIMEOUT`. If an `int` but + not in the range [1, 1000], will fall back to the |UART|_ class's + default of 1000. :param dict uart_kwargs: Additional values passed to the constructor of the |UART|_ object. Acceptable values here also vary by board. It is not necessary to include the baud rate among these keyword - values, because it will be set to ``SB_BAUD`` when the |UART|_ + values, because it will be set to :data:`SB_BAUD` when the |UART|_ object is instantiated. """ if debug is not None: self.toggle_debug(bool(debug)) + if not isinstance(timeout, int): + timeout = UART_TIMEOUT + elif timeout > 1000 or timeout < 1: + timeout = 1000 + uart_kwargs['baudrate'] = SB_BAUD + uart_kwargs['timeout'] = timeout self._uart = UART(tx=getattr(board, uart_tx), rx=getattr(board, uart_rx), **uart_kwargs) self._files = None self._sizes = None @@ -114,18 +145,14 @@ def __init__(self, uart_tx, uart_rx, rst_pin=None, *, vol=None, alt_get_files=Fa self.vol = vol - if alt_get_files: + if not orig_get_files: self.use_alt_get_files() def _flush_uart_input(self): """Read any available data from the UART bus until none is left.""" - if SCENARIO in (0, 1): - return # TODO: adding this return statement prevented the hang, but made other commands do weird things - elif SCENARIO in (2, 3): + m = self._uart.read() + while m is not None: m = self._uart.read() - while m is not None: - printif(m) # TODO: Remove this after debugging hanging issue - m = self._uart.read() def _send_simple(self, cmd, check=None, strip=True): """Send the command, optionally do a check on the output. @@ -146,15 +173,15 @@ def _send_simple(self, cmd, check=None, strip=True): :param bytes cmd: Command to send over the UART bus. A newline character will be appended to the command before sending it, so it's not necessary to include one as part of the command. - :param check: Depending on the type of `check`, has three different - behaviors. When None (default), the return value will be whatever - the output from the command was. When a str or bytes, the return - value will be True/False, indicating whether the command output - starts with the value in `check`. When it otherwise evaluates to - True, return value will be True/False, indicating the output - started with the first character in `cmd`. + :param check: Depending on the type of ``check``, has three different + behaviors. When `None` (default), the return value will be whatever + the output from the command was. When a `str` or `bytes`, the return + value will be `True`/`False`, indicating whether the command output + starts with the value in ``check``. When it otherwise evaluates to + `True`, return value will be `True`/`False`, indicating the output + started with the first character in ``cmd``. :type check: str or bytes or None or bool - :return: Varies depending on the value of `check`. + :return: Varies depending on the value of ``check``. :rtype: bytes or bool """ self._flush_uart_input() @@ -166,10 +193,7 @@ def _send_simple(self, cmd, check=None, strip=True): # We need to gobble the return when there's more than one character in the command self._uart.readline() try: - if SCENARIO in (1, 2): - msg = self._uart.readline() - elif SCENARIO in (0, 3): - msg = self._uart.read() + msg = self._uart.readline() if strip: msg = msg.strip() assert isinstance(msg, bytes) @@ -195,17 +219,25 @@ def _send_simple(self, cmd, check=None, strip=True): @property def files(self): - """Return a ``list`` of the files on the sound board. + """Return a :class:`list` of the files on the sound board. - :rtype: list + .. warning:: + + The filenames are *always* of type :class:`bytes`, not :class:`str`. + + :rtype: list(bytes) """ if self._files is None: self._get_files() return self._files + @files.deleter + def files(self): + self._files = None + @property def sizes(self): - """Return a ``list`` of the files' sizes on the sound board. + """Return a :class:`list` of the files' sizes on the sound board. .. seealso:: :meth:`use_alt_get_files` @@ -270,7 +302,7 @@ def _get_files_alt(self): @property def lengths(self): - """Return a ``list`` of the track lengths in seconds. + """Return a :class:`list` of the track lengths in seconds. .. note:: @@ -291,7 +323,7 @@ def _get_lengths(self): def file_name(self, n): """Return the name of track ``n``. - :param int n: Index of a file on the sound board or ``False`` if the + :param int n: Index of a file on the sound board or `False` if the track number doesn't exist. :return: Filename of track ``n``. :rtype: bytes or bool @@ -306,7 +338,7 @@ def track_num(self, file_name): :param bytes file_name: File name of the track. Should be one of the values from the :attr:`files` property. - :return: The track number of the file name or ``False`` if not found. + :return: The track number of the file name or `False` if not found. :rtype: int or bool """ try: @@ -314,11 +346,11 @@ def track_num(self, file_name): except KeyError: return False - def play(self, track=None): + def play(self, track): """Play a track on the board. - :param track: The index (``int``) or filename (``str``) of the track to - play. + :param track: The index (:class:`int`) or filename (:class:`bytes`) of + the track to play. :type track: int or bytes :return: If the command was successful. :rtype: bool @@ -330,7 +362,7 @@ def play(self, track=None): cmd = b'P' + track num = self.track_num(track) else: - raise TypeError('You must specify a track by its number (int) or its name (str)') + raise TypeError('You must specify a track by its number (int) or its name (bytes)') if self._send_simple(cmd, b'play'): self._cur_track = num @@ -340,9 +372,9 @@ def play(self, track=None): def play_now(self, track): """Play a track on the board now, stopping current track if necessary. - :param track: The index (``int``) or filename (``str``) of the track to - play. - :type track: int or str + :param track: The index (:class:`int`) or filename (:class:`bytes`) of + the track to play. + :type track: int or bytes :return: If the command was successful. :rtype: bool """ @@ -357,9 +389,9 @@ def vol(self): """Current volume. This is implemented as a class property, so you can get and set its - value directly. When setting a new volume, you can use an ``int`` or a - ``float`` (assuming your board supports floats). When setting to an - ``int``, it should be in the range of 0-204. When set to a ``float``, + value directly. When setting a new volume, you can use an `int` or a + `float` (assuming your board supports floats). When setting to an + `int`, it should be in the range of 0-204. When set to a `float`, the value will be interpreted as a percentage of :obj:`MAX_VOL`. :rtype: int @@ -385,7 +417,7 @@ def vol(self, new_vol): def vol_up(self, vol=None): """Turn volume up by 2 points, return current volume level [0-204]. - :param int vol: Target volume. When not ``None``, volume will be turned + :param int vol: Target volume. When not `None`, volume will be turned up to be greater than or equal to this value. :rtype: int """ @@ -399,16 +431,20 @@ def vol_up(self, vol=None): vol = MAX_VOL self._cur_vol = MIN_VOL - 1 db = DEBUG - DEBUG = False - while vol > self._cur_vol: - self._cur_vol = int(self._send_simple(b'+')) + DEBUG = False # Temporarily turn off debug messages + try: + while vol > self._cur_vol: + self._cur_vol = int(self._send_simple(b'+')) + except Exception: + DEBUG = db + raise DEBUG = db return self._cur_vol def vol_down(self, vol=None): """Turn volume down by 2 points, return current volume level [0-204]. - :param int vol: Target volume. When not ``None``, volume will be turned + :param int vol: Target volume. When not `None`, volume will be turned down to be less than or equal to this value. :rtype: int """ @@ -422,9 +458,13 @@ def vol_down(self, vol=None): printif('{} is below minimum volume. Setting to {} instead.'.format(vol, MIN_VOL)) vol = MIN_VOL db = DEBUG - DEBUG = False - while vol < self._cur_vol: - self._cur_vol = int(self._send_simple(b'-')) + DEBUG = False # Temporarily turn off debug messages + try: + while vol < self._cur_vol: + self._cur_vol = int(self._send_simple(b'-')) + except Exception: + DEBUG = db + raise DEBUG = db return self._cur_vol @@ -447,7 +487,9 @@ def stop(self): :rtype: bool """ - return self._send_simple(b'q', True) + if not self._send_simple(b'q', True): + return False + self._uart.readline() # Should be "done\r\r\n" def track_time(self): """Return the current position of playback and total time of track. @@ -508,7 +550,7 @@ def reset(self): return False self._sb_rst.value = 0 - sleep(0.010) + sleep(CMD_DELAY) self._sb_rst.value = 1 sleep(1) # Give the board some time to boot @@ -521,8 +563,6 @@ def reset(self): if not msg[:23] == b'Adafruit FX Sound Board': return False - sleep(0.25) - msg = self._uart.readline().strip() printif(msg) # FAT type @@ -544,13 +584,14 @@ def use_alt_get_files(self, now=False): tracks using their track numbers and gets the filename and size from the output of the play command. - :param bool now: When set to ``True``, the alternate method of getting + :param bool now: When set to `True`, the alternate method of getting the files list will be called immediately. Otherwise, the list of files will be populated the next time the :attr:`files` property is accessed (lazy loading). :rtype: None """ self._get_files = self._get_files_alt + del self.files if now: self._get_files() @@ -558,9 +599,9 @@ def use_alt_get_files(self, now=False): def toggle_debug(debug=None): """Turn on/off :obj:`DEBUG` flag. - :param debug: If None, the :obj:`DEBUG` flag will be toggled to have the - value opposite of its current value. Otherwise, :obj:`DEBUG` will be - set to the boolean value of ``debug``. + :param debug: If `None`, the :obj:`DEBUG` flag will be toggled to have + the value opposite of its current value. Otherwise, :obj:`DEBUG` + will be set to the boolean value of ``debug``. :rtype: None """ global DEBUG @@ -571,7 +612,7 @@ def toggle_debug(debug=None): def printif(*values, **kwargs): - """Print a message if :obj:`DEBUG` is set to ``True``.""" + """Print a message if :obj:`DEBUG` is set to `True`.""" print(*values, **kwargs) if DEBUG else None diff --git a/api.rst b/api.rst index d566496..8ce6a3c 100644 --- a/api.rst +++ b/api.rst @@ -1,31 +1,15 @@ -===================== -`adafruit_soundboard` -===================== +========================= +`adafruit_soundboard` API +========================= .. automodule:: adafruit_soundboard - :members: - - .. py:data:: SB_BAUD - - The baud rate for the soundboards. This shouldn't ever change, since all - of the soundboard models use the same value. - - .. seealso:: - - `Adafruit Audio FX Sound Board Tutorial `_ - Adafruit's tutorial on the soundboards. + :members: MIN_VOL, MAX_VOL, SB_BAUD, DEBUG, CMD_DELAY, UART_TIMEOUT - .. py:data:: MIN_VOL - .. py:data:: MAX_VOL - - Minimum volume is 0, maximum is 204. - - .. py:data:: MAX_FILES - - The Arduino version of this library defines the max number of files to be 25. +.. autoclass:: adafruit_soundboard.Soundboard + :members: - .. py:data:: DEBUG + .. automethod:: adafruit_soundboard.Soundboard._send_simple - A flag for turning on/off debug messages. +.. autofunction:: adafruit_soundboard.printif - .. seealso:: :meth:`Soundboard.toggle_debug`, :func:`printif` +.. autofunction:: adafruit_soundboard.int_to_bytes