diff --git a/Dockerfile b/Dockerfile index af84c4d64aab1451224c234bc01ff92bfe15fe82..6131d5228d5f8da9f499fa3731b38575cc3c7a92 100644 --- a/Dockerfile +++ b/Dockerfile @@ -2,6 +2,7 @@ FROM python:3.9-alpine as sphinx-build-env RUN apk update RUN python -m pip install --upgrade pip RUN pip install -U sphinx +RUN pip install -U sphinx_design RUN pip install sphinxcontrib-svg2pdfconverter RUN apk add librsvg RUN pip install sphinx_rtd_theme diff --git a/VERSION b/VERSION index c03c0b34584b9b6e27e79f0f218f59d9d0f52f88..52daafdf597eab5aac8580ab3cc5fb35f28ca3e9 100644 --- a/VERSION +++ b/VERSION @@ -1,5 +1,5 @@ VERSION_MAJOR = 0 VERSION_MINOR = 0 -PATCHLEVEL = 6 -VERSION_TWEAK = 1 +PATCHLEVEL = 7 +VERSION_TWEAK = 2 EXTRAVERSION = beta diff --git a/beaglebone-ai-64/edge_ai_apps/configuration_file.rst b/beaglebone-ai-64/edge_ai_apps/configuration_file.rst new file mode 100644 index 0000000000000000000000000000000000000000..fa062aea0b3eb7c620b57e58770d86e845e18ea1 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/configuration_file.rst @@ -0,0 +1,375 @@ +.. _ai_64_edgeai_configuration: + +Demo Configuration file +######################### + +The demo config file uses YAML format to define input sources, models, outputs +and finally the flows which defines how everything is connected. Config files +for out-of-box demos are kept in ``edge_ai_apps/configs`` folder. The +folder contains config files for all the use cases and also multi-input and +multi-inference case. The folder also has a template YAML file +``app_config_template.yaml`` which has detailed explanation of all the +parameters supported in the config file. + +Config file is divided in 4 sections: + +#. Inputs +#. Models +#. Outputs +#. Flows + +Inputs +====== + +The input section defines a list of supported inputs like camera, filesrc etc. +Their properties like shown below. + +.. code-block:: yaml + + inputs: + input0: #Camera Input + source: /dev/video2 #Device file entry of the camera + format: jpeg #Input data format suported by camera + width: 1280 #Width and Height of the input + height: 720 + framerate: 30 #Framerate of the source + + input1: #Video Input + source: ../data/videos/video_0000_h264.mp4 #Video file + format: h264 #File encoding format + width: 1280 + height: 720 + framerate: 25 + + input2: #Image Input + source: ../data/images/%04d.jpg #Sequence of Image files, printf style formatting is used + width: 1280 + height: 720 + index: 0 #Starting Index (optional) + framerate: 1 + +All supported inputs are listed in template config file. +Below are the details of most commonly used inputs. + +.. _ai_64_edgeai_camera_sources: + +Camera sources (v4l2) +--------------------- + +**v4l2src** GStreamer element is used to capture frames from camera sources +which are exposed as v4l2 devices. In Linux, there are many devices which are +implemented as v4l2 devices. Not all of them will be camera devices. You need +to make sure the correct device is configured for running the demo successfully. + +``init_script.sh`` is ran as part of systemd, which detects all cameras connected +and prints the detail like below in the UART console: + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps# ./init_script.sh + USB Camera detected + device = /dev/video18 + format = jpeg + CSI Camera 0 detected + device = /dev/video2 + name = imx219 8-0010 + format = [fmt:SRGGB8_1X8/1920x1080] + subdev_id = 2 + isp_required = yes + IMX390 Camera 0 detected + device = /dev/video18 + name = imx390 10-001a + format = [fmt:SRGGB12_1X12/1936x1100 field: none] + subdev_id = /dev/v4l-subdev7 + isp_required = yes + ldc_required = yes + +script can also be run manually later to get the camera details. + +From the above log we can determine that 1 USB camera is connected +(/dev/video18), and 1 CSI camera is connected (/dev/video2) which is imx219 raw +sensor and needs ISP. IMX390 camera needs both ISP and LDC. + +Using this method, you can configure correct device for camera capture in the +input section of config file. + +.. code-block:: bash + + input0: + source: /dev/video18 #USB Camera + format: jpeg #if connected USB camera supports jpeg + width: 1280 + height: 720 + framerate: 30 + + input1: + source: /dev/video2 #CSI Camera + format: auto #let the gstreamer negotiate the format + width: 1280 + height: 720 + framerate: 30 + + input2: + source: /dev/video2 #IMX219 raw sensor that nees ISP + format: rggb #ISP will be added in the pipeline + width: 1920 + height: 1080 + framerate: 30 + subdev-id: 2 #needed by ISP to control sensor params via ioctls + + input3: + source: /dev/video2 #IMX390 raw sensor that nees ISP + width: 1936 + height: 1100 + format: rggb12 #ISP will be added in the pipeline + subdev-id: 2 #needed by ISP to control sensor params via ioctls + framerate: 30 + sen-id: imx390 + ldc: True #LDC will be added in the pipeline + +Make sure to configure correct ``format`` for camera input. ``jpeg`` for USB +camera that supports MJPEG (Ex. C270 logitech USB camera). ``auto`` for CSI +camera to allow gstreamer to negotiate the format. ``rggb`` for sensor +that needs ISP. + +Video sources +------------- + +H.264 and H.265 encoded videos can be provided as input sources to the demos. +Sample video files are provided under ``/opt/edge_ai_apps/data/videos/video_0000_h264.mp4`` +and ``/opt/edge_ai_apps/data/videos/video_000_h265.mp4`` + +.. code-block:: yaml + + input1: + source: ../data/videos/video_0000_h264.mp4 + format: h264 + width: 1280 + height: 720 + framerate: 25 + + input2: + source: ../data/videos/video_0000_h265.mp4 + format: h265 + width: 1280 + height: 720 + framerate: 25 + +Make sure to configure correct ``format`` for video input as shown above. +By default the format is set to ``auto`` which will then use the GStreamer +bin ``decodebin`` instead. + +Image sources +------------- + +JPEG compressed images can be provided as inputs to the demos. A sample set of +images are provided under ``/opt/edge_ai_apps/data/images``. The names of the +files are numbered sequentially and incrementally and the demo plays the files +at the fps specified by the user. + +.. code-block:: yaml + + input2: + source: ../data/images/%04d.jpg + width: 1280 + height: 720 + index: 0 + framerate: 1 + +RTSP sources +------------ + +H.264 encoded video streams either coming from a RTSP compliant IP camera or +via RTSP server running on a remote PC can be provided as inputs to the demo. + +.. code-block:: yaml + + input0: + source: rtsp://172.24.145.220:8554/test # rtsp stream url, replace this with correct url + width: 1280 + height: 720 + framerate: 30 + +.. note:: + + Usually video streams from any IP camera will be encrypted and cannot be + played back directly without a decryption key. We tested RTSP source by + setting up an RTSP server on a Ubuntu 18.04 PC by refering to this writeup, + `Setting up RTSP server on PC + <https://gist.github.com/Santiago-vdk/80c378a315722a1b813ae5da1661f890>`_ + +Models +====== + +The model section defines a list of models that are used in the demo. Path to +the model directory is a required argument for each model and rest are optional +properties specific to given use cases like shown below. + +.. code-block:: yaml + + models: + model0: + model_path: ../models/segmentation/ONR-SS-871-deeplabv3lite-mobv2-cocoseg21-512x512 #Model Directory + alpha: 0.4 #alpha for blending segmentation mask (optional) + model1: + model_path: ../models/detection/TFL-OD-202-ssdLite-mobDet-DSP-coco-320x320 + viz_threshold: 0.3 #Visualization threshold for adding bounding boxes (optional) + model2: + model_path: ../models/classification/TVM-CL-338-mobileNetV2-qat + topN: 5 #Number of top N classes (optional) + +Below are some of the use case specific properties: + +#. **alpha**: This determines the weight of the mask for blending the semantic + segmentation output with the input image ``alpha * mask + (1 - alpha) * image`` +#. **viz_threshold**: Score threshold to draw the bounding boxes for detected + objects in object detection. This can be used to control the number of boxes + in the output, increase if there are too many and decrease if there are very + few +#. **topN**: Number of most probable classes to overlay on image classification + output + +The content of the model directory and its structure is discussed in detail in +:ref:`pub_edgeai_import_custom_models` + + +Outputs +======= + +The output section defines a list of supported outputs. + +.. code-block:: yaml + + outputs: + output0: #Display Output + sink: kmssink + width: 1920 #Width and Height of the output + height: 1080 + connector: 39 #Connector ID for kmssink (optional) + + output1: #Video Output + sink: ../data/output/videos/output_video.mkv #Output video file + width: 1920 + height: 1080 + + output2: #Image Output + sink: ../data/output/images/output_image_%04d.jpg #Image file name, printf style formatting is used + width: 1920 + height: 1080 + +All supported outputs are listed in template config file. +Below are the details of most commonly used outputs + +Display Sink (kmssink) +---------------------- + +When you have only one display connected to the SK, kmssink will try to use +it for displaying the output buffers. In case you have connected multiple +display monitors (e.g. Display Port and HDMI), you can select a specific display +for kmssink by passing a specific connector ID number. +Following command finds out the connected displays available to use. + +**Note**: Run this command outside docker container. The first number in each +line is the connector-id which we will use in next step. + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps# modetest -M tidss -c | grep connected + 39 38 connected DP-1 530x300 12 38 + 48 0 disconnected HDMI-A-1 0x0 0 47 + +From above output, we can see that connector ID 39 is connected. Configure the +connector ID in the output section of the config file. + +Video sinks +----------- +The post-processed outputs can be encoded in H.264 format and stored on disk. +Please specify the location of the video file in the configuration file. + +.. code-block:: yaml + + output1: + sink: ../data/output/videos/output_video.mkv + width: 1920 + height: 1080 + +Image sinks +----------- +The post-processed outputs can be stored as JPEG compressed images. +Please specify the location of the image files in the configuration file. +The images will be named sequentially and incrementally as shown. + +.. code-block:: yaml + + output2: + sink: ../data/output/images/output_image_%04d.jpg + width: 1920 + height: 1080 + +Flows +===== + +The flows section defines how inputs, models and outputs are connected. +Multiple flows can be defined to achieve multi input, multi inference like +below. + +.. code-block:: yaml + + flows: + flow0: #First Flow + input: input0 #Input for the Flow + models: [model1, model2] #List of models to be used + outputs: [output0, output0] #Outputs to be used for each model inference output + mosaic: #Positions to place the inference outputs in the output frame + mosaic0: + width: 800 + height: 450 + pos_x: 160 + pos_y: 90 + mosaic1: + width: 800 + height: 450 + pos_x: 960 + pos_y: 90 + flow1: #Second Flow + input: input1 + models: [model0, model3] + outputs: [output0, output0] + mosaic: + mosaic0: + width: 800 + height: 450 + pos_x: 160 + pos_y: 540 + mosaic1: + width: 800 + height: 450 + pos_x: 960 + pos_y: 540 + +Each flow should have exactly **1 input**, **n models** to infer the given input +and **n outputs** to render the output of each inference. Along with input, models +and outputs it is required to define **n mosaics** which are the position of the +inference output in the final output plane. This is needed because multiple +inference outputs can be rendered to same output (Ex: Display). + +Command line arguments +---------------------- + +Limited set of command line arguments can be provided, run with '-h' or '--help' +option to list the supported parameters. + +.. code-block:: bash + + usage: Run : ./app_edgeai.py -h for help + + positional arguments: + config Path to demo config file + ex: ./app_edgeai.py ../configs/app_config.yaml + + optional arguments: + -h, --help show this help message and exit + -n, --no-curses Disable curses report + default: Disabled + -v, --verbose Verbose option to print profile info on stdout + default: Disabled diff --git a/beaglebone-ai-64/edge_ai_apps/data_flows.rst b/beaglebone-ai-64/edge_ai_apps/data_flows.rst new file mode 100644 index 0000000000000000000000000000000000000000..f328baabe7a5d336190cd26dde156740a4504924 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/data_flows.rst @@ -0,0 +1,468 @@ +.. _ai_64_data_flows: + + +Data Flows +########### + +The **app_edgeai** application at a high level can be split into 3 parts, + + - Input pipeline - Grabs a frame from camera, video, image or RTSP source + - Output pipeline - Sends the output to display or a file + - Compute pipeline - Performs pre-processing, inference and post-processing + +Here are the data flows for each reference demo and the corresponding GStreamer +launch strings that **app_edgeai** application generates. User can interact with +the application via the :ref:`pub_edgeai_configuration` + +.. _ai_64_edgeai_image_classification_data_flow: + +Image classification +==================== + +In this demo, a frame is grabbed from an input source and split into two paths. +The "analytics" path resizes the input maintaining the aspect ratio and crops +the input to match the resolution required to run the deep learning network. +The "visualization" path is provided to the post-processing module which +overlays the detected classes. Post-processed output is given to HW mosaic plugin +which positions and resizes the output window on an empty background before +sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=454, height=256 ! tiovxdlcolorconvert out-pool-size=4 ! video/x-raw, format=RGB ! videobox left=115 right=115 top=16 bottom=16 ! tiovxdlpreproc data-type=10 channel-order=0 mean-0=123.675000 mean-1=116.280000 mean-2=103.530000 scale-0=0.017125 scale-1=0.017507 scale-2=0.017429 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_image_classification.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline for image classification demo with USB camera and display + +.. _ai_64_edgeai_object_detection_data_flow: + +Object Detection +================ + +In this demo, a frame is grabbed from an input source and split into two paths. +The "analytics" path resizes the input to match the resolution required to run +the deep learning network. The "visualization" path is provided to the +post-processing module which overlays rectangles around detected objects. +Post-processed output is given to HW mosaic plugin which positions and resizes +the output window on an empty background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_object_detection.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline for object detection demo with USB camera and display + +.. _ai_64_edgeai_semantic_segmentation_data_flow: + +Semantic Segmentation +===================== + +In this demo, a frame is grabbed from an input source and split into two paths. +The "analytics" path resize the input to match the resolution required to run +the deep learning network. The "visualization" path is provided to the +post-processing module which blends each segmented pixel to a color map. +Post-processed output is given to HW mosaic plugin which positions and resizes +the output window on an empty background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=512, height=512 ! tiovxdlpreproc data-type=10 channel-order=0 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.015625 scale-1=0.015625 scale-2=0.015625 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_semantic_segmentation.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline for semantic segmentation demo with USB camera and display + +.. _ai_64_edgeai_human_pose_estimation_data_flow: + +Human Pose Estimation +===================== + +In this demo, a frame is grabbed from an input source and split into two paths. +The "analytics" path resize the input to match the resolution required to run +the deep learning network. The "visualization" path is provided to the +post-processing module which overlays the keypoints and lines to draw the pose. +Post-processed output is given to HW mosaic plugin which positions and resizes +the output window on an empty background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video2 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=640, height=640 ! tiovxdlpreproc data-type=10 target=0 channel-order=0 mean-0=0.000000 mean-1=0.000000 mean-2=0.000000 scale-0=1.000000 scale-1=1.000000 scale-2=1.000000 tensor-format=bgr out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + tiovxmosaic name=mosaic_0 background=/tmp/background_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_human_pose.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline for Human Pose Estimation demo with USB camera and display + +.. _ai_64_edgeai_video_source_data_flow: + +Video source +============ + +In this demo, a video file is read from a known location and passed to a +de-muxer to extract audio and video streams, the video stream is parsed +and raw encoded information is passed to a HW video decoder. Note that H.264 and +H.265 encoded videos are supported, making use of the respective HW decoders. +The resulting output is split into two paths. The "analytics" path resizes the +input to match the resolution required to run the deep learning network. The +"visualization" path is provided to the post-processing module which does the +required post process required by the model. Post-processed output is given to +HW mosaic plugin which positions and resizes the output window on an empty +background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + filesrc location=/opt/edge_ai_apps/data/videos/video_0000_h264.mp4 ! qtdemux ! h264parse ! v4l2h264dec ! video/x-raw, format=NV12 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_video_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with video file input source and display + +.. _ai_64_edgeai_rtsp_source_data_flow: + +RTSP source +============ + +In this demo, a video file is read from a RTSP source and passed to a +de-muxer to extract audio and video streams, the video stream is parsed +and raw encoded information is passed to a video decoder and the resulting +output is split into two paths. The "analytics" path resizes the input to match +the resolution required to run the deep learning network. The "visualization" +path is provided to the post-processing module which does the required post +process required by the model. Post-processed output is given to HW mosaic plugin +which positions and resizes the output window on an empty background before +sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + rtspsrc location=rtsp://172.24.145.220:8554/test latency=0 buffer-mode=auto ! rtph264depay ! h264parse ! v4l2h264dec ! video/x-raw, format=NV12 !tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_rtsp_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with RTSP based video file source and display + +.. _ai_64_edgeai_rpi_camera_data_flow: + +RPiV2 Camera Sensor (IMX219) +============================ + +In this demo, raw frames in SRGGB8 format are captured form RPiV2 (imx219) +camera sensor. VISS (Vision Imaging Subsystem) is used to process the raw frames +and get the output in NV12, VISS also cotrols the sensor parameters like +exposure, gain etc.. via v4l2 ioctls. The NV12 output is split into two paths. +The "analytics" path resizes the input to match the resolution required to run +the deep learning network. The "visualization" path is provided to the +post-processing module which does the required post process required by the +model. Post-processed output is given to HW mosaic plugin which positions and +resizes the output window on an empty background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video2 io-mode=5 ! video/x-bayer, width=1920, height=1080, format=rggb ! tiovxisp device=/dev/v4l-subdev2 dcc-isp-file=/opt/imaging/imx219/dcc_viss.bin dcc-2a-file=/opt/imaging/imx219/dcc_2a.bin format-msb=7 ! video/x-raw, format=NV12 ! tiovxmultiscaler ! video/x-raw, width=1280, height=720 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_rpi_camera_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with IMX219 sensor, ISP and display + + +.. _ai_64_edgeai_imx390_camera_data_flow: + +IMX390 Camera Sensor +============================ +In this demo, raw frames in SRGGB12 format are captured from IMX390 camera +sensor. VISS (Vision Imaging Subsystem) is used to process the raw frames +and get the output in NV12, VISS also controls the sensor parameters like +exposure, gain etc.. via v4l2 ioctls. This is followed by LDC (Lens Distortion +Correction) required due to the fisheye lens. The NV12 output is split into two paths. +The "analytics" path resizes the input to match the resolution required to run +the deep learning network. The "visualization" path is provided to the +post-processing module which does the required post process required by the +model. Post-processed output is given to HW mosaic plugin which positions and +resizes the output window on an empty background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 ! queue leaky=2 ! video/x-bayer, width=1936, height=1100, format=rggb12 ! tiovxisp sink_0::device=/dev/v4l-subdev7 sensor-name=IMX390-UB953_D3 dcc-isp-file=/opt/imaging/imx390/dcc_viss.bin sink_0::dcc-2a-file=/opt/imaging/imx390/dcc_2a.bin format-msb=11 ! video/x-raw, format=NV12 ! tiovxldc dcc-file=/opt/imaging/imx390/dcc_ldc.bin sensor-name=IMX390-UB953_D3 ! video/x-raw, format=NV12, width=1920, height=1080 !tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=512, height=512 ! tiovxdlpreproc data-type=3 target=0 channel-order=0 tensor-format=bgr out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + tiovxmosaic name=mosaic_0 background=/tmp/background_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. figure:: ./images/edgeai_imx390_camera_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with IMX390 sensor, ISP, LDC and display + +.. _ai_64_edgeai_video_output_data_flow: + +Video output +============ + +In this demo, a frame is grabbed from an input source and split into two paths. +The "analytics" path resizes the input to match the resolution required to run +the deep learning network. The "visualization" path is provided to the +post-processing module which does the required post process required by the +model. Post-processed output is given to HW mosaic plugin which positions and +resizes the output window on an empty background. Finally the video is encoded +using the H.264 HW encoder and written to a video file. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tiovxmultiscaler name=split_01 + split_01. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=1280, height=720 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1280, height=720 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=1280 sink_0::height=720 + ! video/x-raw,format=NV12, width=1920, height=1080 ! v4l2h264enc bitrate=10000000 ! h264parse ! matroskamux ! filesink location=/opt/edge_ai_apps/data/output/videos/output_video.mkv + +.. figure:: ./images/edgeai_video_output.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with video file input source and display + +.. _ai_64_edgeai_single_input_multi_inference_data_flow: + +Single Input Multi inference +============================ + +In this demo, a frame is grabbed from an input source and split into multiple +paths. Each path is further split into two sub-paths one for analytics and +another for visualization. Each path can run any type of network, image +classification, object detection, semantic segmentation and using any +supported run-time. + +For example the below GStreamer pipeline splits the input into 4 paths for +running 4 deep learning networks. First is a semantic segmentation network, +followed by object detection network, followed by two image classification +networks. If we look at the image classification path, the analytics sub-path +resizes the input to maintain the aspect ratio and crops the input to match +the resolution required to run the deep learning network. The visualization +sub-path is provided to the post-processing module which overlays the detected +classes. Post-processed output from all the 4 paths is given to HW mosaic plugin +which positions and resizes the output windows on an empty background before +sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tee name=tee_split0 + tee_split0. ! queue ! tiovxmultiscaler name=split_01 + tee_split0. ! queue ! tiovxmultiscaler name=split_02 + tee_split0. ! queue ! tiovxmultiscaler name=split_03 + tee_split0. ! queue ! tiovxmultiscaler name=split_04 + split_01. ! queue ! video/x-raw, width=512, height=512 ! tiovxdlpreproc data-type=10 channel-order=0 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.015625 scale-1=0.015625 scale-2=0.015625 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + split_02. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_1 max-buffers=2 drop=true + split_02. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_1 max-buffers=2 drop=true + split_03. ! queue ! video/x-raw, width=454, height=256 ! tiovxdlcolorconvert out-pool-size=4 ! video/x-raw, format=RGB ! videobox left=115 right=115 top=16 bottom=16 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_2 max-buffers=2 drop=true + split_03. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_2 max-buffers=2 drop=true + split_04. ! queue ! video/x-raw, width=454, height=256 ! tiovxdlcolorconvert out-pool-size=4 ! video/x-raw, format=RGB ! videobox left=115 right=115 top=16 bottom=16 ! tiovxdlpreproc data-type=10 channel-order=0 mean-0=123.675000 mean-1=116.280000 mean-2=103.530000 scale-0=0.017125 scale-1=0.017507 scale-2=0.017429 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_3 max-buffers=2 drop=true + split_04. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_3 max-buffers=2 drop=true + + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_1 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_1 + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_2 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_2 + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_3 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_3 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=640 sink_0::height=360 + sink_1::startx=960 sink_1::starty=180 sink_1::width=640 sink_1::height=360 + sink_2::startx=320 sink_2::starty=560 sink_2::width=640 sink_2::height=360 + sink_3::startx=960 sink_3::starty=560 sink_3::width=640 sink_3::height=360 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss + +.. _ai_64_edgeai_multi_input_multi_inference_data_flow: + +Multi Input Multi inference +=========================== + +In this demo, a frame is grabbed from multiple input sources and split into +multiple paths. The multiple input sources could be either multiple cameras or +a combination of camera, video, image, RTSP source. Each path is further split +into two sub-paths one for analytics and another for visualization. Each path +can run any type of network, image classification, object detection, +semantic segmentation and using any supported run-time. + +For example the below GStreamer pipeline splits two inputs into 4 paths for +running 2 deep learning networks. First is a object detection network, followed by +image classification networks. If we look at the image classification path, +the analytics sub-path resizes the input to maintain the aspect ratio and crops +the input to match the resolution required to run the deep learning network. +The visualization sub-path is provided to the post-processing module which +overlays the detected classes. Post-processed output from all the 4 paths is +given to HW mosaic plugin which positions and resizes the output windows on an +empty background before sending to display. + +GStreamer input pipeline: + +.. code-block:: bash + + v4l2src device=/dev/video18 io-mode=2 ! image/jpeg, width=1280, height=720 ! jpegdec ! tiovxdlcolorconvert ! video/x-raw, format=NV12 ! tee name=tee_split0 + tee_split0. ! queue ! tiovxmultiscaler name=split_01 + tee_split0. ! queue ! tiovxmultiscaler name=split_02 + split_01. ! queue ! video/x-raw, width=320, height=320 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_0 max-buffers=2 drop=true + split_01. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_0 max-buffers=2 drop=true + split_02. ! queue ! video/x-raw, width=454, height=256 ! tiovxdlcolorconvert out-pool-size=4 ! video/x-raw, format=RGB ! videobox left=115 right=115 top=16 bottom=16 ! tiovxdlpreproc data-type=10 channel-order=1 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.007812 scale-1=0.007812 scale-2=0.007812 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_1 max-buffers=2 drop=true + split_02. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_1 max-buffers=2 drop=true + + filesrc location=/opt/edge_ai_apps/data/videos/video_0000_h264.mp4 ! qtdemux ! h264parse ! v4l2h264dec ! video/x-raw, format=NV12 ! tee name=tee_split1 + tee_split1. ! queue ! tiovxmultiscaler name=split_11 + tee_split1. ! queue ! tiovxmultiscaler name=split_12 + split_11. ! queue ! video/x-raw, width=512, height=512 ! tiovxdlpreproc data-type=10 channel-order=0 mean-0=128.000000 mean-1=128.000000 mean-2=128.000000 scale-0=0.015625 scale-1=0.015625 scale-2=0.015625 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_2 max-buffers=2 drop=true + split_11. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_2 max-buffers=2 drop=true + split_12. ! queue ! video/x-raw, width=454, height=256 ! tiovxdlcolorconvert out-pool-size=4 ! video/x-raw, format=RGB ! videobox left=115 right=115 top=16 bottom=16 ! tiovxdlpreproc data-type=10 channel-order=0 mean-0=123.675000 mean-1=116.280000 mean-2=103.530000 scale-0=0.017125 scale-1=0.017507 scale-2=0.017429 tensor-format=rgb out-pool-size=4 ! application/x-tensor-tiovx ! appsink name=pre_3 max-buffers=2 drop=true + split_12. ! queue ! video/x-raw, width=640, height=360 ! tiovxdlcolorconvert target=1 out-pool-size=4 ! video/x-raw, format=RGB ! appsink name=sen_3 max-buffers=2 drop=true + + +GStreamer output pipeline: + +.. code-block:: bash + + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_0 + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_1 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_1 + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_2 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_2 + appsrc format=GST_FORMAT_TIME is-live=true block=true do-timestamp=true name=post_3 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=640, height=360 ! queue ! mosaic_0.sink_3 + appsrc format=GST_FORMAT_TIME block=true num-buffers=1 name=background_0 ! tiovxdlcolorconvert ! video/x-raw,format=NV12, width=1920, height=1080 ! queue ! mosaic_0.background + tiovxmosaic name=mosaic_0 + sink_0::startx=320 sink_0::starty=180 sink_0::width=640 sink_0::height=360 + sink_1::startx=960 sink_1::starty=180 sink_1::width=640 sink_1::height=360 + sink_2::startx=320 sink_2::starty=560 sink_2::width=640 sink_2::height=360 + sink_3::startx=960 sink_3::starty=560 sink_3::width=640 sink_3::height=360 + ! video/x-raw,format=NV12, width=1920, height=1080 ! kmssink sync=false driver-name=tidss diff --git a/beaglebone-ai-64/edge_ai_apps/datasheet.rst b/beaglebone-ai-64/edge_ai_apps/datasheet.rst new file mode 100644 index 0000000000000000000000000000000000000000..ea8dfebdcbc57a0853b39bfc4be7fa1373fbc1c3 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/datasheet.rst @@ -0,0 +1,132 @@ +.. _ai_64_edgeai_datasheet: + +Datasheet +########## + +This chapter describes the performance measurements of the Edge AI Inference +demos. + +Performance data of the demos can be auto generated by running following +command on target: + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/tests# ./gen_data_sheet.sh + +The performence measurements includes the following + +#. **FPS** : Effective framerate at which the application runs +#. **Total time** : Average time taken to process each frame, which includes + pre-processing, inference and post-processing time +#. **Inference time** : Average time taken to infer each frame +#. **CPU loading** : Loading on different CPU cores present +#. **DDR BW** : DDR read and write BW used +#. **HWA Loading** : Loading on different Hardware accelerators present + +Following are the latest performance numbers of the C++ demos: + +Source : **USB Camera** +==================================== + +Capture Framerate : **30 fps** +Resolution : **720p** +format : **JPEG** + +.. figure:: ./images/edgeai_object_detection.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with USB camera input and display output + +.. csv-table:: + :header: "Model", "FPS", "Total time (ms)", "Inference time (ms)", "A72 Load (%)", "DDR Read BW (MB/s)", "DDR Write BW (MB/s)", "DDR Total BW (MB/s)", "C71 Load (%)", "C66_1 Load (%)", "C66_2 Load (%) ", "MCU2_0 Load (%)", "MCU2_1 Load (%)", "MSC_0 (%)", "MSC_1 (%)", "VISS (%)", "NF (%)", "LDC (%)", "SDE (%)", "DOF (%)" + + ONR-CL-6150-mobileNetV2-1p4-qat,30.80,33.22,3.02,21.60,1596,619,2215,9.0,20.0,9.0,6.0,1.0,22.17,0,0,0,0,0,0 + TFL-CL-0000-mobileNetV1-mlperf,30.69,33.19,1.04,15.93,1425,563,1988,5.0,22.0,9.0,6.0,1.0,21.90,0,0,0,0,0,0 + TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,30.69,33.25,5.00,10.24,1534,570,2104,15.0,29.0,9.0,6.0,1.0,22.67,0,0,0,0,0,0 + TVM-CL-3410-gluoncv-mxnet-mobv2,30.58,33.21,2.02,22.80,1522,617,2139,6.0,20.0,9.0,6.0,1.0,21.84,0,0,0,0,0,0 + + +Source : **Video** +============================== + +Video Framerate : **30 fps** +Resolution : **720p** +Encoding : **h264** + +.. figure:: ./images/edgeai_video_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with video file input source and display output + +.. csv-table:: + :header: "Model", "FPS", "Total time (ms)", "Inference time (ms)", "A72 Load (%)", "DDR Read BW (MB/s)", "DDR Write BW (MB/s)", "DDR Total BW (MB/s)", "C71 Load (%)", "C66_1 Load (%)", "C66_2 Load (%) ", "MCU2_0 Load (%)", "MCU2_1 Load (%)", "MSC_0 (%)", "MSC_1 (%)", "VISS (%)", "NF (%)", "LDC (%)", "SDE (%)", "DOF (%)" + + ONR-CL-6150-mobileNetV2-1p4-qat,30.52,33.46,3.03,14.28,990,403,1393,2.0,7.0,4.0,1.0,1.0,10.27,0,0,0,0,0,0 + TFL-CL-0000-mobileNetV1-mlperf,30.77,33.47,1.07,30.76,746,97,843,2.0,2.0,1.0,1.0,1.0,15.76,0,0,0,0,0,0 + TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,30.56,33.54,5.06,22.58,736,92,828,2.0,2.0,1.0,1.0,1.0,16.9,0,0,0,0,0,0 + TVM-CL-3410-gluoncv-mxnet-mobv2,30.64,33.47,2.01,33.33,712,110,822,1.0,1.0,0.0,1.0,1.0,15.3,0,0,0,0,0,0 + +Source : **CSI Camera (ov5640)** +============================================ + +Capture Framerate : **30 fps** +Resolution : **720p** +format : **YUYV** + +.. figure:: ./images/edgeai_ov5640_camera_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline for with CSI camera (OV5640) input and display output + +.. csv-table:: + :header: "Model", "FPS", "Total time (ms)", "Inference time (ms)", "A72 Load (%)", "DDR Read BW (MB/s)", "DDR Write BW (MB/s)", "DDR Total BW (MB/s)", "C71 Load (%)", "C66_1 Load (%)", "C66_2 Load (%) ", "MCU2_0 Load (%)", "MCU2_1 Load (%)", "MSC_0 (%)", "MSC_1 (%)", "VISS (%)", "NF (%)", "LDC (%)", "SDE (%)", "DOF (%)" + + ONR-CL-6150-mobileNetV2-1p4-qat,29.57,34.09,3.02,12.21,1671,699,2370,8.0,45.0,9.0,6.0,1.0,21.35,0,0,0,0,0,0 + TFL-CL-0000-mobileNetV1-mlperf,29.41,34.15,1.01,10.27,1502,645,2147,5.0,47.0,9.0,6.0,1.0,20.96,0,0,0,0,0,0 + TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,29.36,34.65,5.00,10.5,1610,655,2265,14.0,53.0,9.0,6.0,1.0,21.47,0,0,0,0,0,0 + TVM-CL-3410-gluoncv-mxnet-mobv2,29.38,34.17,2.01,11.66,1596,698,2294,6.0,45.0,9.0,5.0,1.0,21.10,0,0,0,0,0,0 + +Source : **CSI Camera with VISS (imx219)** +====================================================== + +Capture Framerate : **30 fps** +Resolution : **1080p** +format : **SRGGB8** + +.. figure:: ./images/edgeai_rpi_camera_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with IMX219 sensor, ISP and display + +.. csv-table:: + :header: "Model", "FPS", "Total time (ms)", "Inference time (ms)", "A72 Load (%)", "DDR Read BW (MB/s)", "DDR Write BW (MB/s)", "DDR Total BW (MB/s)", "C71 Load (%)", "C66_1 Load (%)", "C66_2 Load (%) ", "MCU2_0 Load (%)", "MCU2_1 Load (%)", "MSC_0 (%)", "MSC_1 (%)", "VISS (%)", "NF (%)", "LDC (%)", "SDE (%)", "DOF (%)" + + ONR-CL-6150-mobileNetV2-1p4-qat,30.64,33.19,3.01,15.72,1781,853,2634,9.0,16.0,9.0,13.0,1.0,31.78,0,22.37,0,0,0,0 + TFL-CL-0000-mobileNetV1-mlperf,30.59,33.14,1.04,12.78,1612,798,2410,5.0,18.0,9.0,13.0,1.0,31.65,0,22.31,0,0,0,0 + TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,30.56,33.07,5.00,13.30,1730,809,2539,15.0,25.0,9.0,13.0,1.0,32.6,0,22.19,0,0,0,0 + TVM-CL-3410-gluoncv-mxnet-mobv2,30.48,33.14,2.01,12.91,1708,852,2560,7.0,16.0,9.0,13.0,1.0,31.83,0,22.26,0,0,0,0 + +Source : **IMX390 over FPD-Link** +============================================= + +Capture Framerate : **30 fps** +Resolution : **1080p** +format : **SRGGB12** + +.. figure:: ./images/edgeai_imx390_camera_source.png + :scale: 60 + :align: center + + GStreamer based data-flow pipeline with IMX390 sensor, ISP, LDC and display + +.. csv-table:: + :header: "Model", "FPS", "Total time (ms)", "Inference time (ms)", "A72 Load (%)", "DDR Read BW (MB/s)", "DDR Write BW (MB/s)", "DDR Total BW (MB/s)", "C71 Load (%)", "C66_1 Load (%)", "C66_2 Load (%) ", "MCU2_0 Load (%)", "MCU2_1 Load (%)", "MSC_0 (%)", "MSC_1 (%)", "VISS (%)", "NF (%)", "LDC (%)", "SDE (%)", "DOF (%)" + + ONR-CL-6150-mobileNetV2-1p4-qat,30.59,33.15,3.09,25.18,2207,1102,3309,10.0,16.0,9.0,14.0,1.0,31.73,0,22.94,0,10.8,0,0 + TFL-CL-0000-mobileNetV1-mlperf,30.53,33.15,1.21,16.20,2019,1040,3059,5.0,18.0,9.0,15.0,1.0,32.80,0,23.34,0,10.10,0,0 + TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,30.43,33.13,5.02,23.7,2201,1067,3268,15.0,25.0,9.0,14.0,1.0,32.80,0,22.88,0,9.95,0 + TVM-CL-3410-gluoncv-mxnet-mobv2,30.44,33.16,2.12,21.50,2111,1100,3211,7.0,16.0,9.0,15.0,1.0,32.28,0,22.88,0,10.6,0,0 diff --git a/beaglebone-ai-64/edge_ai_apps/docker_environment.rst b/beaglebone-ai-64/edge_ai_apps/docker_environment.rst new file mode 100644 index 0000000000000000000000000000000000000000..ed82a3a3575d42905c24cbcf306de9771aa33fb8 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/docker_environment.rst @@ -0,0 +1,233 @@ +.. _ai_64_edgeai_docker_env: + +Docker Environment +################### + +Docker is a set of "platform as a service" products that uses the OS-level +virtualization to deliver software in packages called containers. +Docker container provides a quick start environment to the developer to +run the out of box demos and build applications. + +The Docker image is based on Ubuntu 20.04.LTS and contains different +open source components like OpenCV, GStreamer, Python and pip packages +which are required to run the demos. The user can choose to install any +additional 3rd party applications and packages as required. + +.. _ai_64_edgeai_docker_build_ontarget: + +Building Docker image +====================== + +The `docker/Dockerfile` in the edge_ai_apps repo describes the recipe for +creating the Docker container image. Feel free to review and update it to +include additional packages before building the image. + +.. note:: + Building Docker image on target using the provided Dockerfile will take + about 15-20 minutes to complete with good internet connection. + Building Docker containers on target can be slow and resource constrained. + The Dockerfile provided will build on target without any issues but if + you add more packages or build components from source, running out of memory + can be a common problem. As an alternative we highly recommend trying + QEMU builds for cross-compiling the images for arm64 architecture on a PC + and then load the compiled image on target. + +Initiate the Docker image build as shown, + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/docker#./docker_build.sh + +Running the Docker container +============================ + +Enter the Docker session as shown, + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/docker#./docker_run.sh + +This will start a Ubuntu 20.04.LTS image based Docker container and the prompt +will change as below, + +.. code-block:: bash + + [docker] debian@beaglebone:/opt/edge_ai_apps# + + +The Docker container has been created in privilege mode, so that it has root +capabilities to all devices on the target system like Network etc. +The container file system also mounts the target file system of /dev, /opt to +access camera, display and other hardware accelerators the SoC has to offer. + +.. note:: + + It is highly recommended to use the docker_run.sh script to launch the + Docker container because this script will take care of saving any changes + made to the filesystem. This will make sure that any modifications to + the Docker filesystem including new package installation, updates to + some files and also command history is saved automatically and is + available the next time you launch the container. The container will + be committed only if you exit from the container explicitly. If you restart + the board without exiting container, any changes done from last saved state + will be lost. + +.. note:: + + After building and running the docker container, one needs to run + ``setup_script.sh`` before running any of the demo applications. + Please refer to :ref:`pub_edgeai_install_dependencies` for more details. + +.. _ai_64_edgeai_docker_additional_commands: + +Handling proxy settings +======================= + +If the board running the Docker container is behind a proxy server, the default +settings for downloading files and installing packages via apt-get will not work. +If you are running the board from TI network, docker build and run scripts will +automatically detect and configure necessary proxy settings + +For other cases, you need to modify the script ``/usr/bin/setup_proxy.sh`` +to add the custom proxy settings required for your network. + +Additional Docker commands +========================== + +.. note:: + This section is provided only for additional reference and not required to + run out-of-box demos + +**Commit Docker container** + +Generally, containers have a short life cycle. If the container has any local +changes it is good to save the changes on top of the existing Docker image. +When re-running the Docker image, the local changes can be restored. + +Following commands show how to save the changes made to the last container. +Note that this is already done automatically by ``docker_run.sh`` when you exit +the container. + +.. code-block:: bash + + cont_id=`docker ps -q -l` + docker commit $cont_id edge_ai_kit + docker container rm $cont_id + + +For more information refer: +`Commit Docker image <https://docs.docker.com/engine/reference/commandline/commit/>`_ + +**Save Docker Image** + +Docker image can be saved as tar file by using the command below: + +.. code-block:: bash + + docker save --output <pre_built_docker_image.tar> + +For more information refer here. +`Save Docker image <https://docs.docker.com/engine/reference/commandline/save/>`_ + +**Load Docker image** + +Load a previously saved Docker image using the command below: + +.. code-block:: bash + + docker load --input <pre_built_docker_image.tar> + +For more information refer here. +`Load Docker image <https://docs.docker.com/engine/reference/commandline/load/>`_ + +**Remove Docker image** + +Docker image can be removed by using the command below: + +.. code-block:: bash + + Remove selected image: + docker rmi <image_name/ID> + + Remove all image: + docker image prune -a + +For more information refer +`rmi reference <https://docs.docker.com/engine/reference/commandline/rmi/>`_ and +`Image prune reference <https://docs.docker.com/engine/reference/commandline/image_prune/>`_ + +**Remove Docker container** + +Docker container can be removed by using the command below: + +.. code-block:: bash + + Remove selected container: + docker rm <container_ID> + + Remove all container: + docker container prune + +For more information refer here. +`rm reference <https://docs.docker.com/engine/reference/commandline/rm/>`_ and +`Container Prune reference <https://docs.docker.com/engine/reference/commandline/container_prune/>`_ + +Relocating Docker Root Location +=============================== +The default location for Docker files is **/var/lib/docker**. Any Docker images +created will be stored here. This will be a problem anytime the SD card is +updated with a new targetfs. If a secondary storage (SSD or USB based storage) +is available, then it is recommended to relocate the default Docker root +location so as to preserve any existing Docker images. Once the relocation +has been done, the Docker content will not be affected by any future targetfs +updates or accidental corruptions of the SD card. + +The following steps outline the process for Docker root directory relocation +assuming that the current Docker root is not at the desired location. If the +current location is the desired location then exit this procedure. + +1. Run 'Docker info' command and inspect the output. Locate the line with + content **Docker Root Dir**. It will list the current location. + +2. To preserve any existing images, export them to .tar files for importing + later into the new location. + +3. Inspect the content under /etc/docker to see if there is a file by name + **daemon.json**. If the file is not present then create **/etc/docker/docker.json** + and add the following content. Update the 'key:value' pair for the key "graph" + to reflect the desired root location. If the file already exists, then make + sure that the line with "graph" exists in the file and points to the desired + target location. + +.. code-block:: json + + { + "graph": "/run/media/nvme0n1/docker_root", + "storage-driver": "overlay", + "live-restore": true + } + +In the configuration above, the key/value pair +**'"graph": "/run/media/nvme0n1/docker_root"'** defines the root location +**'/run/media/nvme0n1/docker_root'.** + +4. Once the daemon.json file has been copied and updated, run the following + commands + +.. code-block:: bash + + $ systemctl restart docker + $ docker info + +Make sure that the new Docker root appears under **Docker Root Dir** value. + +5. If you exported the existing images in step (2) then import them and they + will appear under the new Docker root. + +6. Anytime the SD card is updated with a new targetfs, steps (1), (3), and + (4) need to be followed. + +**Additional references** + +| https://docs.docker.com/engine/reference/commandline/images/ +| https://docs.docker.com/engine/reference/commandline/ps/ diff --git a/beaglebone-ai-64/edge_ai_apps/getting_started.rst b/beaglebone-ai-64/edge_ai_apps/getting_started.rst new file mode 100644 index 0000000000000000000000000000000000000000..93d960a4b6b78b7b147da493e5a5671c7c4b48f7 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/getting_started.rst @@ -0,0 +1,248 @@ +.. _ai_64_edgeai_getting_started: + +Getting Started +################# + +.. _ai_64_edgeai_getting_started_harware: + +Hardware setup +=============== + +BeagleBone® AI-64 has TI's TDA4VM SoC which houses dual core A72, high performance vision +accelerators, video codec accelerators, latest C71x and C66x DSP, high bandwidth +realtime IPs for capture and display, GPU, dedicated safety island and security +accelerators. The SoC is power optimized to provide best in class performance +for perception, sensor fusion, localization and path planning tasks in robotics, +industrial and automotive applications. + +For more details visit https://www.ti.com/product/TDA4VM + +.. _ai_64_edgeai_hw_requirements_eaik: + +BeagleBone® AI-64 +----------------- + +BeagleBone® AI-64 brings a complete system for developing artificial intelligence (AI) +and machine learning solutions with the convenience and expandability of the BeagleBone® +platform and the peripherals on board to get started right away learning and building +applications. With locally hosted, ready-to-use, open-source focused tool chains and +development environment, a simple web browser, power source and network connection +are all that need to be added to start building performance-optimized embedded +applications. Industry-leading expansion possibilities are enabled through +familiar BeagleBone® cape headers, with hundreds of open-source hardware examples +and dozens of readily available embedded expansion options available off-the-shelf. + +To run the demos on BeagleBone® AI-64 you will require, + + - BeagleBone® AI-64 + - USB camera (Any V4L2 compliant 1MP/2MP camera, Eg. Logitech C270/C920/C922) + - Full HD eDP/HDMI display + - Minimum 16GB high performance SD card + - 100Base-T Ethernet cable connected to internet + - UART cable + - External Power Supply or Power Accessory Requirements + + a. Nominal Output Voltage: 5VDC + b. Maximum Output Current: 5000 mA + +Connect the components to the SK as shown in the image. + +.. figure:: ./images/board_connections_bbai_64.jpg + :align: center + + BeagleBone® AI-64 for Edge AI connections + +.. _ai_64_edgeai_usb_camera: + +USB Camera +---------- + +UVC (USB video class) compliant USB cameras are supported on the BeagleBone® AI-64. +The driver for the same is enabled in linux image. The linux image has been tested with +C270/C920/C922 versions of Logitech USB cameras. Please refer to +:ref:`pub_edgeai_multiple_usb_cams` to stream from multiple USB cameras +simultaneously. + +.. _ai_64_edgeai_imx219_sensor: + +IMX219 Raw sensor +------------------ + +**IMX219 camera module** from **Raspberry pi / Arducam** is supported by BeagleBone® AI-64. +It is a 8MP sensor with no ISP, which can transmit raw SRGGB8 frames over CSI lanes at 1080p 60 fps. +This camera module can be ordered from +https://www.amazon.com/Raspberry-Pi-Camera-Module-Megapixel/dp/B01ER2SKFS +The camera can be connected to any of the 2 RPi zero 22 pin camera headers on BB AI-64 as +shown below + +.. figure:: + :scale: 20 + :align: center + + TODO: IMX219 CSI sensor connection with BeagleBone® AI-64 for Edge AI + +Note that the headers have to be lifted up to connect the cameras + +.. note:: To be updated + By default IMX219 is disabled. After connecting the camera you can enable it + by specifying the dtb overlay file in + ``/run/media/mmcblk0p1/uenv.txt`` as below, + + ``name_overlays=k3-j721e-edgeai-apps.dtbo k3-j721e-sk-rpi-cam-imx219.dtbo`` + + Reboot the board after editing and saving the file. + +Two RPi cameras can be connected to 2 headers for multi camera usecases + +Please refer :ref:`pub_edgeai_camera_sources` to know how to list all the cameras +connected and select which one to use for the demo. + +By default imx219 will be configured to capture at 8 bit, but it also supports +10 bit capture in 16 bit container. To use it in 10 bit mode, below steps are +required: + + - Modify the ``/opt/edge_ai_apps/scripts/setup_cameras.sh`` to set the + format to 10 bit like below + + .. code-block:: bash + + CSI_CAM_0_FMT='[fmt:SRGGB8_1X10/1920x1080]' + CSI_CAM_1_FMT='[fmt:SRGGB8_1X10/1920x1080]' + + - Change the imaging binaries to use 10 bit versions + + .. code-block:: bash + + mv /opt/imaging/imx219/dcc_2a.bin /opt/imaging/imx219/dcc_2a_8b.bin + mv /opt/imaging/imx219/dcc_viss.bin /opt/imaging/imx219/dcc_viss_8b.bin + mv /opt/imaging/imx219/dcc_2a_10b.bin /opt/imaging/imx219/dcc_2a.bin + mv /opt/imaging/imx219/dcc_viss_10b.bin /opt/imaging/imx219/dcc_viss.bin + + - Set the input format in the ``/opt/edge_ai_apps/configs/rpiV2_cam_example.yaml`` + as ``rggb10`` + + +Software setup +============== + +.. _ai_64_edgeai_prepare_sd_card: + +Preparing SD card image +----------------------- + +Download the ``bullseye-xfce-edgeai-arm64`` image from the links below and +flash it to SD card using `Balena etcher <https://www.balena.io/etcher/>`_ tool. + +- To use via SD card: `bbai64-debian-11.4-xfce-edgeai-arm64-2022-08-02-10gb.img.xz <bbai64-debian-11.4-xfce-edgeai-arm64-2022-08-02-10gb.img.xz>`_ +- To flash on eMMC: `bbai64-emmc-flasher-debian-11.4-xfce-edgeai-arm64-2022-08-02-10gb.img.xz <https://rcn-ee.net/rootfs/bb.org/testing/2022-08-02/bullseye-xfce-edgeai-arm64/bbai64-emmc-flasher-debian-11.4-xfce-edgeai-arm64-2022-08-02-10gb.img.xz>`_ + + +The Balena etcher tool can be installed either on Windows/Linux. Just download the +etcher image and follow the instructions to prepare the SD card. + +.. figure:: ./images/balena_etcher.png + :scale: 100 + :align: center + + Balena Etcher tool to flash SD card with Processor linux image Linux for Edge AI + +The etcher image is created for 16 GB SD cards, if you are using larger SD card, +it is possible to expand the root filesystem to use the full SD card capacity +using below steps + +.. code-block:: bash + + #find the SD card device entry using lsblk (Eg: /dev/sdc) + #use the following commands to expand the filesystem + #Make sure you have write permission to SD card or run the commands as root + + #Unmount the BOOT and rootfs partition before using parted tool + umount /dev/sdX1 + umount /dev/sdX2 + + #Use parted tool to resize the rootfs partition to use + #the entire remaining space on the SD card + #You might require sudo permissions to execute these steps + parted -s /dev/sdX resizepart 2 '100%' + e2fsck -f /dev/sdX2 + resize2fs /dev/sdX2 + + #replace /dev/sdX in above commands with SD card device entry + +.. _ai_64_edgeai_poweron_boot: + +Power ON and Boot +----------------- +Ensure that the power supply is disconnected before inserting the SD card. +Once the SD card is firmly inserted in its slot and the board is powered ON, +the board will take less than 20sec to boot and display a wallpaper as +shown in the image below. + +.. figure:: + :scale: 25 + :align: center + + TODO: BeagleBone® AI-64 wallpaper upon boot + +You can also view the boot log by connecting the UART cable to your PC and +use a serial port communications program. + +For **Linux OS minicom** works well. +Please refer to the below documentation on 'minicom' for more details. + +https://help.ubuntu.com/community/Minicom + +When starting minicom, turn on the colors options like below: + +.. code-block:: bash + + sudo minicom -D /dev/ttyUSB2 -c on + +For **Windows OS Tera Term** works well. +Please refer to the below documentation on 'TeraTerm' for more details + +https://learn.sparkfun.com/tutorials/terminal-basics/tera-term-windows + +.. note:: + Baud rate should be configured to 115200 bps in serial port communication + program. You may not see any log in the UART console if you connect to it + after the booting is complete or login prompt may get lost in between boot + logs, press ENTER to get login prompt + +As part of the linux systemd ``/opt/edge_ai_apps/init_script.sh`` is executed +which does the below, + + - This kills weston compositor which holds the display pipe. This step will + make the wallpaper showing on the display disappear and come back + - The display pipe can now be used by 'kmssink' GStreamer element while + running the demo applications. + - The script can also be used to setup proxies if connected behind a + firewall. + +Once Linux boots login as ``root`` user with no password. + +.. _ai_64_edgeai_connecting_remotely: + +Connect remotely +---------------- +If you don't prefer the UART console, you can also access the device with the +IP address that is shown on the display. + +With the IP address one can ssh directly to the board, view the contents and run +the demos. + +For best experience we recommend using VSCode which can be downloaded from +here. + +https://code.visualstudio.com/download + +You also require the "Remote development extension pack" installed in VSCode +as mentioned here: + +https://code.visualstudio.com/docs/remote/ssh + +.. figure:: + :scale: 90 + :align: center + + TODO: Microsoft Visual Studio Code for connecting to BeagleBone® AI-64 for Edge AI via SSH diff --git a/beaglebone-ai-64/edge_ai_apps/images/TDA4VM-SK-SD-Boot.png b/beaglebone-ai-64/edge_ai_apps/images/TDA4VM-SK-SD-Boot.png new file mode 100644 index 0000000000000000000000000000000000000000..0e5535c416d2f325887302a7ac6f81dda493781a Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/TDA4VM-SK-SD-Boot.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/balena_etcher.png b/beaglebone-ai-64/edge_ai_apps/images/balena_etcher.png new file mode 100644 index 0000000000000000000000000000000000000000..37518a24b38411768e9f982ca98b66af9ff51101 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/balena_etcher.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/board_connections_bbai_64.jpg b/beaglebone-ai-64/edge_ai_apps/images/board_connections_bbai_64.jpg new file mode 100644 index 0000000000000000000000000000000000000000..70fab59661d3f1c4bcef1f0bea08d2f00c708453 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/board_connections_bbai_64.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/board_connections_tda4vm_evm.jpg b/beaglebone-ai-64/edge_ai_apps/images/board_connections_tda4vm_evm.jpg new file mode 100644 index 0000000000000000000000000000000000000000..df79286995995be556c21fccdc332dc3e14227e2 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/board_connections_tda4vm_evm.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/board_connections_tda4vm_sk.jpg b/beaglebone-ai-64/edge_ai_apps/images/board_connections_tda4vm_sk.jpg new file mode 100644 index 0000000000000000000000000000000000000000..bfff3ff7e3829394f00a8d8544bcb77c27839d78 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/board_connections_tda4vm_sk.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/boot_wallpaper.jpg b/beaglebone-ai-64/edge_ai_apps/images/boot_wallpaper.jpg new file mode 100644 index 0000000000000000000000000000000000000000..ec67b1494862f25fb2d6fe8b7ca6dce9264de517 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/boot_wallpaper.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/csi_camera_connection.png b/beaglebone-ai-64/edge_ai_apps/images/csi_camera_connection.png new file mode 100644 index 0000000000000000000000000000000000000000..9e8505cdd176f44952449e64f8f16f13a1d3fab3 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/csi_camera_connection.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos.drawio b/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos.drawio new file mode 100644 index 0000000000000000000000000000000000000000..85349890e0048450b0e5862d0830a7d53fe4d1e0 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos.drawio @@ -0,0 +1 @@ +<mxfile host="Electron" modified="2021-05-19T15:45:06.473Z" agent="5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) draw.io/14.6.13 Chrome/89.0.4389.128 Electron/12.0.7 Safari/537.36" etag="dAgc-ZEs3_cQbep0zx-v" version="14.6.13" type="device" pages="3"><diagram name="CPP Demo Data Flow" id="XUTmI-InrPcYLR0GOj1Y">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</diagram><diagram name="Python Demo Data Flow" id="OihEl4gLeUUpuXvin1QQ">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</diagram><diagram id="ZmDZ4aD7BjtF5CznldHT" name="Page-3">zZVdT4MwFIZ/DZdLaMtw17JNTVzULIveVjgDDNCmK4P56y1ygBHmdIm6XdG+54s+5zS1mJeWN4rLaCECSCxqB6XFphalZEyY+VTKDhWbkloJVRyg1gnL+B0aR1TzOIBNz1ELkehY9kVfZBn4uqdxpUTRd1uLpF9V8hAGwtLnyVB9jgMdoUpsuzPcQhxGWHoyRkPKG2cUNhEPRLEnsZnFPCWErldp6UFS0Wu43Ob523y2XVDmJffqafVy9bAa1cnmp4S0R1CQ6d9N7daptzzJkdddtgZTxwcjP8YS8Ox61wANlcglhoHSUB5qI39t3O0f/jxpiZpZBJGCVjsTh9lp0y4cQzrBfdH11HVQi/ba2QZynKOwzd2xMgvEdQI6OkA3QGVIZQFUSWyLXRdRrGEpuV9ZC3PdjBbp1BSdErP8Eum36NzD6M6Hhg3QSAUjqYR/BBE5AyLM4vaHy3UHs0XoAYDuX/Fzjt3KSwRInQsjOB5OoNjoCx5Bh/4bQbPtHq1P297bz2Yf</diagram></mxfile> \ No newline at end of file diff --git a/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos_CPP_Demo_Data_Flow.png b/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos_CPP_Demo_Data_Flow.png new file mode 100644 index 0000000000000000000000000000000000000000..4fdfb28d92167d3cae1741b51a5ba54ee11bc7f5 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos_CPP_Demo_Data_Flow.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos_Python_Demo_Data_Flow.png b/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos_Python_Demo_Data_Flow.png new file mode 100644 index 0000000000000000000000000000000000000000..9d14585f006af0bf7a386d23761f0e538a93a8b6 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edge_ai_demos_Python_Demo_Data_Flow.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-image-classify.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-image-classify.jpg new file mode 100644 index 0000000000000000000000000000000000000000..863354e4527ce4a773eb5b55f7b097e3ad589745 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-image-classify.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-multi-input-multi-infer.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-multi-input-multi-infer.jpg new file mode 100644 index 0000000000000000000000000000000000000000..9e5b755d973ef0dbed96c336d75e69c88463346e Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-multi-input-multi-infer.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-object-detect.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-object-detect.jpg new file mode 100644 index 0000000000000000000000000000000000000000..9a51ce21ac64eb5462a6ba0bfcf5141481367ac2 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-object-detect.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image1.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image1.jpg new file mode 100644 index 0000000000000000000000000000000000000000..d5f55221c72fb718a8cb028c208dd7b31fce6209 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image1.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image2.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image2.jpg new file mode 100644 index 0000000000000000000000000000000000000000..7a2e8bbf99660335ca2324d84e38180c5c374f1b Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image2.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image3.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image3.jpg new file mode 100644 index 0000000000000000000000000000000000000000..bbf27eb8aff1ddb0fbb1081cfdd59b988557387f Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-overview-image3.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-components.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-components.png new file mode 100644 index 0000000000000000000000000000000000000000..aed3fe95c2175c71472eefa3152237dd2b5b30b8 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-components.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-feature.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-feature.png new file mode 100644 index 0000000000000000000000000000000000000000..5f8b57e616809485873e53041a6d431218c0b713 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-feature.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-overview.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-overview.png new file mode 100644 index 0000000000000000000000000000000000000000..1d8d1697abb7d84ceebb14ce88e338519ac28ab2 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-overview.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-programming-env.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-programming-env.png new file mode 100644 index 0000000000000000000000000000000000000000..ae1540213003af3d533889750f2781dc373578da Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-programming-env.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-roadmap.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-roadmap.png new file mode 100644 index 0000000000000000000000000000000000000000..19e35d27b2cd6cae2384badbda5a74cbc126f879 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-sdk-roadmap.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai-single-input-multi-infer.jpg b/beaglebone-ai-64/edge_ai_apps/images/edgeai-single-input-multi-infer.jpg new file mode 100644 index 0000000000000000000000000000000000000000..75080e0fa4bc5fef05614c56a7c7f00255284651 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai-single-input-multi-infer.jpg differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_human_pose.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_human_pose.png new file mode 100644 index 0000000000000000000000000000000000000000..c87fbac45194a3f456d673797b4db47054fe1161 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_human_pose.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_image_classification.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_image_classification.png new file mode 100644 index 0000000000000000000000000000000000000000..b889d3a0248025d10a1e6f75f8515bc1dc7134ca Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_image_classification.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_imx390_camera_source.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_imx390_camera_source.png new file mode 100644 index 0000000000000000000000000000000000000000..505f411e4e7d70d2ad90688ded00403004547626 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_imx390_camera_source.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_object_detection.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_object_detection.png new file mode 100644 index 0000000000000000000000000000000000000000..a7f57db96e090b5ab7bbd87391a242f2fd27c4e2 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_object_detection.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_ov5640_camera_source.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_ov5640_camera_source.png new file mode 100644 index 0000000000000000000000000000000000000000..217c62592655a4cd4cc10a2f82694d60e55f847b Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_ov5640_camera_source.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_rpi_camera_source.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_rpi_camera_source.png new file mode 100644 index 0000000000000000000000000000000000000000..0d7fc30d7d09cdb14b6c0640b10d8d1c98e6ba42 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_rpi_camera_source.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_rtsp_source.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_rtsp_source.png new file mode 100644 index 0000000000000000000000000000000000000000..418a3611a45e539f14cba32ea0839a8006b2e2ad Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_rtsp_source.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_semantic_segmentation.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_semantic_segmentation.png new file mode 100644 index 0000000000000000000000000000000000000000..fba820e60a2e990719a059f8af9e13c8713fc098 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_semantic_segmentation.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_video_output.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_video_output.png new file mode 100644 index 0000000000000000000000000000000000000000..8230859d00c2f69167c7a8bfc0375608fd10dd0b Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_video_output.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/edgeai_video_source.png b/beaglebone-ai-64/edge_ai_apps/images/edgeai_video_source.png new file mode 100644 index 0000000000000000000000000000000000000000..826c14ae55df9f3af566ee7af4fbedb0063625ce Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/edgeai_video_source.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/favicon.ico b/beaglebone-ai-64/edge_ai_apps/images/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..d14b5d9868819d951cb3953b5761bc8ca323a24e Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/favicon.ico differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/j721esk-ap.png b/beaglebone-ai-64/edge_ai_apps/images/j721esk-ap.png new file mode 100644 index 0000000000000000000000000000000000000000..f01438d1389a6cb18c95da5548e5c9ca33c50451 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/j721esk-ap.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/model_downloader.png b/beaglebone-ai-64/edge_ai_apps/images/model_downloader.png new file mode 100644 index 0000000000000000000000000000000000000000..8f029a88a0ebd72a91f91f6490759cdab5d10359 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/model_downloader.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/open-src-components.png b/beaglebone-ai-64/edge_ai_apps/images/open-src-components.png new file mode 100644 index 0000000000000000000000000000000000000000..0bc8e05b05a7f158dce9163fcf11aea514661f5e Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/open-src-components.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/perf_plots.png b/beaglebone-ai-64/edge_ai_apps/images/perf_plots.png new file mode 100644 index 0000000000000000000000000000000000000000..9c29fc45408039f5b3c29b6ad09585b9dd219e7d Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/perf_plots.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/rpi_camera_connection.png b/beaglebone-ai-64/edge_ai_apps/images/rpi_camera_connection.png new file mode 100644 index 0000000000000000000000000000000000000000..2bcfcf4bfc1062482f4c05e25de797d067a12e40 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/rpi_camera_connection.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/vs_code.png b/beaglebone-ai-64/edge_ai_apps/images/vs_code.png new file mode 100644 index 0000000000000000000000000000000000000000..a8aa2f0788a812adbfab574c79affa0ad0dbcc88 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/vs_code.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/images/wifi-oob-iw-command.png b/beaglebone-ai-64/edge_ai_apps/images/wifi-oob-iw-command.png new file mode 100644 index 0000000000000000000000000000000000000000..85d23a3ec535ce2d8d4bd0ca04e8d3cf4d671304 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/images/wifi-oob-iw-command.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/index.rst b/beaglebone-ai-64/edge_ai_apps/index.rst new file mode 100644 index 0000000000000000000000000000000000000000..12c74feeaebca64cfa75cc69ce6fa5b2f17f4986 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/index.rst @@ -0,0 +1,19 @@ +.. _ai_64_edgeai_home: + +Edge AI +======== + +.. toctree:: + :maxdepth: 1 + + getting_started + running_simple_demos + inference_models + configuration_file + running_advance_demos + docker_environment + data_flows + performance_visualizer + sdk_components + datasheet + test_report diff --git a/beaglebone-ai-64/edge_ai_apps/inference_models.rst b/beaglebone-ai-64/edge_ai_apps/inference_models.rst new file mode 100644 index 0000000000000000000000000000000000000000..91fbc396034d6688cec98b31f4cf8c87c214cfde --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/inference_models.rst @@ -0,0 +1,125 @@ +.. _ai_64_edgeai_inference_models: + +DL models for Edge Inference +############################## + +Model Downloader Tool +===================== + +`TI Model Zoo <https://github.com/TexasInstruments/edgeai-modelzoo>`_ +is a large collection of deep learning models validated to work on TI processors +for edge AI. It hosts several pre-compiled model artifacts for TI hardware. + +Use the **Model Downloader Tool** to download more models on target as shown, + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps# ./download_models.sh + +The script will launch an interactive menu showing the list of available, +pre-imported models for download. The downloaded models will be placed +under ``/opt/model_zoo/`` directory + +.. figure:: ./images/model_downloader.png + :align: center + + Model downloader tool menu option to download models + +The script can also be used in a non-interactive way as shown below: + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps# ./download_models.sh --help + +.. _ai_64_edgeai_import_custom_models: + +Import Custom Models +==================== + +The BeagleBone® AI-64 Linux for Edge AI also supports importing pre-trained custom +models to run inference on target. + +The SDK makes use of pre-compiled DNN (Deep Neural Network) models and performs +inference using various OSRT (open source runtime) such as TFLite runtime, +ONNX runtime and Neo AI-DLR. In order to infer a DNN, SDK expects the DNN and +associated artifacts in the below directory structure. + +.. code-block:: bash + + TFL-OD-2010-ssd-mobV2-coco-mlperf-300x300 + │ + ├── param.yaml + │ + ├── artifacts + │  ├── 264_tidl_io_1.bin + │  ├── 264_tidl_net.bin + │  ├── 264_tidl_net.bin.layer_info.txt + │  ├── 264_tidl_net.bin_netLog.txt + │  ├── 264_tidl_net.bin.svg + │  ├── allowedNode.txt + │  └── runtimes_visualization.svg + │ + └── model + └── ssd_mobilenet_v2_300_float.tflite + +DNN directory structure +----------------------- + +Each DNN must have the following 3 components: + +#. **model**: This directory contains the DNN being targeted to infer +#. **artifacts**: This directory contains the artifacts generated after the + compilation of DNN for SDK, and described in :ref:`pub_edgeai_compile_artifacts` +#. **param.yaml**: A configuration file in yaml format to provide basic + information about DNN, and associated pre and post processing parameters. + More details can be find :ref:`pub_edgeai_params` + +.. _ai_64_edgeai_params: + +Param file format +----------------- + +Each DNN has it’s own pre-process, inference and post-process +parameters to get the correct output. This information is typically available in +the training software that was used to train the model. In order to convey this +information to the SDK in a standardized fashion, we have defined a set of +parameters that describe these operations. These parameters are in the +param.yaml file. + +Please see sample yaml files for various tasks such as image classification, +semantic segmentation and object detection in +`edgeai-benchmark examples <https://github.com/TexasInstruments/edgeai-benchmark/tree/master/examples/configs/yaml>`_. +Descriptions of various parameters are also in the yaml files. If users want to +bring their own model to the SDK, then they need to prepare this information +offline and get to the SDK. In next section we explain how to prepare this +information + +.. _ai_64_edgeai_compile_artifacts: + +DNN compilation for SDK – Basic Instructions +-------------------------------------------- + +The BeagleBone® AI-64 Linux for Edge AI supports three different runtimes to infer +a DNN, and user can choose a run time depending on the format of DNN. +We recommend users to use different run times and compare the performance and +select the one which provides best performance. User can find the steps to +generate the artifacts directory at +`Edge AI TIDL Tools <https://github.com/TexasInstruments/edgeai-tidl-tools/blob/master/examples/osrt_python/README.md#model-compilation-on-pc>`_ + +DNN compilation for SDK – Advanced Instructions +----------------------------------------------- + +For beginners who are trying to compile models for the SDK, we recommend the +basic instructions given in the previous section. However, DNNs have lot of +variety and some models may need a different kind of preprocessing or postprocessing +operations. In order to help customers deal with different kinds of models, we +have prepared a model zoo in the repository +`edgeai-modelzoo <https://github.com/TexasInstruments/edgeai-modelzoo>`_ + +For the DNNs which are part of TI’s model zoo, one can find the compilation +settings and pre-compiled model artifacts in +`edgeai-benchmark <https://github.com/TexasInstruments/edgeai-benchmark>`_ +repository. Instructions are also given to compile custom models. +When using `edgeai-benchmark`_ for model compilation, the yaml file is automatically +generated and artifacts are packaged in the way SDK understands. Please +follow the instructions in the repository to get started. diff --git a/beaglebone-ai-64/edge_ai_apps/performance_visualizer.rst b/beaglebone-ai-64/edge_ai_apps/performance_visualizer.rst new file mode 100644 index 0000000000000000000000000000000000000000..0813bc68f1cbf2359308ee9d910b5a4a2d8accd0 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/performance_visualizer.rst @@ -0,0 +1,103 @@ +.. _ai_64_edgeai_perf_viz_tool: + +Performance Visualization Tool +============================== + +The performance visualization tool can be used to view all the performance +statistics recorded when running the edge AI C++ demo application. This includes +the CPU and HWA loading, DDR bandwidth, Junction Temperatures and FPS obtained. +Refer to :ref:`pub_edgeai_available_statistics` for details on the performance +metrics available to be plotted. + +This tool works as follows: + + - Logging: When running the application, the performance statistics can be + recorded and stored in log files. This is done automatically when running + the C++ application, but the Python application does not generate logs. + However a standalone binary executable is provided that can be run in parallel + with the Python application, which will generate these performance logs. + - Visualization: There is a Python script which parses these logs and plots graphs, + which can be easily viewed by a visiting a URL in any browser. This script uses + Streamlit package to update the graphs in real-time, as the Edge AI application + runs in parallel. However, since Streamlit is not supported in the SDK out of box, + this script needs to run on docker. Please refer to :ref:`pub_edgeai_docker_env` + for building and running a docker container. + + +Generating Performance Logs +=========================== +Each log file contains real-time values for some performance metrics, averaged +over a 2s window. The temperature sensor values are sampled in real time, every 2s. +The performance visualization tool then parses these log files one by one based on +the modification timestamps. + + +The edge AI C++ demo will automatically generate log files and store them in the directory +``../perf_logs``, that is, one level up from where the C++ app is run. For +example, if the app is run from ``edge_ai_apps/apps_cpp``, the logs will be +stored in ``edge_ai_apps/perf_logs``. + +Similarly, there is a binary executable that can be compiled that does the same logging +standalone. The source for this is available under ``edge_ai_apps/scripts/perf_stats/``. +The README.md file has simple instructions to build and run this standalone logger binary. +After building it, use following command to print the statistics on the terminal as well +as save them in log files that can be parsed. + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/scripts/perf_stats/build# ../bin/Release/ti_perfstats -l + + +Running the Visualization tool +============================== + +To use this tool, simply start a docker session and then run the command given +below. This script expects some log files to be present in the directory +``edge_ai_apps/perf_logs`` after running any C++ demo. One can also bring up this +tool while running the demo but it might affect the performance of the demo itself +as it consumes a bit of ARM cycles during launch but stabilizes over a certain +duration. + +.. code-block:: bash + + [docker] debian@beaglebone:/opt/edge_ai_apps# streamlit run scripts/perf_vis.py --theme.base="light" + +This script also accepts the log directory as a command line argument as follows: + +.. code-block:: bash + + [docker] debian@beaglebone:/opt/edge_ai_apps# streamlit run scripts/perf_vis.py --theme.base="light" -- -D <path/to/logs/directory/> + +A network URL can be seen in the terminal output. The graphs can be viewed by +visiting this URL in any browser. The plotted graphs will keep updating based +on the available log files. + +.. figure:: ./images/perf_plots.png + :align: center + :scale: 75% + + Performance visualizer dashboard showing CPU and HWA loading, DDR bandwidth, + Junction Temperatures and the FPS obtained + +To exit press Ctrl+C in the terminal. + +.. _ai_64_edgeai_available_statistics: + +Available options +----------------- + +Average frames per second (FPS) recorded by the application is displayed by +default. Using the checkboxes in the sidebar, one can select which performance +metrics to view. There are 14 metrics available to be plotted, as seen from +the above image: + + - CPU Load: Total loading for the A72(mpu1_0), R5F(mcu2_0/1), C66x(c6x_1/2) and C71x(c7x_1) DSPs. + - HWA Load: Loading (percentage) for the various available hardware accelerators. + - DDR Bandwidth: Average read, write and total bandwidth recorded in the previous 2s interval. + - Junction Temperatures: The live temperatures recorded at various junctions + - Task Table: A separate graph for each cpu showing the loading due to various tasks running on it. + - Heap Table: A separate graph for each cpu showing the heap memory usage statistics. + +For the first three metrics, there is a choice to view line graphs with a 30s +history or bar graphs with only the real-time values. The remaining eleven have +real-time bar graphs as the only option. \ No newline at end of file diff --git a/beaglebone-ai-64/edge_ai_apps/running_advance_demos.rst b/beaglebone-ai-64/edge_ai_apps/running_advance_demos.rst new file mode 100644 index 0000000000000000000000000000000000000000..ec7ff25728186c2f3d538c2a2ad6e491d1ec56e0 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/running_advance_demos.rst @@ -0,0 +1,115 @@ +.. _ai_64_edgeai_running_advance_demos: + +Running Advance demos +####################### + +The same Python and C++ demo applications can be used to run multiple inference +models and also work with multiple inputs with just simple changes in the config +file. + +From a repo of input sources, output sources and models one can define advance +dataflows which connect them in various configurations. Details on configuration +file parameters can be found in :ref:`pub_edgeai_configuration` + +Single input multi inference demo +--------------------------------- + +Here is an example of a single-input, multi-inference demo which takes a camera +input and run multiple networks on each of them. + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/apps_python# ./app_edgeai.py ../configs/single_input_multi_infer.yaml + +Sample output for single input, multi inference demo is as shown below, + +.. figure:: ./images/edgeai-single-input-multi-infer.jpg + :align: center + + Sample output showing single input, mutli-inference output + +We can specify the output window location and sizes as shown in the +configuration file, + +.. code-block:: yaml + + flows: + flow0: + input: input0 + models: [model0, model1, model2, model3] + outputs: [output0, output0, output0, output0] + mosaic: + mosaic0: + width: 800 + height: 450 + pos_x: 160 + pos_y: 90 + mosaic1: + width: 800 + height: 450 + pos_x: 960 + pos_y: 90 + mosaic2: + width: 800 + height: 450 + pos_x: 160 + pos_y: 540 + mosaic3: + width: 800 + height: 450 + pos_x: 960 + pos_y: 540 + +Multi input multi inference demo +-------------------------------- + +Here is an example of a multi-input, multi-inference demo which takes a camera +input and video input and runs multiple networks on each of them. + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/apps_python# ./app_edgeai.py ../configs/multi_input_multi_infer.yaml + +Sample output for multi input, multi inference demo is as shown below, + +.. figure:: ./images/edgeai-multi-input-multi-infer.jpg + :align: center + + Sample output showing multi-input, mutli-inference output + +We can specify the output window location and sizes as shown in the +configuration file, + +.. code-block:: yaml + + flows: + flow0: + input: input0 + models: [model1, model2] + outputs: [output0, output0] + mosaic: + mosaic0: + width: 800 + height: 450 + pos_x: 160 + pos_y: 90 + mosaic1: + width: 800 + height: 450 + pos_x: 960 + pos_y: 90 + flow1: + input: input1 + models: [model0, model3] + outputs: [output0, output0] + mosaic: + mosaic0: + width: 800 + height: 450 + pos_x: 160 + pos_y: 540 + mosaic1: + width: 800 + height: 450 + pos_x: 960 + pos_y: 540 diff --git a/beaglebone-ai-64/edge_ai_apps/running_simple_demos.rst b/beaglebone-ai-64/edge_ai_apps/running_simple_demos.rst new file mode 100644 index 0000000000000000000000000000000000000000..2f02583b9f66ca2218ec88f5760f43459536f8b1 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/running_simple_demos.rst @@ -0,0 +1,94 @@ +.. _ai_64_edgeai_running_simple_demos: + +Running Simple demos +##################### + +This chapter describes how to run Python and C++ demo applications in +`edge_ai_apps <https://git.ti.com/cgit/edgeai/edge_ai_apps>`_ with live +camera and display. + +.. note:: + + Please note that the Python demos are useful for quick prototyping + while C++ demos are similar by design but tuned for performance. + +.. _ai_64_edgeai_python_demos: + +Running Python based demo applications +====================================== + +Python based demos are simple executable scripts written for image +classification, object detection and semantic segmentation. Demos are +configured using a YAML file. Details on configuration file parameters can +be found in :ref:`pub_edgeai_configuration` + +Sample configuration files for out of the box demos can be found in +``edge_ai_apps/configs`` this folder also contains a template config file +which has brief info on each configurable parameter ``edge_ai_apps/configs/app_config_template.yaml`` + +Here is how a Python based image classification demo can be run, + +.. code-block:: bash + :linenos: + + # go to edge-ai-apps folder + debian@beaglebone:~$ cd /opt/edge_ai_apps/apps_python + + # enable root (password: temppwd) + debian@beaglebone:~$ sudo su + [sudo] password for beaglebone: + + # use edge-ai-apps + debian@beaglebone:/opt/edge_ai_apps/apps_cpp# sudo ./app_edgeai.py ../configs/image_classification.yaml + +The demo captures the input frames from connected USB camera and passes +through pre-processing, inference and post-processing before sent to display. +Sample output for image classification and object detection demos are as below, + +.. |logo1| image:: ./images/edgeai-image-classify.jpg + :align: middle +.. |logo2| image:: ./images/edgeai-object-detect.jpg + :align: middle + ++---------+---------+ +| |logo1| | |logo2| | ++---------+---------+ + +To exit the demo press Ctrl+C. + +.. _ai_64_edgeai_cpp_demos: + +Building and running C++ based demo applications +================================================ + +C++ apps needs to be built directly on target and requires header files of +different deep-learning runtime framework and its dependencies which are +installed in the setup script. The setup script builds the C++ apps when +executed. However one can also follow below steps to clean build C++ apps + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/apps_cpp# rm -rf build bin lib + debian@beaglebone:/opt/edge_ai_apps/apps_cpp# mkdir build + debian@beaglebone:/opt/edge_ai_apps/apps_cpp# cd build + debian@beaglebone:/opt/edge_ai_apps/apps_cpp/build# cmake .. + debian@beaglebone:/opt/edge_ai_apps/apps_cpp/build# make -j2 + +Run the demo once the application is successfully built + +.. code-block:: bash + + debian@beaglebone:/opt/edge_ai_apps/apps_cpp# ./bin/Release/app_edgeai ../configs/image_classification.yaml + +To exit the demo press Ctrl+C. + +.. note:: + + Both Python and C++ applications are similar by construction and can accept + the same config file and command line arguments + +.. note:: + The C++ apps built on Yocto Linux may not run in Docker as there could be + a mismatch in Glib and other related tools. So its **highly recommended** to + rebuild the C++ apps within the Docker environment. + diff --git a/beaglebone-ai-64/edge_ai_apps/sdk_components.rst b/beaglebone-ai-64/edge_ai_apps/sdk_components.rst new file mode 100644 index 0000000000000000000000000000000000000000..36b1431cec980a2a69a43ea34f41656e27f019c4 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/sdk_components.rst @@ -0,0 +1,111 @@ +.. _ai_64_edgeai_sdk_components: + +SDK Components +################ + +The BeagleBone® AI-64 Linux for Edge AI can be divided into 3 parts, Applications, +BeagleBone® AI-64 Linux and Processor SDK RTOS. Users can get the latest application +updates and bug fixes from the public repositories (GitHub and git.ti.com) +which aligns with the SDK releases done quarterly. One can also build every component +from source by following the steps here, :ref:`pub_edgeai_sdk_development_flow` + +.. figure:: ./images/edgeai-sdk-components.png + :scale: 50 + :align: center + + BeagleBone® AI-64 Linux for Edge AI components + +.. _ai_64_edgeai_applications: + +Edge AI Applications +==================== + +The edge AI applications are designed for users to quickly evaluate various Deep Learning +networks on TDA4 SoC. The user can run standalone examples and Jupyter notebook applications +to evaluate inference models either from `TI Edge AI Model Zoo <https://github.com/TexasInstruments/edgeai-modelzoo>`_ +or a custom network. +Once a network is finalized for performance and accuracy it can also be +easily integrated in a typical capture-inference-display usecase using example +GStreamer based applications for rapid prototyping and deployment. + +.. _ai_64_edgeai_tidl_tools: + +edgeai-tidl-tools +----------------- + +This application repository provides standalone Python and C/C++ examples to +quickly evaluate inference models using TFLite, ONNX and NeoAI-DLR runtime +using file based inputs. It also houses the Jupyter notebooks similar to +`TI Edge AI Cloud <https://dev.ti.com/edgeai/>`_ which can be executed right on +the TDA4VM Starter Kit. + +For more details on using this application repo please refer to the documentation +and source code found here: https://github.com/TexasInstruments/edgeai-tidl-tools + +.. _ai_64_edgeai_modelzoo: + +edgeai-modelzoo +--------------- + +This repo provides collection of example Deep Neural Network (DNN) Models +for various computer vision tasks. A few example models are packaged as part of +the SDK to run out-of-box demos. More can be downloaded using a download script +made available in the edge_ai_apps repo. + +For more details on the the pre-imported models and related documentation please visit: +https://github.com/TexasInstruments/edgeai-modelzoo + + +.. _ai_64_edgeai_apps: + +edge_ai_apps +------------ +These are plug-and-play Deep Learning applications which support running open +source runtime frameworks such as TFLite, ONNX and NeoAI-DLR with a live camera +and display. They help connect realtime camera, video or RTSP sources to DL +inference to live display, bitstream or RTSP sinks. + +The latest source code with fixes can be pulled from: https://git.ti.com/cgit/edgeai/edge_ai_apps + +.. _ai_64_edgeai_gst_plugins: + +edgeai-gst-plugins +------------------ +This repo provides the source of custom GStreamer plugins which helps offload +tasks to TDA4 hardware accelerators and advanced DSPs with the help of +edgeai-tiovx-modules. The repo gets downloaded, built and installed as part +of the :ref:`pub_edgeai_install_dependencies` step. + +Source code and documentation: https://github.com/TexasInstruments/edgeai-gst-plugins + +.. _ai_64_edgeai_tiovx_modules: + +edgeai-tiovx-modules +-------------------- +This repo provides OpenVx modules which help access underlying hardware +accelerators in the TDA4 SoC and serves as a bridge between GStreamer +custom elements and underlying OpenVx custom kernels. The repo gets downloaded, +built and installed as part of the :ref:`pub_edgeai_install_dependencies` step. + +Source code and documentation: https://github.com/TexasInstruments/edgeai-tiovx-modules + +.. _ai_64_edgeai_psdk_rtos: + +Processor SDK RTOS +================== + +The BeagleBone® AI-64 Linux for Edge AI gets all the HWA drivers, optimized libraries, OpenVx framework +and more from Processor SDK RTOS + +For more information visit `Processor SDK RTOS Getting Started Guide <https://software-dl.ti.com/processor-sdk-rtos/esd/docs/latest/rtos/index_overview.html>`_. + + +.. _ai_64_edgeai_psdk_linux: + +BeagleBone® AI-64 Linux +=================== + +The BeagleBone® AI-64 Linux for Edge AI gets all the Linux kernel, filesystem, device-drivers and more +from BeagleBone® AI-64 Linux + +For more information visit `BeagleBone® AI-64 Linux Software Developer's Guide <https://software-dl.ti.com/jacinto7/esd/processor-sdk-linux-jacinto7/08_02_00_03/exports/docs/devices/J7/linux/index.html>`_. diff --git a/beaglebone-ai-64/edge_ai_apps/static/theme_overrides.css b/beaglebone-ai-64/edge_ai_apps/static/theme_overrides.css new file mode 100644 index 0000000000000000000000000000000000000000..27fb69dd258d4bb229f4985c96340e6a63a253f7 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/static/theme_overrides.css @@ -0,0 +1,233 @@ + +/* Open Sans Latin + base64 encode done on opensans_latin_v13.woff2 */ +@font-face { + font-family: 'Open Sans'; + font-style: normal; + font-weight: 400; + src: url('data:font/woff2;base64,d09GMgABAAAAADzUABIAAAAAhjgAADxwAAEZmgAAAAAAAAAAAAAAAAAAAAAAAAAAGhYbDBx0BmAAgUwIgSIJjzQREAqBpESBjUIBNgIkA4ZsE70QC4M8AAQgBYIyB4QoDIIJG8J3CQg7WxX5j0PthLRbJVvedpUoKlTzFUWwcUCG52fZ/////3lJxxAlWAMiaNd12/9PqJmJKW5iyEExieloQ9KxVuZAQ07T1g7vfgVloomBA/UEiuEMCGVsPBsLc9rD3dPKhTisOejgNULCLyjxoeYlBGuZsjRKads6GS6b8QRDIttvKSa68bAbLrwP31TZsuGVyrv7lPcTS4bG9tKpUh76wlIon84X/sDUs5lHxlTP/if+Ns2eosU/nf/Bfw/riGqqb+ffLOvzVA9JiiYPkV3xr7qrZ/bum4pAxULkIPkZgBBQJu/2AG2zs1GxcgYq2FMQpj1RETESsbAKmzJyopiJzNnDKkTX5pwurDm2/l+kLvK/+x/AP6A2ZgrZRvjS1KSQBer9XuGyk3Iw3zfX+1eWZFmG4CQDj7CYnr8CXBV4ueyfuup7tsiSJbYtuxH/jmfaSz7qPupZoCCZ+kF0QXRpQIvhcQTov6omea3KKY0pFP4oK73IomFDl+ISgLSDg2T3m9LqWHnZhsdUX6Zbs2x9S8bARE7Ay+y7Qf/nllVniRftubDauc4suapKNQPzE8citiadRhUeQAB2Or6gcE01zLrqvv3azJyYfdvFLJRBRNN+rORtiKb4r90jEQolkS9CXdeapUUtovJYdjvKac7KcpzMaA1l/uGrT4aVVrGvMVAuLrBgPpJTtAv0AsQv4mpVYhTdrISsR3HVOUDRuJh4nIdPy752XK3pgwsJHaOiCfRBCMYnBZa2NQIs0MJBCLKQcLAUFiLHsSb1NBwDuVM++S0pbMpCb4wnNT80UzMTE9QxNEqUXUj+XtWyBRikfCGFLhWNnelMh67A+x+BHyAogRS0C1I6L0lJt9RGUJGUzgZAjQekU04ENxHSBYHaRMdLMVd257vKTeW69pbnoi9NTzZ4puUyQp2QCOOWBcHlBVHZINQPsqpeQjArJqPxaQKweFrTR1P/k+Xq+/E66DKZ2KGI2ZjJZNOIfxEploB4nMfYbCEuPnLbpwkSihwCCsbniQTAguiPVErkU4mUNL+UgcJFiymBc06kZQAgwjK+x5hwzplJMQDIJhWEDSTOGJKlWpoIMEUadBm6RwiOj8PFkZ74BMmB3NPHvuQZdmtwhPKxaCIC1O5pSECsQvXqWxLjy0SwLDAMtWW45JcEf9qVQU4AAOvKcpvthncBrrn3f6P2TFH5pL6ErAoIPAgbyH0bD2BVQtHJQvXeU9iytiHI6oIFw3FX7KYd/ws48OV6eVger6oIUYdoQXQh+hBbCAZCg0ws5P//30+hOw99btm1DAYcrQBRhWi8jM0FaH419ud9hVS/0ut62vSv6Q/T11+iySYajVNbtUV/b5/exPSeyXTDklT41IpZiv4R2m3Dfcp9KWoG9CMAOmZLgoOTwv9t/XjG7O9no/SNV51dlkUn2AXcpNQTLI0adKlSGjJgNWg3YlhoCCEQH+Dv5+vj7eXp4Y5zw7piXJyd0I6nHeztbG2srSxPIeDmZidNDQ30YVA9XR1tprpw1uhWsieLKhwIhGlDCFAphITMFTNSlJXCvFS7BnGYV1RMYWUSk86rwpJKpWQBZbBSCbm2Z1laCiZ6RfcBbuU87snz7Dx7R5L4WZIjvKrod+rtYCMx9O8mwhLRGcX3ibFESNJCqbD2cGBXU/ncuJtPLlOhOxfmmcPUG5mPZuIouu1w9CKPimlUnXUEk4VEgpJLUlkHBxWtYuH7JsyXkXXj7AS4sDsaU8uIxeWw2Z6h9iA3uWnrBnnWSoOFDnWyWxohn666Da3a54egEGCM4tGe/q6VLupikmKFCY7LkmodtkTc48IxKTyS9g+Eon9/U0Gfmor/91YEWgYeaf9eEUU9e7fPaySLVll/ZPSApEe2n04/31YHDfx09bMcma9lC2vvLpFPB/ysabdbsZqgVGlqkCjKQZaCw9RL5J8Wfq78fbpRRBgkAAxoN+77EOTWB8wGH4agD/WBsmqI94A+FBiQym357qjcbDkjVNVS8mFxqGkbhEW5Q0rbxes+9DWsbK1TFiIP2X/hky3m5JlDhoxIiRi5oUREAgQUJuwkwnKXsPYD82bP2nAFm5nLbZFniAVCw4pU+qg9HOyqPE6GeWwEVpAL1OD4ReXMWGnlZg8RK1l6TAzjjnw3zNBtlCkyLGpSVg1rGp+rMhHlAiC6HaUwo4ZUViqrXkR1biihit4A5P8VJcWHBXYcMDHm+VEtaISkshedHnqp6MBKe1ILH2yAviHdNTbXS94mvwv/XjJusom6TmqUVBr6dCPYTMTqM4WFYrEPaeGMiEBczEg9JsYNFvESiiy+aBSdP6UREzNPxAMb5yCrnRydl5oLRghNYHpYqm5Mgj6bGcLtSEgUSpzEqZvZQFUoCd/491qYH9vIUigQkxnUr0rZ1DjE2gzGWOAk0qO4y+JrS01dMjSQ4BtaLQQN1jAHBNY3Z6by+icjUAwNoFPEHxGQX4a1QACbwPpnIaq0UdSymS9t3E1qlrFMKJ/BEWWA/Wb4dE4kRd6bmY5kiLfD64zt8+WeHB+cZTDDLXesW9PnCSGUaC/E0/g8RZElPKlHYd4R7O5c/qbimL+pNfRGnjFI1Yl+T0jLaWSclz9Gb1iMoXN8hCL0wWchLqS69sXHGrHBDM6USOEEjrLDEcJdS7xNCJA/n3BSiMbMyQ2EY0xIu4G4X+P7uAO45rs8N8kzl2cvL4xlqOUwSd5HLl85EA942MMMthyFVM0aP+LaJViU0jrK0n3ILadidX5Df0hDFXwCIWJ0r4T58QM6uleTuOxYSd5zM7DK9LQcKM+QeJzGwd2LDM6BRtSKpoM5txsh5UZ5vkuBS74pKpYGVnlVvQ5o+QbwNE+9qX3yYO35Jmz+n5HIpUfJ2vyfcGnkNDFvIWM3+l0SAJ48tws8taOvAL+PojsDoRsZILghY5rGbIwyhURLrWp55YC9KERyteUrrJvBxgK83RyXPcVHMCoV1zkeyAymKLA4qBuCE17rUFLPeFWiZKX9NFIPkWfjuZHm4voKGHuz5gt7xXpTvCZPhpARHTMjea8/wnPKhOSEpZhhrY3kuKgm88Euz4ZRsxiLqrUUXDCM2Sh/KjJmOI83BylGUktaOGk4F61UfJ35fCC5yHDGsBjkhYxBFaw8KhGxVNLhVakk7yq4dSzlWFZijCc1zkeTkYwYnyJoLlobOWOUa2uxks1KRCtDLiVjk7zikMsiyxS/O+RxwCZCqTBKXOZTycx5EuMsI/y25wHe4lzgXyOx2rgm/VNMKc6TRgCgiHfgRiM/FYcSvBA7nUJ2UqhAvL6znIECFAsKRIF0WBJkLQgKX9nAKe6KytVSyscM/YoBzpAcZ0LQDwAHwYTaxCTnK7Eg7dJpjEys+mX+ymSpJybTABwIUK3FISB+vfEPpaYfX6dhYCS/ArvbXpMxAOujddHcKCy9oAuosBwO/d66aodBZ6D7bZbgOVjxhKzQcG9DNEl3hATNHPFDklxcc4NylfRkG88t2VfSoUwj1nL97IcKprRrU0xLaftUQLvKqBE0YCLFJ3XjQTVlx5kKoTYQIvTnMDpYqQumFfIquUaT3+TlFnBAUcDaS6Ykgm4D6mHZ3T5LS7ASqCHVNII69RkiqGYX8m+KFuRyLBfKpASn+ovFuh8NjOEFNgt3km/K1AYKDnTQbKHVKfxpUg74CJVZ/QzVmPSrAGHPRrKIU8EQJywFPBHiZyoRp9ZiqatfCUYZAy10RUjDNhY0ApFk/RCyRru7v1GbH2EdpTUYrJNxdP9Ib3+Ky+LdDlcvu7brITIlW20bGBWyvH0bbUTBXw42+LKEajBc93Xyvzg460i8KDqQpbI8P3hvRfdjuH3yYpVuP7ea3AJpxwwZtyHxnXa/nKVrKjEQL0ouoeIxVpsuMrmtKoWLpgAemjUmg13xepWbjuhaStVsQCvFIriQZRKfAWu5xpjxjqLR+92hNWoQUpec70wOSdjJSPAalndek2pgvd9VjXFyCq44wwNIpqzRTtwaEINuYuVywyQb9DmkDnaB+Qq/KBrBntgVwLySNWNI3LTiTgU3U4wpJTHRMn+LtfsYleDP1XfFrU2cbdHOGW1GMxq7WE17UZzrbpJsXeX1zvkSyQmmvmRPyMTrQqFtbSYuvTlGSycJKvDElm2kcqatiRtM+XAOliz6XR6Ql9bpw5unABUyOZmgUjozRPHLFnqBm/wT1OcWK8qbpMEwmn6fkxQEFdGYoZxDwFdZbaHVC/ztMekCVCMn9lawhkKL7TRk45pkUC+5tZSrFzs5CgqwTej3T7yWhaE5gLHwn+0h+1rVeqVTATVL7Tx7iWvUzv/Jv4Af1N+6fX3rQ6vzx0zd7AM9pZGZ08pB7ixFyHMPMxRekmvk8+3fknvFVjbKAAscRdHPQn3bUdqk4DItx4uAP4uJoEdaEbR9+hVL1jrs8QEmOsah78uflhUlTGTIDO74+gLrewWOW3kkDAFWPFFoTSDBczNkV+TIpFW0FdcpU7qk8OkJ36V4143X8/RiB6b0k02n+PhuQfsjDBbaNp6O24mTETI5ypKMLGM+Slrrls1SDdfGMEMfrBVHTthLvtRiagZGJqAA2xWj/LQSS0wDnrjYqKDWY/RsARcIH4dVTFwMoU+ztku1jDMPKPHUi0CfPBUERXHmkBwha9YGGWoT4OvQAnlRhh0FcT9RN9eB9p2k9lUBEL4Kdj9SqUBb3XloMCJaYYQk4unEKTGtoGNjJ0UjPt+F0WKcKYdukt4QMO73z5xqVuff5E9rtA/fYyz/MnjX2wNnKQ6l27H9CWD+RO1+1pn9i7z/JfyrHBDGtR4e3FSyagyypIf+rO7UH43VZHiezC5WmiLqWU+xfgZqHdvoYE0uUczILmJXnvYQMuvrM9eFXgB8GyglGRDci42j/97nBN/c2w9KTWDMBTid2ylcc0NoRoKMCdY+9XNfyvR1x19xfBrMibf6RUaTKowUrS5N7Dy3sl8q6yweBdOJHvfOYiQnu8iCUlX75Ez+yPPLUvSO5cXrlTFVGP2yqazi+DhaRUHVMTJzYyZcq0nRnU9ws8xROWtlaAxirS8Tbqt4mbE2nUu2ExdSdJ1Dt2/jOykL4PKLdSEIpa8QNypx84hVFSRca1604waGkUD0TyhAH/RT2vxpWMp0yEZk9toonGknwSgn97RBT1ampeL6tJ/CfGT4kdwVLcjyqvbqO67OKxmgbJzJWVBM5mqTNsYBERk3uj8adjIxDI+ZHZM8eh835zUILIO2+SXKwJp9UIGbuKzbWJZ2NadauiyPJhjjgDOiMrQMWYVi8Vc2bUH2U+rnIASPySLy+6FkoLOAyxMMvE85z04IXu4pv+jPSofeKtXzYo3loMlVucEoR1aBZtgFt7Uk1A8CTZfhJlFQSN9GQQcIfRJ/0WFcWY1XK0BA4YkZuXRbkev7njNeq0XSet7ub29Cdc6t6TT5P6n/DEquLSQhlN6OtJJXVa7taVdntCPWBuVy2kThB89AhmQoY9JUclVtB3yZE/5EKRI/WR31zf0nq3wnRUu1XfBKJSb+87VxAAI/X7ErAZUgZgPNnfDmXeiVlaqbMzzlt6XPsvZ0nbW6A6D3zUKXbqGHxqvKtoc5crxS7qLwGAbJaLPtkpjrRpyNdM8fjceFhJSYMO9r6iLFvEKJGYZlsTIJ4YkyUMEx91H8UGDAEN59eGv0JettWx37+NmLtlctzWeP1eNg4+4jgYG+nMlx36HAQPeh7bEXxQc1sI6fag097394Tlss8Fq3N2tXtZetrm6FQcFnPOKL8d6J0UGhGVHi3LgvPV3c4QwXYohJghOdXYX9k1GzOZ8QGGkfo0ey6Vgbnh64wGlPreR2h3dUFDlzlGNHXDSrWo11Xy3N0knfiB38owN/fZ1biv5wcfbt5smBxuWm298/t27VXO5rGe3a0JCEi7E3mof7ai61bn/+/vdtpXF5oGWq646GkbKicU19htpCodYv1/GIaVIvaTrii2uulsZCRYZxDdzwQKW7pnusdoxdw56oNYw4fq0WFwNCkYvUk9AIMFEOf5/1gSXXSXKxx9jYojEuTk6uNkjn06onF3i6SjzO5v9mKzjvPyn4SP959KOYUnxM/v7xZyniS4Zju+4Qp03fzq4NyhlqhxjeHJBniFExDgfjwWSGcR0UttJ4hjE9m9lp1Yin11cVEwfs4nyCs7aGP955tz9Xe1bmcMfRGvTZ5PTLyLru2UZN0LTS6HzLns5vkrRl3TCTnkize/DoAiOJgabGlvHNjorUlrzi6FFkmis+MxT73o9x+GHVBdvv98Cv12+z2STuyQz/Qe7rRvF5kHlUz41qe92Dojegl5svIAL//Sr16XM13bFImFk9f84iJ6pHg2BpOc7vcofyKMq2tKxpnzEcGS3f6KUa9+EXVNIGnD7+ceP58a2XfwjffSEe+H6zoXtutLn5fHNYu3tIKTbJd7YlCaMJe6Wp2XWbCUtAaEOZN2+8uzCUxpxtwc0Gtl8s4P17/8n7v29d/htRW+2dw83qfX9ZjBaWy2F2tp+vzY8527GCbXNkshUGk4rqc+LD6WU9Tg122dXhCU3Z3s54bBiL5qLQ6KuW+Pm7oYQF2DJgIYiy+uLN7sv/xHbbn7JyoB/MG88uDjU1DrUSWe7BJViS/2xLgpP51lyqzC6ofLfu9CHFj5U5JdVs7V11nZu+2stxYmfls3b479R2wimJUcOqB6ptdbpO6xknVcG/YCn09kLAeMFzlCAtlnS3jhh8JQVLDUhkZBT4tkOj4bYVl2PvXDu4NO6un+EZW5QeT0CTo3RmDX3DSy6bVMaDhuAPvaXJQfQkCT+QQVBbTGLHyK2ZWx/+W8aMq2KUVA+V55hE8r2JhVYZ2ZbZZ+5Lk0G/I5zczPd/JzCv4b+nNjUkCh+21z4F+D5cnhe6P/y/tR05aylhNwUCo2XeNXph12N+dwIKu9yo9+MymzS6DiHoFipFGVLG123A7bxzekKTlxlNIzWqEl7WsHXQo4//SQxiVLnnsioKptOW/8rUQ8zOVbVU8UouuSMXkF+U0SraYOs/u9Ys2zutn7lEVpKej+BPL95/PY53az7vrU9gvEupKia3AyK8ZUJe3nyIlpbOiV9NO3CEseEf5jkbNe9k8iM7ItWZttSxj5eTSmRnpAVq9b8vm1ToJ1iPF7km/iUvKJj75v0Wwal9A78oaeKgbghBqLKadFfdbP0/A2MzgjMVGxjLEMwYQ0qeiQxrxip4xyXmkDOY2eDI1Wu7mu9uKK6zKh1IZh123N3yzDfeKnLRYLK1/bRFZEFSXXcHzqFTAhrfeSBNFO+/xqYGpK6AtF2lLfCfgVXE95sgGT7XkeBcXkbcvQ5Sl4JqufWaEZ8/iYpttneJrElOiKrC2MU04vaijf6x+PLtF8LY4Cf867dvDUNsOfip2Uf1Ccf63Qqg3192Pd9q7m++Ur91FVX5wa72O/W2E5H0S8M9hkcGRYmKvrquNksBt3d0YQl1TVsxUfDmd9zhhsbmsZIrT3BoHeaKCv+zcK3zUCjswCR5bPwM1cs+usG/GXkTN7XZP/rv5t5Dni+qles5XtlJTfKyii73q4K/8p+4hKplu0vT3unEde5A/dt5c3+d1ASBINr/TI1dXNxcW72wub60BENv8BYhP3gjkA0eV0lozmZAp0uv36ZPPdwPGJGnHiiikRtCSmXm3XIvbo+oEnhcaWql+RWlLnUeako2Lo8rhyw1Z0NW5etRN6XjmFxox0oVo5HBffX3+oXLKc7pzgvr83r2D5kPl7TKSqwmzLWolRchjsdMeDnrc9e1s8dOLZMMly1A9K+teh1A5JkORPI8T/dUKKZEEhYLSam+6KUEBK5/Qo8dp3nAZlHfer9Vtem36ynrkUNAH2SgkbPXipVuamPIXK5CAOQsyD9xAJc/uDR81kte5oIIpjQjP796Am9R7GeyjIhZvzy5fPSFdnZmtCvuvOiG6FQaNSwtws34Zdj9j/rS0p+N76XO+Nj5g3XrcgBRRU5+iVg8iDp4s2asffXNm67rY13tT/fute9hEhQ+CkPBCiCTH77ndK5b1CWUalj4jG0P8PuqG21b0631bfpaxSmcV/yPMBmZTyb3nmX3S4n6CHld5bD8H6EyJnuMtXRH7x37+lgL+/nOftehc7HCN1UDMXFhq8d3Pkk/JX0TXl7aHJmdcG8LhnVDxNdelXePS3U6nKvl2Bm92FTZ5Gyue67ZObTFBNN0jcujtDCRoYU1bio2Loo40rbKnEmWW/+ja8uIYZIV4aS7cdjq3NX2Rp7eVVX+296W+JJmqlqLte8pe8dQjFGgbspwGpe/ej0Io6D1AwhtJ9bt3ShMDU3HsDbJRakig/bUUjrt29Gvn/SiooyejYq8MsZkz6V7Dkr8e3L4hBwJuA83jLZ14+ruTWQQB8eZN7ShGRu2W1t3GBpn6xl/iBqpxCi4+OACg//hfMZIukVsPJni4RPBsTHZG5f2H9K2GCKFqeyhzdghGysximUQ9/mUY/2ArbJMRvbZ1tkL8Zy3vzinu+nioSMHV6neDjLzHQtNEfC5AIITLsTlXFy6un8T28Dhid4L9zD0Fn6LIeTrm/kXIgKrCwNLmX5sfVFdIDAJljT++FoagUKI7A8/GhwceXiwN3L//OD5Q8NJBRdIS0YFrTXBBVWp3A5yDB31zXh4ZCRlI8XSL16MtF8Q2IZZREMNfa+2b/c9dPTO5nTOCu/DXp/dGjWppx3W/cKau8dGklKfTt77PXdUXKW9jM8dnnMYL+EqReFgjDHh7meVu1ykUczL/H4vDq5afhtknzJONbZCmHS9OH/nYR6G7WPAPRnIqA0goDPC+hxY1mcCT6d6VjxgX50dcbOtGHKpyM5xdzK9HWdVj1rFVjVaytFM/cHZJRPJssanfevO+XVjXN9F/3e69DQzLoKEUlzHbQJzQM+lQ/QB1Sot4s1wCbQb16/s3Ub1qQvj5j5U+76QcgEN0bxums8lLlVV8xsMY+QuyGBKM/LOmWV5Lm6K1J0xLeYBGoFzFW/QuGs0q3EFGgip23710YT8d5zpIrxQ+QpkUHvwEsfr1tAEv2xx6dSly7oAFDf3KcvnUD4StBrw81XBnCbtN9eli+wqL3dZZipmHHfN/iw0/QQZFheEKH1TCn2BjTB5KJlII2YJ1F0R2BllmycXheTWlwpQidEWZ/njvNpXIisG/Kf64Hbmm2q9VqwHQML1yuKn15syWHkuUru0P9llEUMPKxWozy8KSTbvujV2RaAuw/U8TzSipOdkKZj63zhV7EzojEtrwmSdCIE5nYhRN7cscQzpSEhvwBVB3LQ9tENVoA/HB/8+nVxlnr0d9r011xk3G6Sro4n9zOfkUy0iN5h3trWXZf/6p9ZhonLIQ7AeIgqvLbdIn+jff3C3Z3u4vt72tPH5raujePvyLq/GgsKCOgNOCedJPLryZeJrO1uk/pbZ92fXS67bzz27hVvcz3dPqaiNIFaUJ+EYJQm4qvJIIrPaPSu7zD2RUUEkVpUnYEsq4nAV5URieaV7ypmCrO5eGu1cD5nS20+hdQ9C3J7m35tSFxIULXEYc5gTgofA9iX2Y3mA2rzgXBBHt2lQ/bd68XooaN4oiFDlDXU3wGcQ8u7UbET2aETroxgx+hXqXjSqLdo+H99lecGlvSM1zL/O3ZDl11L3jnX3YQm6zRu6YhpQUefpZvZnqg3FuB7JDHUiN8d4lyVz7AWrfDPxLZlL1i7ygXEuCgQyyhumn46yiNaHoQsRaz4YBbvhaVnd1SuqmGSPFPeXBRWaFn6LP4Fd3fl8f+/xj2tn6HY29FQLy2bcjc2IiK3tGxGbWxHbGyuhETu7OxF3VkOTNyYnTU0nJqaMJqcMZ6YmjQynZvb+kxWr46KjgidG10VlxCYmdiUPidCEDJht8UkWxlFkKyajKjlvffpSfNYWFxEjlCmUaG3lr0/oyMZAIiCR2VhCR416cAcNoxWhFUnDBncYObeqyFQzrWtO7P+Z5hcdgPYkoCNSZiv9AvO9NI9rRu7EVMa2FYhw2KXOcnabee5EoBMVwSVgObHvrQzuAoOxwC0L9G9oxgcyuPMURgCN7O9HyXrelrOyAvAUsk6de3jyh1chBP9oiKclxhfjiIqBYRBOwYwMVnlLaZZfSEdL9eTQkp4B6o2ff6KL8rTZwJnFhhEW1aqgxaXgYKD5DT01JynWB+Pg6+LinGlMwYQe1Z6KrOrOtlFt+/pvxrQ2elvJBD155VumAqFQ9L/xnst3e4f5qtRaJDI4rzqCiRLi+D50P+jAOYUVEKYAojOHuScqci/w/vaffws/+PftSYPOfWcHjHViHnS+tqnMQ2JYpA7kjZRNO18/A0sTH1SabPBshiVlXh+qBOM1pjFdiCKhhPCKSOcoiMnLzKExXkfLyIX+wWFuJ2toQaA/PywvmUTMyy8g0klJ0jBbBrhxZmG1IrUiaJjgphJKl7WeuJi49zgy36M1Gp9LsvdQ9bKOL8vK92uHxZjbVqxHXb/KvzCG1U12DAnxj4wPRlMjtOeNK/PsHDDOaCes1WNacQJlf5pEsizwcU/w+e6MyXneVieGr9mhqY7xWNm/upDtY4oGDoQbZlazO9idc/4F2Vwi2uaNcEFHwc+rscEgqiJhu86Kio2KiYhaXUwkMTI+djUhKi4+OlYtvez90PqrqC7HJ7VrdiM9PfS41x0RXZHjS2q9tjWfIpveZSzEpfjPpIqODs3GovFdRgfTJNzIkBEHnwlya5V9YXasvZR+3qjHAL6a0h14rPLI862GofYr1du3LJ3avlY2UIJdrZx1IKNYa2+L2siE7oS+lbH5yykY9uuzZjg+zzMgJd212D8qP9ARQ8KiPKwdQ0/5VQaWKD7jcubmHl18eue3bjYKJSLM+n/N/Su2mEzdIKn5OU0DeGSxu+F1VCxjKMgwEhU8FpWx/EbiaeoVsRpbYrcuXr+NAGuDJOgUvvUrXidVDC6NTnZEpYSENA+PsOPOiV4QG0yz27fIqzkvcU6su2Q4pj6QX/lYXaSa8MlfT5XBz1+Lijx+HpdLtT68dVWajUt9W050iIBjaGdxCR7H1Z4pd7I7G4bqh1Va50e7/ZOHp3gD0303PO75ZzOprmalxl/FfsIS1WNrM41E3fleyMgEoCY7E4Onkb0jGGc6gTTAI5vqic+meUaXF3ei4H+OoxJ2mR4oSirg0HEwZBG3y8BakAcIkmec56NSGdcpZNQnhf/JRBL82fTe07w+MELMy/EmBwABnH2USKib+kjnosANo7N88xl+6+5pidTov65yRhYAdBXC3aKrUpKJTHRiqLJ0pzffoMNA3dtu/dyfANzACLD4lRUEoQdp4kOjFWGF7x2ou9nVzyThIHERkw9lt7IdyDaZ13KLt45Bz0kPxKo8Ivt0g/TbgvU7tb2/2hx0PhweZN3de8K+e57Txsdk3S9pGpQ8B+kpGyTIrn8urkwoSeaEppUXlqrkBFuWMyqZAUiqqsXhXGx+dmJCDi02NpcSn1CQnRph7WSPRM5lYW192QaFttdCYB6Z+EN9THFQXQuTx7HgXEoV0YwA9TZ1g0IQfx3HgXOzqqD+91nhLDm59fRlgCUniF9ZUAqNkbTKLXD6F/pTXoHAezU5Xy/fVUObL/99DCuvrO7liRVoUmwh4GijW74i9RcDg0OVsChxq7witTgnJDhcHTL4IS+rKI1ELSjIoZxJSSUzFPrxT1PXkNwyFKgb/uBZ03mN+RWmgFg45ejT8l8pWBWtsVN6DTq/vT7WzXLqaEaPH0PAJwFOKUvNVMCXVTqkL6ynd7w1FI2tOU8gP6T+KiUTF8Wi7ebtloRO51pkkfRahYiHZ/bjx33DP+xHofEc2GOLtY8LsAxkAWVt0gCjPcD0EgZ2ShnA2vI7sNHdqjld1AQcfV/vSa0+q/TR/gH0fc/YUenZog9aOt65r0ZAOB0zfJle+Crh/UisKEeusFHKQEjlFcFoZUPv/sWgod2Rwpf733uTyMvwUP5WP+JTneUhnB4S4cu44avYe5McylYe7FiE0+0w7EXYm4Ql21n5EbTLuoHzrTgLoW8QsvxhyOqzIaSvO1b+irpy1BnYqJtH26MUQj84ZPmvDtJ89Eo9PmXm7zyc3iHhy7PhvUmzxahfVL6F/f4f4vyuCKEfbV82h6z+0kFawR/8Ic6vIIR+Ycjy4h2k1aJE2BYHIdUzNZCJ7ARbgP+2NVMOYB1yRZ6XlFzCWrGEK4V7Fp7fh1rnw8qM+aO1dE2rD7IXBwyXtd1nTypsUPk+7MYlhv0SGJFdZbJpZdWx+L9vP/w058WPYbhNLhAYD4pjIUsmkOvlGk9Ae6vxmAHaJ6pAHTCs1rbQ19IWHakImgdhbqKKZoq5027aAtS/beYAH8yh7TKaK+pH7SrmAN60J8xFWm0OwKGVvJqoyK+h3h50qLs9kQ8wnYk6U6y1GY6hpLxNaSpoB/0hIE6mI0YBZXOlz4M+ij6KrEJnBpB7+ka2cqyLteJNIMgQaJvSAkAEOSSiwALKzo6zPyLdR2qzPwsXC7H6izblK01ArjhoLwILdg/vBF5ub68UiQDi1pEq5siO2vsmdSbSLab049/vbP01kQZhA9pT2yXN1Pl4d3txEfcg0g9JUkONP5BVivDf+eg89HXrfNK7rnO3c9aATViSx2BsTnb/OP0N9jEuj/dsOipxwZ2UPhldJM+hqMYThuMeJzjimpC4o+IGDSKqRT2bZ+/tq/KdUaKuwDr7qdaU7r2W/LASNVbDJAYkFFt37nzft+3Dn+u6o4NIOIUDkwi8ASkf+xyv+k4nAVs26jy/b916HDF+WShyA9AQx8nbdq1o+wai9/qg3zA+HEfF6C3xUNCHMaQisLUik74/xR/qulPCJyJId6+6TjNeW7BErLvRYnGzE7eWnjqkro7qweX2jt7WzbphgcF/3Uo6M0WVBS6c/i6jU6hgTJeH483x0VJvmnHFnz7r5Eu6YLaP9C1pUbNBh8Os9TP+ihmG/45Q81rRxt43MpiZWf5XlUWDoaL/6EbY11IZskC6vjxFZedYU3Eu+cnQ9ZnlKc5LEBGyf1x30PsBC2qh1pZNYuqA819OQlH9QLq9DmBx1drYkv0ETKKE7QE+3qVmfm7gWYFWOtInTlBABRTsFrgoFJDN+Qe8ghZKvWLvUkCUOUyNhGN914OMiMeal6/hjPvtQ42Gyo4tacDkQIlKFTcjj3SALhlsfY+BogaDOmEzBJ3IaWjFayxLDm/A41m1xPrtEAJj4HWh6PMexarqvCZGsV5PXrNmYlybaAIaVPhsqv535p7wEIoxWHkuEjKRN3Bjk0cssmwm54t1LuNcGHQssiIifD6EenIAdCnl/MlLSusXaK2GGhfTDFw+nwPQEW3WMu8u3nKcuh0q4muKBmX9o5xRlvfOz+pxvoQoAlvi8BVEseIFQLzIKZpN8R0ac85xLAzdbqnbz2raa/Z7CfVkhWxguR8FGDgV+8FRWA/dQggAy6pt9STWF1qHXVGjPVhqFdKWfgbfH6/PybWUVqCKOrxHW4virYMPdNXNRDV134XcvmewdVqU6iQMJVF4BGhbi6/rsVQQ4En0JzjB6TVJgs33LaD1lj4nEQqoGhpZyead99yzOcBI46cdwMquJu9ZVVa8xGKlLPZN1Mubp/qHrBgC7At9Qy4v9fPn+MeZyx9+xHF6QpujH/0y35+6wMOjCyFPdUmkYBiLyWB0+EROuk6UIROcO/czqMD518X0AY+FOTZaiHVys8jMoeq8wpgO3fEV2kPcjxWQ9fOGcUV6xzXvaw8v8TiPdVcIQw0R1NXYIZgCUZ8hSLpFkln38GYKKjIJEklt0T9BKYWgn0gR6fJjTHyfuL9iCkg4EedSyTaNjGbinSqKrEnT7izD52MYvgSA1Dq+RGLd78dIQj1Lj3hCSCIKbTXxJ7JMc2RSsEmHjDGLX6jlKXW9M+TpgwVbIbSLHp+MhpJ5RYKXc7O/OZmOAByNgMfLeBY1HGZEY+m6FtuuTMSIaDHuELcIfcTYy6R5nN48lr6HYfQ5sj4xmVgzoTFMUy8pTH16ZUGJhQxQ5/MIcyHwZ4SIZtMrGIodYeuHQmFiRQWAOd84IWXt+BKFbTYCYajnb8i0b+tUE6H6yPKczYOskSs0fEapwCxfFqESFJmb/eKdezIJElAHgh82coKqyha+vMLI21YBusHRhkSJObmx7GLYyoQwC+Uulk2Ekc3eZUga4jiVdUDVUhKpJTGEmVruJjPYn6nZiOGg1Fr94AQiNwlGwiRbZGOqJSNajU9egUDxozGwBjX2DCVjFcBJWCfIEjaYkOSiEl9OEKWeZnS9BSgJCMk1cKSnWoxqfOavNZnaD9QR6DdMDntCmPEfxUUCtqQBErC1Ldd91kgonepcgrXZL7Lk+AthG/xjs5hukeWXk6bxGXrYJGm5vOfMX7CU5kppyeAKqaDskrmPhZ5rEaMx9QJJvp4rqIt3wlGItBT99uIKWAf40K5eYvH9wsEgXjg6RKoo9rTgK1ch0Gs1phF0N1I3m/vSABKq7C7kAG5ReCNRxUZLLGoRc8SjQOfDmF+JEWjciXt5bQpYLHLJCinnNDVWXioBT00BpKbvCczwAS9SDP0Y1vHOQ7wzj2PzfC2nPyZ3IQkkfbkYTdeN+NnbyDypqleyH94jFtnBZLHeejik9/z07s57wM5WN2o1bI7Qpt9J3bbO9/2xJ9aPtEzJsfIq1ppPHtuStwLAFzh3Zw/Aax5FxCqqjYkGp9hHVNqTIuX/nPY9+CHPW0rHQWVfUF0kg1bLIFeFAdlflFgtcikpiiYZvQRYm2VsZUGm55B2HT03O9Y3YIYERKIBgoWw7rkpnfMZxx33S7Raj6qy9A1WRwcpmjY2GI5JJbnT/E/sfF+33VeMZAgFIAU7XpWSDzam7KtkIHJyAyiDnRpRmfjTLbJuZRh4/5iyybFZuvKZXyxxjr5frhjLTTqDkSvZzt5P0Dk7aE5qxxUaDBrbVB0MoxlPJpUKgcUBwSp4n5U1c+sFFPecw+hz05AXkwTsh/v+u0w09dG7k0BdT7cUecveW3vjNbFFwXOfED5EEWy4R19MwxoWZB8++Ec4gHgJURKYrHkWZSVVG7LC6Lu/yj1Jf+fDrwD3TDnJtalTH2x6ugD+3Lh8OJ7ESiuWJunkuUFagviyrgvk+RYjEaRfm35SSmXBSX9EVQt4jVihTdBqwwyK7adx0dqbUx8owyxySevNjnjNYJBHAGM7hKzwxSR8hzTOM/HuPkgpsyifAVj3c+SVREw+bWA81vpI0OW03FJEbZir1rOQOTRefgB5QyOuLVHYjmDrv7R/OUK1wRQLu/cDAKq/bbwDFi285lWmmLNgsDjj83FD0lmo8DBFiskRvrN/ls51y+YXxb4OXLVT5dJ8DWQylXVTv4M3TYBg3xBqIF++3pi/x3HRc6mNYKGhe95Py3LNYOu6dnQ7t3avWSmDJB6c606XLNZCd9bHySOYSxOmITVzjZPJQuXBu6CfTvos/4NI6UV/zkxp5EG73IUraubALTXq5tzoTxDoYfK8aSaS4o7+5FyiHz4dPaJzPt/eMy+53Q/hoFzlOLcHmvutTuDrxqCFw/br0c4N24jMNp0W3OzJs+dUYjLySqaYrE8xNXl4XIZuiC2Y/T2okht7i9JIhyKhtR//M6/nIyGmpgI0oJTbNIVz9/+H9fi+VR9izUG0o5/PUtylRen6qomwdW0nf9WUhKKp/ISkIcw64Za66sA5j3kLdlbxOkX67Lm0FkDP2mfB+KlPYbYh9aHm1hIkV+VuFLNevIxedIV57pw3sOIPK2Ly4YO/69LBYiEmSTcD9tpr7TQms7lSS7w4akMUPE0EvV/7B+dXTvrGSfJ5GNIQssF8qrKbLyCRhzQsQ3/Nlzmci/xKzJ0ai4nIaYAQiqyoIy1NMVRpuI5xH2c77/n1dtiYEEAKDy7LlHuln/vKWhnPQaaAquw/QLAdA0W+6XQO6N9K7oN2LkMDtH8/mRX2J9ZTMOc6ytcULn0b5sv/wqO3JZ2Awh1ZB+lY+Ef4ywOMD2TOJYAtZI9uwTiVmh9rOtVwfvLk98UqKPtfvSAvdVZMlsnIjJYF3h9gnEi+GLvNZvaejXX5S9yyR/j07/8fWvOmPhZLNnB88rHUonwssp9qer+vPbmsvcCixgRSpYGQhRQ80PASi0YRSkVE/agksk6JHoyzW7TIMRMOWlWPT+4GPoYKoUap/IPhTyW5CeXl4soyww0ImMqhKSiLd8XBsP/Yb2L/smfWpPfZTNyHV3Akyxidu5tWs6nsPw0M1NKo4Lx8U4Y7HXxz3yTmTE61JRzb/5puraZxgud11xMFgDdQK/dQTeaNmr/E+IwIDwrz2LCfolHhV/ybj/3ME57jGPwYgQV5qGIzNgJPqY/hr9wuqeKqcQSKNSp81+GgV/GS44SVY9wq6fzz6v2BjJcQpcSFO5C7k/eHm5vNlRs4lGkUnzyVeAaIfZh0bZx8o8q7MlDnnw5WgWf95XqAlbvdpl+s+Lav6+ltDvr3j4ePto+uNHf47p7nq63DuLvdVk8ynd5TaAuJPU5cuRlJy0aM+m3i0McychfySsX2GxJ4N/AxiFYNRap3nMrHDfyXWeZUno8A6qbSLljKn3Z8eiVHq2u00R3nvC8SdD8oRVF9Rh/7Z/8GP8YkAnmggovybRkiC7ytcQkrIQacCg0I8kNIWGs0qwdH+B74lLZz/YDczXf+RNNRUYhRe9u9uR5o0fz6Kq/+vqtyXMyDsXTofuk6p66pYkaHhbWOutXSgxvKiiO/vMIrFDkxTzitx4HgK/AKjztUlSmPAPe9pmYOY1DjmJLrufFdJ21QSj+i5mj2hj77FA/hmSTzOM0054UPvW3TTIe1WrdlOVpn+LfuuMt3D3QYYTyXKenK2xviclsvyOIKkLW9Eb9ffrzgjnQXiWEwuyWS3Ve7QXIB50Oqp7KP1/iCbrbYTcIN+rpxQCqW0vKkZv/7KzTs/MGnFdHXJKyvTf4SJDnklRZ02taSdrWev8WwJUzNZXCQcX0FTYraWryRyugVGdii5W9n9jXA/tfUR+Gb2XxVqjbny2E85uCTJ19eG+FeP2F3lvcZB44hJlPCtj38aWQECQjatV+1IHMt3ABj55/GcVpVgzw1O60cFqdIt7s81vEdWU3Gk/FVPZ2YyXTSThDnw6QKw2IBb1iw8JLaZwgNTNLRbq30q2jGMcE7odKxOL70RUFyY9aLYabhrBOi3u+mRVoUSNsVmo2h7Q9pLd+mJ8fYeSDcopo/Qo+mLpHGP7Wp8g2dy7WtAAQAYrl7ur8Qd39v7P8AiX9fXfbNrRC16qx+XFn5mz09nBuYxALMfZNNtQBMbpmOAOYe+qdtVmD9zXxHfZS3rYzmkoCSN7sGVj2F2Uwgs1XZ0CG58ZFjA7NECljkHDJFkN49OX5yLK0Ciw5W+UyxShKJvkzH05G0Cc2U6WU6WUqxHdThCDFHLbJmOjEi5DVZQNvEVgMraoehJDUnOkQfUG9OUPR6dL7rTQNto037AFpjpyj3BPW1IOOlZDp5ey8Xcw0wnuFE9gDl1YC8I35tACdWoiK8IzMIonb0f6QnqpMqMt+YqTuoiz2oWVRQeDQzlkG4Dc0UUQNFtDcnshidGclmdIk8tBVn7JWKcSV79dMOaFNzQykIGLajkggGCSJYnxNQQRwyOT4qEVdvnnMoyNUccUIXiNpmJXXEQj1xahgRXw5h618C7IFzVD1VAv//xelKtE2IXNKpR/53M80Bmed07IHpcl4VAHZ37fUbEM+cNiPqnEtIvktRZvi/vwkw/0YaaZvMObcEamqkEO8mgTH2G1v5lxmOS1Hb8ARThSIcUET7Q8UoyDzWHeUGMueWQO0JKcS7SQBw84DCzeYUq2isRs7LAeaKZJIOlIDDJ/nBEU4Nte3kbWporw5OgoYGLFbJFLCY6BcSsytzqQ0nL9PEKcomxZFOs62lOxQ7W0TM9015QkAAVr7ZpS21LADMD9cRAe3kxUtcS0as2Vw7incZj6nrw+QXuCFrEpKldNJlCtlypBAJEwWBiIH2QlaiyoGrCFU1dY2aVAvaOrp6+gaGRo46jqsTOHX25wYA3AAAAAAAAAAAAAAAAH/rgH3t1NJ7bPjslz+XQ742/GfH4OBq0ZwEMoVKs9mWc9uwY7c9H+f6v2ucKYgCAIAFAAAAAAAAAPDZeoZMxu57fJ/v1+lX0R/Ema/dVA5sLGdbGouxcizGgjYW7dm6Tvf0TPq3axrD7HuLcTlr1+6II6cuV5EbkUoUCYKIgfasMrw+S7bXzz788sciiUyh0mxu7DFeRERERERERET829HX5puaxiSz73U+NuRAUAIRAxkOz0KqPRERERERfc6eCYex3lf6DL/8wT2bGNkFx3VZs0hKBoVKs9mWc9uwY7c9Hz9ioR+zc8MwMzMzMzMzM/Pn1vT/hdn7Fnu9jvU3xlaiRCBiIMPtWShRi4iIiIiIDGwhIl2t6ffVTGF229ThHhn3PX6fpV/+xJmvxxYPz2eVnwSEbgJbshSVLmRkywWrgJDXMCnim4srQmgCEgUGWkSSJFOoNJuVbCvHXoVU1dQ1alrVgraOrp6+gaGRo45r1gmcOuscXaMb9FDLHimDlWfh62ka3+x96whBCUQMZNj+38Ju09gwu0177G8K4x1hLMZiLPI45AO/QCFDCUiWQulCRrYcKiASJioJRAy0aJUEMoVKs1kpQGWoqKqpa2hqaevo6ukbGBo56jhYJ3DqzHU36NKbjL5timPYAAAAAAAAABQJ9UmSqvNLklRdLj1f+jyOa4sF8JEHG/kkv0Ahw6bOHIvFscXieFNY/NacQyRYVBKIGGjRKglkCpVms1KYylBRVVPX0NTS1tHV0zcwNHK0STbwC6/diAw8SzYdwBgzLJLIFCrN5rnupe+f3zNHMEY/Zquz4772CMOK+OYMztOLdfvy/OOTMEY/ZgF/AAAAAHDeNw9j9DEKwzfoaxt7AsTY7WOHjx0ydjZERERExNidOuEdx+nqXNclShwQERFRdmNmZmbmL79vBz+zvsvrwXz8AoWsShVlSElZRVVNXUNTS1tHV0/fwNDIkZv06J6J8u1IXl4+VA8olORBob4n5BxKzkI+Qila2AGGYRiGYbgiQyKFG5FSbxERERER6R8dHo/H4/F4fP+78SGXjKy877HPO/lEVBEEQVRVVRVRJBoeRc1QMxRFUTMzM0MNZUeSJEmSJEmSJNu2bdu2bdu2AQAAAACACgAAAAAAAAAAAACghQl/i4GZIZm+SZJMr52nlqUVq611VGvjvsW/nLVrd8SRU5eryI1IJYoEQcRAe1YZXp8lk8KQSCIiIiIiAwMLIZHJZFVVVVXVgYGFkikUipmZmZnZwMDCKHhUKhUAAAAYGFiASqPRSJIkyYGBBWnOWlVVVVVVVVVV9XwfHzDeEdw9PL28ffrZ/vlj42yMMcYYY4wfkyRJkmTbtm37uCztTrrJDwAAAAAAAAAAAEDAnyMU8SLee++9VFBKaV3rtDHGWGutc8557z0QyT0AAEg3kiRJkgcAAAAAQEREREQkIiIiImJmZmZmFhERERFRVVVVVT3vsTRGP2bzZ2ZmZmZ9EwYz58595ty6COacc8455/zuLl363aV3nSTxM1MHBAA+U0N5YrdpQbLMhQNiYX/8umRyFLY6Q84RIA841ARJH7I4Vog9iF3YfKwIVeHHipJFuyClBSrnK0uCDBB4MTJQBUiQhC5NDAonVHEHJF7mKCBO5iUcNAjnA0KCPh4EAnwpBVmvoiLJtEshmGl5R/5AS0ZbHpnFlrk/J2cmRlZO4gwkoZTLRJGEwTSkWGiqG5U5L+5cuPKBl5NPsmAG/gtSy3uiKpkNfQaIX2VaSJgIUWJAxEmQ/Nb+O540GbLkyFOgSIkyFarUqDtBgyYt2iB06NIDBaPPgCEjxkycZMYcHIIFJJRTLFmxZsOWHXsOTnOE5sSZCwxXWG6/Fnqd8+DJizcfvvz4C4AXiCBIsBChwhCFixApSrQYEwaVY7qiw1sVGtQ6Z9RQCKkJYROyO4yvvqnXqcqap77oMeaH734aMGnLpimx4jSJd1OCG7a9k+jArj3TknzW7J67DiX74Fi1FCSp0qXJ0CcTWVYTX3/paLLleC9XvjwFihRa0q9EsTNKHfnoIr4Zs+574oE583gWreNasKHSuKuuuRwiPoVojDGJuV9fOJ+Y2nHLbXcy49K0NDH6eSQ43AkjnnlZAoUal0lJEHa9lJJ50zDWGY5QZ2EBTjyfTqHSsxIopEzKmYzEIlCCiaehu1xnXwgBIqB1lDZOaM4NLQGZlXfRmp4IeR+gr+3lidT2xMK09VGyBBgSSlBT+F/bhKMtCuU+uyO13fUOCkh5gohAPSqRjpTQTtRd6nQV2mdvNe0Pfq63hS5BA3mCoIxUIshZftSiyPu0RWQOWURwlUV2zh6LLNLXH0ogBNL5dz5QYtonwhO1KOM37OCTOOCf7qQWBMkoZo4SJA6awEeImlzJlQaij6aA0Ghqh/huXr69gfO+BPM1DWufSXB7NW9+n6uyyNy7FwSr9+4/CLLe+8GnLABej+9aBQA=') format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215; +} + +/* Open Sans Semibold Latin + base64 encode done on opensans_semibold_latin_v13.woff2 */ +@font-face { + font-family: 'Open Sans'; + font-style: normal; + font-weight: 600; + src: url('data:font/woff2;base64,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') format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2212, U+2215; +} + +/* Font Awesome + base64 encode done on opensans_semibold_latin_v13.woff2 */ +@font-face { + font-family: 'FontAwesome'; + font-style: normal; + src: url('data:font/woff2;base64,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') format('woff2'); +} + + +/* Override table width restrictions */ +@media screen and (min-width: 767px) { + + .wy-table-responsive table td { + /* !important prevents the common CSS stylesheets from + overriding this as on RTD they are loaded after this stylesheet */ + white-space: normal !important; + } + + .wy-table-responsive { + overflow: auto !important; + } + + .wy-nav-content { + max-width: none; + } +} + +/* Hide the TI header on small screens */ +@media screen and (max-width: 768px) { + #tiHeader { + display: none !important; + } +} + +/* Override colors */ + +.wy-nav-top { + background-color: #CC0000; /* TI red */ +} + +.wy-nav-side { + background-color: #f9f9f9 !important; /* TI website Nav background*/ + position: absolute; +} + +.wy-side-nav-search { + background-color: #555; /* TI dark gray */ +} + +.wy-side-nav-search input[type=text] { + border-color : rgba(255,255,255,0.8); /* Lighter gray outline */ +} + + +/* Links in general. */ +a { + color: #189; /* TI website inline link color */ +} + +/* Inline code */ +a .rst-content code { + color: #189; +} + +/* SIDEBAR +=============================================================================*/ + +.wy-menu-vertical a { + color: #555; + background-color: #f9f9f9 !important; /* TI website Nav background*/ + border-top: none !important; + border-right: none !important; + border-bottom: solid 1px #c9c9c9 !important; +} + +.wy-menu-vertical li.on a, .wy-menu-vertical li.current > a { + color: #555; /* TI website current nav link */ +} + +.wy-nav-side li.current { + background-color: #f9f9f9 !important; /* TI website Nav background*/ +} + +.wy-side-nav-search>div.version { + color: rgba(255,255,255,0.8); + font-weight: 600; +} + +/* FONTS AND SIZES +=============================================================================*/ + +body { + font-family: 'Open Sans', sans-serif; + font-size: 16px; + color: #555; + background-color: #fff; +} + +h1, h2, h3, h4, h5, h6 { + font-family: "Open Sans", sans-serif; + font-weight: 600; + color: #333; + -webkit-font-smoothing: antialiased; +} + +h1 { + font-size: 2em; +} + +h2 { + font-size: 1.6em; + border-bottom: 1px solid #ccc; +} + +h3 { + font-size: 1.4em; +} + +h4 { + font-size: 1.2em; +} + +h5 { + font-size: 1em; +} + +h6 { + font-size: 1em; +} + +body>h2:first-child, body>h1:first-child, body>h1:first-child+h2, body>h3:first-child, body>h4:first-child, body>h5:first-child, body>h6:first-child { + margin-top: 0; + padding-top: 0; +} + +a:first-child h1, a:first-child h2, a:first-child h3, a:first-child h4, a:first-child h5, a:first-child h6 { + margin-top: 0; + padding-top: 0; +} + +h1+p, h2+p, h3+p, h4+p, h5+p, h6+p { + margin-top: 10px; +} + +/* HEADER AND FOOTER +=============================================================================*/ + +#tiHeader { + background-color: white; +} + +div.top { + padding-top: 15px; + padding-bottom: 15px; +} + +#tiHeader ul { + list-style-type: none; + padding-left: 30px; + background-color: white; +} + +#tiHeader .nav { + background: #c00; + height: 41.375px; +} + +#tiHeader #top_logo { + height: 36px; +} + +#content { + padding: 1em; + max-width: 1200px; + overflow: auto; + margin: 0 auto; +} + +#tiFooter { + clear: both; + color: #b0b0b0; + font-size: .9em; + padding: 1em 2em; + padding: 1em 2rem; + border-top: 1px solid #e0e0e0; + background: #fff; +} + +#tiFooter p { + max-width: 60em; +} + +#tiFooter a { + color: #b0b0b0; +} + +#tiFooter a:hover { + color: #c00; +} + +.codeblock, pre.literal-block, .rst-content .literal-block, .rst-content pre.literal-block, div[class^='highlight'] { +font-size: 12px; +line-height: 1.5; +display: block; +overflow: auto; +color: #404040; +padding: 12px 12px; +} + +.codeblock,div[class^='highlight'] { +padding: 0; +} diff --git a/beaglebone-ai-64/edge_ai_apps/test_report.rst b/beaglebone-ai-64/edge_ai_apps/test_report.rst new file mode 100644 index 0000000000000000000000000000000000000000..55aaae2f5098051ba86f847165fa9385c2c55289 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/test_report.rst @@ -0,0 +1,231 @@ +.. _ai_64_edgeai_test_report: + +Test Report +############ + +Here is the summary of the sanity tests we ran with both Python and C++ demos. +Test cases vary with different inputs, outputs, runtime, models, python/c++ apps. + +1. Inputs: + + * Camera (Logitech C270, 1280x720, JPEG) + * Camera (Omnivision OV5640, 1280x720, YUV) + * Camera (Rpi v2 Sony IMX219, 1920x1080, RAW) + * Image files (30 images under edge_ai_apps/data/images) + * Video file (10s video 1 file under edge_ai_apps/data/videos) + * RSTP Video Server + +2. Outputs: + + * Display (eDP or HDMI) + * File write to SD card + +3. Inference Type: + + * Image classification + * Object detection + * Semantic segmentation + +4. Runtime/models: + + * DLR + * TFLite + * ONNX + +5. Applications: + + * Python + * C++ + +6. Platform: + + * Host OS + * Docker + + +Demo Apps test report +===================== + +Single Input Single Output +-------------------------- + +.. csv-table:: + :header: "Category", "# test case", "Pass", "Fail" + + Host OS - Python,99,99,0 + Host OS - C++,99,99,0 + +.. csv-table:: + :header: "S.No", "Models", "Input", "Output", "Host OS-C++", "Host OS-Python", "Docker-C++", "Docker-Python", "Comments" + + 1,TVM-CL-3410-gluoncv-mxnet-mobv2,Image,Display,Pass,Pass,Pass,Pass + 2,TVM-CL-3410-gluoncv-mxnet-mobv2,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 3,TVM-CL-3410-gluoncv-mxnet-mobv2,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 4,TVM-CL-3410-gluoncv-mxnet-mobv2,Video,Display,Pass,Pass,Pass,Pass + 5,TVM-CL-3410-gluoncv-mxnet-mobv2,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 6,TVM-CL-3410-gluoncv-mxnet-mobv2,USB Camera,Display,Pass,Pass,Pass,Pass + 7,TVM-CL-3410-gluoncv-mxnet-mobv2,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 8,TVM-CL-3410-gluoncv-mxnet-mobv2,CSI Camera,Display,Pass,Pass,Pass,Pass + 9,TVM-CL-3410-gluoncv-mxnet-mobv2,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 10,TVM-CL-3410-gluoncv-mxnet-mobv2,RPI Camera,Display,Pass,Pass,Pass,Pass + 11,TVM-CL-3410-gluoncv-mxnet-mobv2,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 12,TVM-CL-3410-gluoncv-mxnet-mobv2,RTSP - Video,Display,Pass,Pass,Pass,Pass + 13,TVM-CL-3410-gluoncv-mxnet-mobv2,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 14,TFL-CL-0000-mobileNetV1-mlperf,Image,Display,Pass,Pass,Pass,Pass + 15,TFL-CL-0000-mobileNetV1-mlperf,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 16,TFL-CL-0000-mobileNetV1-mlperf,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 17,TFL-CL-0000-mobileNetV1-mlperf,Video,Display,Pass,Pass,Pass,Pass + 18,TFL-CL-0000-mobileNetV1-mlperf,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 19,TFL-CL-0000-mobileNetV1-mlperf,USB Camera,Display,Pass,Pass,Pass,Pass + 20,TFL-CL-0000-mobileNetV1-mlperf,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 21,TFL-CL-0000-mobileNetV1-mlperf,CSI Camera,Display,Pass,Pass,Pass,Pass + 22,TFL-CL-0000-mobileNetV1-mlperf,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 23,TFL-CL-0000-mobileNetV1-mlperf,RPI Camera,Display,Pass,Pass,Pass,Pass + 24,TFL-CL-0000-mobileNetV1-mlperf,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 25,TFL-CL-0000-mobileNetV1-mlperf,RTSP - Video,Display,Pass,Pass,Pass,Pass + 26,TFL-CL-0000-mobileNetV1-mlperf,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 27,ONR-CL-6360-regNetx-200mf,Image,Display,Pass,Pass,Pass,Pass + 28,ONR-CL-6360-regNetx-200mf,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 29,ONR-CL-6360-regNetx-200mf,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 30,ONR-CL-6360-regNetx-200mf,Video,Display,Pass,Pass,Pass,Pass + 31,ONR-CL-6360-regNetx-200mf,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 32,ONR-CL-6360-regNetx-200mf,USB Camera,Display,Pass,Pass,Pass,Pass + 33,ONR-CL-6360-regNetx-200mf,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 34,ONR-CL-6360-regNetx-200mf,CSI Camera,Display,Pass,Pass,Pass,Pass + 35,ONR-CL-6360-regNetx-200mf,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 36,ONR-CL-6360-regNetx-200mf,RPI Camera,Display,Pass,Pass,Pass,Pass + 37,ONR-CL-6360-regNetx-200mf,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 38,ONR-CL-6360-regNetx-200mf,RTSP - Video,Display,Pass,Pass,Pass,Pass + 39,ONR-CL-6360-regNetx-200mf,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 40,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,Image,Display,Pass,Pass,Pass,Pass + 41,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 42,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 43,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,Video,Display,Pass,Pass,Pass,Pass + 44,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 45,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,USB Camera,Display,Pass,Pass,Pass,Pass + 46,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 47,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,CSI Camera,Display,Pass,Pass,Pass,Pass + 48,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 49,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,RPI Camera,Display,Pass,Pass,Pass,Pass + 50,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 51,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,RTSP - Video,Display,Pass,Pass,Pass,Pass + 52,TVM-OD-5020-yolov3-mobv1-gluon-mxnet-coco-416x416,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 53,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,Image,Display,Pass,Pass,Pass,Pass + 54,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 55,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 56,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,Video,Display,Pass,Pass,Pass,Pass + 57,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 58,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,USB Camera,Display,Pass,Pass,Pass,Pass + 59,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 60,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,CSI Camera,Display,Pass,Pass,Pass,Pass + 61,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 62,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,RPI Camera,Display,Pass,Pass,Pass,Pass + 63,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 64,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,RTSP - Video,Display,Pass,Pass,Pass,Pass + 65,TFL-OD-2020-ssdLite-mobDet-DSP-coco-320x320,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 66,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,Image,Display,Pass,Pass,Pass,Pass + 67,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 68,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 69,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,Video,Display,Pass,Pass,Pass,Pass + 70,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 71,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,USB Camera,Display,Pass,Pass,Pass,Pass + 72,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 73,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,CSI Camera,Display,Pass,Pass,Pass,Pass + 74,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 75,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,RPI Camera,Display,Pass,Pass,Pass,Pass + 76,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 77,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,RTSP - Video,Display,Pass,Pass,Pass,Pass + 78,ONR-OD-8050-ssd-lite-regNetX-800mf-fpn-bgr-mmdet-coco-512x512,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 79,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,Image,Display,Pass,Pass,Pass,Pass + 80,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 81,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 82,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,Video,Display,Pass,Pass,Pass,Pass + 83,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 84,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,USB Camera,Display,Pass,Pass,Pass,Pass + 85,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 86,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,CSI Camera,Display,Pass,Pass,Pass,Pass + 87,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 88,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,RPI Camera,Display,Pass,Pass,Pass,Pass + 89,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 90,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,RTSP - Video,Display,Pass,Pass,Pass,Pass + 91,TVM-SS-5720-deeplabv3lite-regnetx800mf-cocoseg21-512x512,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 92,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,Image,Display,Pass,Pass,Pass,Pass + 93,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 94,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 95,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,Video,Display,Pass,Pass,Pass,Pass + 96,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 97,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,USB Camera,Display,Pass,Pass,Pass,Pass + 98,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 99,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,CSI Camera,Display,Pass,Pass,Pass,Pass + 100,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 101,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,RPI Camera,Display,Pass,Pass,Pass,Pass + 102,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 103,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,RTSP - Video,Display,Pass,Pass,Pass,Pass + 104,TFL-SS-2580-deeplabv3_mobv2-ade20k32-mlperf-512x512,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + 105,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,Image,Display,Pass,Pass,Pass,Pass + 106,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,Image,Video-Filewrite,Fail,Fail,Fail,Fail + 107,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,Image,Image-Filewrite,Pass,Pass,Pass,Pass + 108,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,Video,Display,Pass,Pass,Pass,Pass + 109,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,Video,Video-Filewrite,Pass,Pass,Pass,Pass + 110,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,USB Camera,Display,Pass,Pass,Pass,Pass + 111,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,USB Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 112,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,CSI Camera,Display,Pass,Pass,Pass,Pass + 113,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,CSI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 114,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,RPI Camera,Display,Pass,Pass,Pass,Pass + 115,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,RPI Camera,Video-Filewrite,Pass,Pass,Pass,Pass + 116,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,RTSP - Video,Display,Pass,Pass,Pass,Pass + 117,ONR-SS-8610-deeplabv3lite-mobv2-ade20k32-512x512,RTSP - Video,Video-Filewrite,Pass,Pass,Pass,Pass + + +Single Input Multi Output +------------------------- +.. csv-table:: + :header: "Category", "# test case", "Pass", "Fail" + + Host OS - Python,15,15,0 + docker - Python,15,15,0 + Host OS - C++,15,15,0 + Docker - C++,15,15,0 +.. csv-table:: + :header: "S.No", "Models", "Input", "Output", "Host OS-C++", "Host OS-Python", "Docker-C++", "Docker-Python", "Comments" + + 1,"2 Models (TFL-CL, ONR-SS)",%04d.jpg,Display,Pass,Pass,Pass,Pass, + 2,"3-Models (TVM-CL, TFL-OD, ONR-SS)",%04d.jpg,Display,Pass,Pass,Pass,Pass, + 3,"4-Models (TVM-SS, TFL-OD, ONR-SS, ONR-CL)",%04d.jpg,Display,Pass,Pass,Pass,Pass, + 4,"2 Models (TFL-CL, ONR-SS)",video_0000.mp4,Display,Pass,Pass,Pass,Pass, + 5,"3-Models (TVM-CL, TFL-OD, ONR-SS)",video_0000.mp4,Display,Pass,Pass,Pass,Pass, + 6,"4-Models (TVM-SS, TFL-OD, ONR-SS, ONR-CL)",video_0000.mp4,Display,Pass,Pass,Pass,Pass, + 7,"2 Models (TFL-CL, ONR-SS)",USB_camera,Display,Pass,Pass,Pass,Pass, + 8,"3-Models (TVM-CL, TFL-OD, ONR-SS)",USB_camera,Display,Pass,Pass,Pass,Pass, + 9,"4-Models (TVM-SS, TFL-OD, ONR-SS, ONR-CL)",USB_camera,Display,Pass,Pass,Pass,Pass, + 10,"2 Models (TFL-CL, ONR-SS)",CSI_camera,Display,Pass,Pass,Pass,Pass, + 11,"3-Models (TVM-CL, TFL-OD, ONR-SS)",CSI_camera,Display,Pass,Pass,Pass,Pass, + 12,"4-Models (TVM-SS, TFL-OD, ONR-SS, ONR-CL)",CSI_camera,Display,Pass,Pass,Pass,Pass, + 13,"2 Models (TFL-CL, ONR-SS)",rtsp,Display,Pass,Pass,Pass,Pass, + 14,"3-Models (TVM-CL, TFL-OD, ONR-SS)",rtsp,Display,Pass,Pass,Pass,Pass, + 15,"4-Models (TVM-SS, TFL-OD, ONR-SS, ONR-CL)",rtsp,Display,Pass,Pass,Pass,Pass, + +Multi Input Multi Output +------------------------ +.. csv-table:: + :header: "Category", "# test case", "Pass", "Fail" + + Host OS - Python,8,8,0 + docker - Python,8,8,0 + Host OS - C++,8,8,0 + Docker - C++,8,8,0 +.. csv-table:: + :header: "S.No", "Models", "Input", "Output", "Host OS-C++", "Host OS-Python", "Docker-C++", "Docker-Python", "Comments" + + 1,"2 Models (TVM-CL, TFL-OD)","%04d.jpg,video_0000.mp4",Display,Pass,Pass,Pass,Pass, + 2,"2 Models (TVM-OD, ONR-SS)","%04d.jpg,rtsp",Video-Filewrite,Pass,Pass,Pass,Pass, + 3,"2 Models (ONR-CL, TVM-SS)","%04d.jpg,USB_camera",Display,Pass,Pass,Pass,Pass, + 4,"3-Models (TVM-CL, TFL-OD, ONR-SS)","%04d.jpg,CSI_camera,rtsp",Video-Filewrite,Pass,Pass,Pass,Pass, + 5,"3-Models (TVM-CL, TFL-OD, ONR-SS)","video_0000.mp4,rtsp,%04d.jpg",Display,Pass,Pass,Pass,Pass, + 6,"3-Models (TFL-CL, ONR-CL, TVM-SS)","video_0000.mp4,USB_camera,CSI_camera",Video-Filewrite,Pass,Pass,Pass,Pass, + 7,"4-Models (TVM-CL, TFL-SS, ONR-OD, TFL-CL)","USB_camera,CSI_camera",Display,Pass,Pass,Pass,Pass, + 8,"4-Models (TVM-SS, TFL-SS, ONR-SS, ONR-OD)","USB_camera,video_0000.mp4",Video-Filewrite,Pass,Pass,Pass,Pass, + +.. note:: + * Video file from RTSP server used for RTSP input test + * Please refer to the :ref:`pub_edgeai_known_issues` section for more details diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/__init__.py b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/__init__.py new file mode 100644 index 0000000000000000000000000000000000000000..88ddcce25cef361552426e7288e24197b29a89ea --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/__init__.py @@ -0,0 +1,17 @@ +"""Sphinx ReadTheDocs theme. + +From https://github.com/ryan-roemer/sphinx-bootstrap-theme. + +""" +import os + +VERSION = (0, 1, 9) + +__version__ = ".".join(str(v) for v in VERSION) +__version_full__ = __version__ + + +def get_html_theme_path(): + """Return list of HTML theme paths.""" + cur_dir = os.path.abspath(os.path.dirname(os.path.dirname(__file__))) + return cur_dir diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/breadcrumbs.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/breadcrumbs.html new file mode 100644 index 0000000000000000000000000000000000000000..131d497c75408d3d5e8f4b3a2dd4af2f406a414b --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/breadcrumbs.html @@ -0,0 +1,31 @@ +{# Support for Sphinx 1.3+ page_source_suffix, but don't break old builds. #} + +{% if page_source_suffix %} +{% set suffix = page_source_suffix %} +{% else %} +{% set suffix = source_suffix %} +{% endif %} + +<div role="navigation" aria-label="breadcrumbs navigation"> + <ul class="wy-breadcrumbs"> + <li><a href="{{ pathto(master_doc) }}">Docs</a> »</li> + {% for doc in parents %} + <li><a href="{{ doc.link|e }}">{{ doc.title }}</a> »</li> + {% endfor %} + <li>{{ title }}</li> + <li class="wy-breadcrumbs-aside"> + {% if pagename != "search" %} + {% if display_github %} + <a href="https://{{ github_host|default("github.com") }}/{{ github_user }}/{{ github_repo }}/blob/{{ github_version }}{{ conf_py_path }}{{ pagename }}{{ suffix }}" class="fa fa-github"> Edit on GitHub</a> + {% elif display_bitbucket %} + <a href="https://bitbucket.org/{{ bitbucket_user }}/{{ bitbucket_repo }}/src/{{ bitbucket_version}}{{ conf_py_path }}{{ pagename }}{{ suffix }}" class="fa fa-bitbucket"> Edit on Bitbucket</a> + {% elif show_source and source_url_prefix %} + <a href="{{ source_url_prefix }}{{ pagename }}{{ suffix }}">View page source</a> + {% elif show_source and has_source and sourcename %} + <a href="{{ pathto('_sources/' + sourcename, true)|e }}" rel="nofollow"> View page source</a> + {% endif %} + {% endif %} + </li> + </ul> + <hr/> +</div> diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/footer.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/footer.html new file mode 100644 index 0000000000000000000000000000000000000000..1c9eb567ac4c8ff7ce31f1f96b63bb3134fb80a9 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/footer.html @@ -0,0 +1,47 @@ +<footer> + {% if next or prev %} + <div class="rst-footer-buttons" role="navigation" aria-label="footer navigation"> + {% if next %} + <a href="{{ next.link|e }}" class="btn btn-neutral float-right" title="{{ next.title|striptags|e }}" accesskey="n">Next <span class="fa fa-arrow-circle-right"></span></a> + {% endif %} + {% if prev %} + <a href="{{ prev.link|e }}" class="btn btn-neutral" title="{{ prev.title|striptags|e }}" accesskey="p"><span class="fa fa-arrow-circle-left"></span> Previous</a> + {% endif %} + </div> + {% endif %} + + <hr/> + + <div role="contentinfo"> + <p> + <a href="http://www.ti.com/corp/docs/legal/copyright.shtml">{{ copyright }}</a>, Texas Instruments Incorporated. All rights reserved. <br> + <a href="http://www.ti.com/corp/docs/legal/trademark/trademrk.htm">Trademarks</a> | <a href="http://www.ti.com/corp/docs/legal/privacy.shtml">Privacy policy</a> | <a href="http://www.ti.com/corp/docs/legal/termsofuse.shtml">Terms of use</a> | <a href="http://www.ti.com/lsds/ti/legal/termsofsale.page">Terms of sale</a> + + {%- if build_id and build_url %} + {% trans build_url=build_url, build_id=build_id %} + <span class="build"> + Build + <a href="{{ build_url }}">{{ build_id }}</a>. + </span> + {% endtrans %} + {%- elif commit %} + {% trans commit=commit %} + <span class="commit"> + Revision <code>{{ commit }}</code>. + </span> + {% endtrans %} + {%- elif last_updated %} + {% trans last_updated=last_updated|e %}Last updated on {{ last_updated }}.{% endtrans %} + {%- endif %} + + </p> + </div> + + {%- if show_sphinx %} + {% trans %}Built with <a href="http://sphinx-doc.org/">Sphinx</a> using a <a href="https://github.com/snide/sphinx_rtd_theme">theme</a> provided by <a href="https://readthedocs.org">Read the Docs</a>{% endtrans %}. + {%- endif %} + + {%- block extrafooter %} {% endblock %} + +</footer> + diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/layout.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/layout.html new file mode 100644 index 0000000000000000000000000000000000000000..b39100465e132e5c3ee8df3334b93eece1aebe99 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/layout.html @@ -0,0 +1,230 @@ +{# TEMPLATE VAR SETTINGS #} +{%- set url_root = pathto('', 1) %} +{%- if url_root == '#' %}{% set url_root = '' %}{% endif %} +{%- if not embedded and docstitle %} + {%- set titlesuffix = " — "|safe + docstitle|e %} +{%- else %} + {%- set titlesuffix = "" %} +{%- endif %} + +<!DOCTYPE html> +<!--[if IE 8]><html class="no-js lt-ie9" lang="en" > <![endif]--> +<!--[if gt IE 8]><!--> <html class="no-js" lang="en" > <!--<![endif]--> +<head> + <meta charset="utf-8"> + {{ metatags }} + <meta name="viewport" content="width=device-width, initial-scale=1.0"> + {% block htmltitle %} + <title>{{ title|striptags|e }}{{ titlesuffix }}</title> + {% endblock %} + + {# FAVICON #} + {% if favicon %} + <link rel="shortcut icon" href="{{ pathto('_static/' + favicon, 1) }}"/> + {% endif %} + + {# CSS #} + + {# OPENSEARCH #} + {% if not embedded %} + {% if use_opensearch %} + <link rel="search" type="application/opensearchdescription+xml" title="{% trans docstitle=docstitle|e %}Search within {{ docstitle }}{% endtrans %}" href="{{ pathto('_static/opensearch.xml', 1) }}"/> + {% endif %} + + {% endif %} + + {# RTD hosts this file, so just load on non RTD builds #} + {% if not READTHEDOCS %} + <link rel="stylesheet" href="{{ pathto('_static/' + style, 1) }}" type="text/css" /> + {% endif %} + + {% for cssfile in css_files %} + <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" /> + {% endfor %} + + {% for cssfile in extra_css_files %} + <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" /> + {% endfor %} + + {%- block linktags %} + {%- if hasdoc('about') %} + <link rel="author" title="{{ _('About these documents') }}" + href="{{ pathto('about') }}"/> + {%- endif %} + {%- if hasdoc('genindex') %} + <link rel="index" title="{{ _('Index') }}" + href="{{ pathto('genindex') }}"/> + {%- endif %} + {%- if hasdoc('search') %} + <link rel="search" title="{{ _('Search') }}" href="{{ pathto('search') }}"/> + {%- endif %} + {%- if hasdoc('copyright') %} + <link rel="copyright" title="{{ _('Copyright') }}" href="{{ pathto('copyright') }}"/> + {%- endif %} + <link rel="top" title="{{ docstitle|e }}" href="{{ pathto('index') }}"/> + {%- if parents %} + <link rel="up" title="{{ parents[-1].title|striptags|e }}" href="{{ parents[-1].link|e }}"/> + {%- endif %} + {%- if next %} + <link rel="next" title="{{ next.title|striptags|e }}" href="{{ next.link|e }}"/> + {%- endif %} + {%- if prev %} + <link rel="prev" title="{{ prev.title|striptags|e }}" href="{{ prev.link|e }}"/> + {%- endif %} + {%- endblock %} + {%- block extrahead %} {% endblock %} + + {# Keep modernizr in head - http://modernizr.com/docs/#installing #} + <script src="{{ pathto('_static/js/modernizr.min.js', 1) }}"></script> + +</head> + +<body class="wy-body-for-nav" role="document"> + <header id="tiHeader"> + <div class="top"> + <ul> + <li id="top_logo"> + <a href="http://www.ti.com"> + <img src="{{ pathto('_static/img/ti_logo.png', 1) }}"/> + </a> + </li> + </ul> + </div> + <div class="nav"></div> + </header> + <div class="wy-grid-for-nav"> + + {# SIDE NAV, TOGGLES ON MOBILE #} + <nav data-toggle="wy-nav-shift" class="wy-nav-side"> + <div class="wy-side-scroll"> + <div class="wy-side-nav-search"> + {% block sidebartitle %} + + {% if logo and theme_logo_only %} + <a href="{{ pathto(master_doc) }}"> + {% else %} + <a href="{{ pathto(master_doc) }}" class="icon icon-home"> {{ project }} + {% endif %} + + {% if logo %} + {# Not strictly valid HTML, but it's the only way to display/scale it properly, without weird scripting or heaps of work #} + <img src="{{ pathto('_static/' + logo, 1) }}" class="logo" /> + {% endif %} + </a> + + {% if theme_display_version %} + {%- set nav_version = version %} + {% if READTHEDOCS and current_version %} + {%- set nav_version = current_version %} + {% endif %} + {% if nav_version %} + <div class="version"> + {{ nav_version }} + </div> + {% endif %} + {% endif %} + + {% include "searchbox.html" %} + + {% endblock %} + </div> + + <div class="wy-menu wy-menu-vertical" data-spy="affix" role="navigation" aria-label="main navigation"> + {% block menu %} + {% set toctree = toctree(maxdepth=4, collapse=theme_collapse_navigation, includehidden=True) %} + {% if toctree %} + {{ toctree }} + {% else %} + <!-- Local TOC --> + <div class="local-toc">{{ toc }}</div> + {% endif %} + {% endblock %} + </div> + </div> + </nav> + + <section data-toggle="wy-nav-shift" class="wy-nav-content-wrap"> + + {# MOBILE NAV, TRIGGLES SIDE NAV ON TOGGLE #} + <nav class="wy-nav-top" role="navigation" aria-label="top navigation"> + <i data-toggle="wy-nav-top" class="fa fa-bars"></i> + <a href="{{ pathto(master_doc) }}">{{ project }}</a> + </nav> + + + {# PAGE CONTENT #} + <div class="wy-nav-content"> + <div class="rst-content"> + {% include "breadcrumbs.html" %} + <div role="main" class="document" itemscope="itemscope" itemtype="http://schema.org/Article"> + <div itemprop="articleBody"> + {% block body %}{% endblock %} + </div> + </div> + {% include "footer.html" %} + </div> + </div> + + </section> + + </div> + {% include "versions.html" %} + + {% if not embedded %} + + <script id="documentation_options" data-url_root="{{ url_root }}" src="{{ pathto('_static/documentation_options.js', 1) }}"></script> + {%- for scriptfile in script_files %} + <script type="text/javascript" src="{{ pathto(scriptfile, 1) }}"></script> + {%- endfor %} + + <script src="http://www.ti.com/assets/js/headerfooter/analytics.js" type="text/javascript" charset="utf-8"></script> + + {% endif %} + + {# RTD hosts this file, so just load on non RTD builds #} + {% if not READTHEDOCS %} + <script type="text/javascript" src="{{ pathto('_static/js/theme.js', 1) }}"></script> + {% endif %} + + {# STICKY NAVIGATION #} + {% if theme_sticky_navigation %} + <script type="text/javascript"> + jQuery(function () { + SphinxRtdTheme.StickyNav.enable(); + }); + + var menuHeight = window.innerHeight; + + var contentOffset = $(".wy-nav-content-wrap").offset(); + var contentHeight = $(".wy-nav-content-wrap").height(); + var contentBottom = contentOffset.top + contentHeight; + + function setNavbarTop() { + var scrollTop = $(window).scrollTop(); + var maxTop = scrollTop + menuHeight; + + // If past the header + if (scrollTop > contentOffset.top && maxTop < contentBottom) { + stickyTop = scrollTop - contentOffset.top; + } else if (maxTop > contentBottom) { + stickyTop = scrollTop - contentOffset.top - (maxTop - contentBottom); + } else { + stickyTop = 0; + } + + $(".wy-nav-side").css("top", stickyTop); + } + + $(document).ready(function() { + setNavbarTop(); + $(window).scroll(function () { + setNavbarTop(); + }); + }); + </script> + {% endif %} + + {%- block footer %} {% endblock %} + +</body> +</html> diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/layout_old.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/layout_old.html new file mode 100644 index 0000000000000000000000000000000000000000..deb8df2a1a7489361e005f71114907dbead9a4b8 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/layout_old.html @@ -0,0 +1,205 @@ +{# + basic/layout.html + ~~~~~~~~~~~~~~~~~ + + Master layout template for Sphinx themes. + + :copyright: Copyright 2007-2013 by the Sphinx team, see AUTHORS. + :license: BSD, see LICENSE for details. +#} +{%- block doctype -%} +<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" + "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> +{%- endblock %} +{%- set reldelim1 = reldelim1 is not defined and ' »' or reldelim1 %} +{%- set reldelim2 = reldelim2 is not defined and ' |' or reldelim2 %} +{%- set render_sidebar = (not embedded) and (not theme_nosidebar|tobool) and + (sidebars != []) %} +{%- set url_root = pathto('', 1) %} +{# XXX necessary? #} +{%- if url_root == '#' %}{% set url_root = '' %}{% endif %} +{%- if not embedded and docstitle %} + {%- set titlesuffix = " — "|safe + docstitle|e %} +{%- else %} + {%- set titlesuffix = "" %} +{%- endif %} + +{%- macro relbar() %} + <div class="related"> + <h3>{{ _('Navigation') }}</h3> + <ul> + {%- for rellink in rellinks %} + <li class="right" {% if loop.first %}style="margin-right: 10px"{% endif %}> + <a href="{{ pathto(rellink[0]) }}" title="{{ rellink[1]|striptags|e }}" + {{ accesskey(rellink[2]) }}>{{ rellink[3] }}</a> + {%- if not loop.first %}{{ reldelim2 }}{% endif %}</li> + {%- endfor %} + {%- block rootrellink %} + <li><a href="{{ pathto(master_doc) }}">{{ shorttitle|e }}</a>{{ reldelim1 }}</li> + {%- endblock %} + {%- for parent in parents %} + <li><a href="{{ parent.link|e }}" {% if loop.last %}{{ accesskey("U") }}{% endif %}>{{ parent.title }}</a>{{ reldelim1 }}</li> + {%- endfor %} + {%- block relbaritems %} {% endblock %} + </ul> + </div> +{%- endmacro %} + +{%- macro sidebar() %} + {%- if render_sidebar %} + <div class="sphinxsidebar"> + <div class="sphinxsidebarwrapper"> + {%- block sidebarlogo %} + {%- if logo %} + <p class="logo"><a href="{{ pathto(master_doc) }}"> + <img class="logo" src="{{ pathto('_static/' + logo, 1) }}" alt="Logo"/> + </a></p> + {%- endif %} + {%- endblock %} + {%- if sidebars != None %} + {#- new style sidebar: explicitly include/exclude templates #} + {%- for sidebartemplate in sidebars %} + {%- include sidebartemplate %} + {%- endfor %} + {%- else %} + {#- old style sidebars: using blocks -- should be deprecated #} + {%- block sidebartoc %} + {%- include "localtoc.html" %} + {%- endblock %} + {%- block sidebarrel %} + {%- include "relations.html" %} + {%- endblock %} + {%- block sidebarsourcelink %} + {%- include "sourcelink.html" %} + {%- endblock %} + {%- if customsidebar %} + {%- include customsidebar %} + {%- endif %} + {%- block sidebarsearch %} + {%- include "searchbox.html" %} + {%- endblock %} + {%- endif %} + </div> + </div> + {%- endif %} +{%- endmacro %} + +{%- macro script() %} + <script type="text/javascript"> + var DOCUMENTATION_OPTIONS = { + URL_ROOT: '{{ url_root }}', + VERSION: '{{ release|e }}', + COLLAPSE_INDEX: false, + FILE_SUFFIX: '{{ '' if no_search_suffix else file_suffix }}', + HAS_SOURCE: {{ has_source|lower }} + }; + </script> + {%- for scriptfile in script_files %} + <script type="text/javascript" src="{{ pathto(scriptfile, 1) }}"></script> + {%- endfor %} +{%- endmacro %} + +{%- macro css() %} + <link rel="stylesheet" href="{{ pathto('_static/' + style, 1) }}" type="text/css" /> + <link rel="stylesheet" href="{{ pathto('_static/pygments.css', 1) }}" type="text/css" /> + {%- for cssfile in css_files %} + <link rel="stylesheet" href="{{ pathto(cssfile, 1) }}" type="text/css" /> + {%- endfor %} +{%- endmacro %} + +<html xmlns="http://www.w3.org/1999/xhtml"> + <head> + <meta http-equiv="Content-Type" content="text/html; charset={{ encoding }}" /> + {{ metatags }} + {%- block htmltitle %} + <title>{{ title|striptags|e }}{{ titlesuffix }}</title> + {%- endblock %} + {{ css() }} + {%- if not embedded %} + {{ script() }} + {%- if use_opensearch %} + <link rel="search" type="application/opensearchdescription+xml" + title="{% trans docstitle=docstitle|e %}Search within {{ docstitle }}{% endtrans %}" + href="{{ pathto('_static/opensearch.xml', 1) }}"/> + {%- endif %} + {%- if favicon %} + <link rel="shortcut icon" href="{{ pathto('_static/' + favicon, 1) }}"/> + {%- endif %} + {%- endif %} +{%- block linktags %} + {%- if hasdoc('about') %} + <link rel="author" title="{{ _('About these documents') }}" href="{{ pathto('about') }}" /> + {%- endif %} + {%- if hasdoc('genindex') %} + <link rel="index" title="{{ _('Index') }}" href="{{ pathto('genindex') }}" /> + {%- endif %} + {%- if hasdoc('search') %} + <link rel="search" title="{{ _('Search') }}" href="{{ pathto('search') }}" /> + {%- endif %} + {%- if hasdoc('copyright') %} + <link rel="copyright" title="{{ _('Copyright') }}" href="{{ pathto('copyright') }}" /> + {%- endif %} + <link rel="top" title="{{ docstitle|e }}" href="{{ pathto('index') }}" /> + {%- if parents %} + <link rel="up" title="{{ parents[-1].title|striptags|e }}" href="{{ parents[-1].link|e }}" /> + {%- endif %} + {%- if next %} + <link rel="next" title="{{ next.title|striptags|e }}" href="{{ next.link|e }}" /> + {%- endif %} + {%- if prev %} + <link rel="prev" title="{{ prev.title|striptags|e }}" href="{{ prev.link|e }}" /> + {%- endif %} +{%- endblock %} +{%- block extrahead %} {% endblock %} + </head> + <body> +{%- block header %}{% endblock %} + +{%- block relbar1 %}{{ relbar() }}{% endblock %} + +{%- block content %} + {%- block sidebar1 %} {# possible location for sidebar #} {% endblock %} + + <div class="document"> + {%- block document %} + <div class="documentwrapper"> + {%- if render_sidebar %} + <div class="bodywrapper"> + {%- endif %} + <div class="body"> + {% block body %} {% endblock %} + </div> + {%- if render_sidebar %} + </div> + {%- endif %} + </div> + {%- endblock %} + + {%- block sidebar2 %}{{ sidebar() }}{% endblock %} + <div class="clearer"></div> + </div> +{%- endblock %} + +{%- block relbar2 %}{{ relbar() }}{% endblock %} + +{%- block footer %} + <div class="footer"> + {%- if show_copyright %} + {%- if hasdoc('copyright') %} + {% trans path=pathto('copyright'), copyright=copyright|e %}© <a href="{{ path }}">Copyright</a> {{ copyright }}.{% endtrans %} + {%- else %} + {% trans copyright=copyright|e %}© Copyright {{ copyright }}.{% endtrans %} + {%- endif %} + {%- endif %} + {%- if last_updated %} + {% trans last_updated=last_updated|e %}Last updated on {{ last_updated }}.{% endtrans %} + {%- endif %} + {%- if show_sphinx %} + {% trans sphinx_version=sphinx_version|e %}Created using <a href="http://sphinx-doc.org/">Sphinx</a> {{ sphinx_version }}.{% endtrans %} + {%- endif %} + </div> + <p>asdf asdf asdf asdf 22</p> +{%- endblock %} + </body> +</html> + diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/search.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/search.html new file mode 100644 index 0000000000000000000000000000000000000000..331ce7a540d1093bc09b9bce54ab8f799409fdd3 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/search.html @@ -0,0 +1,52 @@ +{# + basic/search.html + ~~~~~~~~~~~~~~~~~ + + Template for the search page. + + :copyright: Copyright 2007-2013 by the Sphinx team, see AUTHORS. + :license: BSD, see LICENSE for details. +#} +{%- extends "layout.html" %} +{% set title = _('Search') %} +{% set script_files = script_files + ['_static/searchtools.js'] %} +{% set script_files = script_files + ['_static/language_data.js'] %} +{% block footer %} + <script type="text/javascript"> + jQuery(function() { Search.loadIndex("{{ pathto('searchindex.js', 1) }}"); }); + jQuery(function() { Search.loadIndex("{{ pathto('language_data.js', 1) }}"); }); + </script> + {# this is used when loading the search index using $.ajax fails, + such as on Chrome for documents on localhost #} + <script type="text/javascript" id="searchindexloader"></script> + {{ super() }} +{% endblock %} +{% block body %} + <noscript> + <div id="fallback" class="admonition warning"> + <p class="last"> + {% trans %}Please activate JavaScript to enable the search + functionality.{% endtrans %} + </p> + </div> + </noscript> + + {% if search_performed %} + <h2>{{ _('Search Results') }}</h2> + {% if not search_results %} + <p>{{ _('Your search did not match any documents. Please make sure that all words are spelled correctly and that you\'ve selected enough categories.') }}</p> + {% endif %} + {% endif %} + <div id="search-results"> + {% if search_results %} + <ul> + {% for href, caption, context in search_results %} + <li> + <a href="{{ pathto(item.href) }}">{{ caption }}</a> + <p class="context">{{ context|e }}</p> + </li> + {% endfor %} + </ul> + {% endif %} + </div> +{% endblock %} diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/searchbox.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/searchbox.html new file mode 100644 index 0000000000000000000000000000000000000000..35ad52c5f63c7c8446778ee360ee5854dbd75156 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/searchbox.html @@ -0,0 +1,9 @@ +{%- if builder != 'singlehtml' %} +<div role="search"> + <form id="rtd-search-form" class="wy-form" action="{{ pathto('search') }}" method="get"> + <input type="text" name="q" placeholder="Search docs" /> + <input type="hidden" name="check_keywords" value="yes" /> + <input type="hidden" name="area" value="default" /> + </form> +</div> +{%- endif %} diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/css/badge_only.css b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/css/badge_only.css new file mode 100644 index 0000000000000000000000000000000000000000..7e17fb148c63fa9780c3dd65cef5b7593927ef62 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/css/badge_only.css @@ -0,0 +1,2 @@ +.fa:before{-webkit-font-smoothing:antialiased}.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;content:""}.clearfix:after{clear:both}@font-face{font-family:FontAwesome;font-weight:normal;font-style:normal;src:url("../font/fontawesome_webfont.eot");src:url("../font/fontawesome_webfont.eot?#iefix") format("embedded-opentype"),url("../font/fontawesome_webfont.woff") format("woff"),url("../font/fontawesome_webfont.ttf") format("truetype"),url("../font/fontawesome_webfont.svg#FontAwesome") format("svg")}.fa:before{display:inline-block;font-family:FontAwesome;font-style:normal;font-weight:normal;line-height:1;text-decoration:inherit}a .fa{display:inline-block;text-decoration:inherit}li .fa{display:inline-block}li .fa-large:before,li .fa-large:before{width:1.875em}ul.fas{list-style-type:none;margin-left:2em;text-indent:-0.8em}ul.fas li .fa{width:0.8em}ul.fas li .fa-large:before,ul.fas li .fa-large:before{vertical-align:baseline}.fa-book:before{content:"ï€"}.icon-book:before{content:"ï€"}.fa-caret-down:before{content:""}.icon-caret-down:before{content:""}.fa-caret-up:before{content:""}.icon-caret-up:before{content:""}.fa-caret-left:before{content:""}.icon-caret-left:before{content:""}.fa-caret-right:before{content:""}.icon-caret-right:before{content:""}.rst-versions{position:fixed;bottom:0;left:0;width:300px;color:#fcfcfc;background:#1f1d1d;border-top:solid 10px #343131;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;z-index:400}.rst-versions a{color:#2980B9;text-decoration:none}.rst-versions .rst-badge-small{display:none}.rst-versions .rst-current-version{padding:12px;background-color:#272525;display:block;text-align:right;font-size:90%;cursor:pointer;color:#27AE60;*zoom:1}.rst-versions .rst-current-version:before,.rst-versions .rst-current-version:after{display:table;content:""}.rst-versions .rst-current-version:after{clear:both}.rst-versions .rst-current-version .fa{color:#fcfcfc}.rst-versions .rst-current-version .fa-book{float:left}.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version.rst-out-of-date{background-color:#E74C3C;color:#fff}.rst-versions .rst-current-version.rst-active-old-version{background-color:#F1C40F;color:#000}.rst-versions.shift-up .rst-other-versions{display:block}.rst-versions .rst-other-versions{font-size:90%;padding:12px;color:gray;display:none}.rst-versions .rst-other-versions hr{display:block;height:1px;border:0;margin:20px 0;padding:0;border-top:solid 1px #413d3d}.rst-versions .rst-other-versions dd{display:inline-block;margin:0}.rst-versions .rst-other-versions dd a{display:inline-block;padding:6px;color:#fcfcfc}.rst-versions.rst-badge{width:auto;bottom:20px;right:20px;left:auto;border:none;max-width:300px}.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge .fa-book{float:none}.rst-versions.rst-badge.shift-up .rst-current-version{text-align:right}.rst-versions.rst-badge.shift-up .rst-current-version .fa-book{float:left}.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge .rst-current-version{width:auto;height:30px;line-height:30px;padding:0 6px;display:block;text-align:center}@media screen and (max-width: 768px){.rst-versions{width:85%;display:none}.rst-versions.shift{display:block}img{width:100%;height:auto}} +/*# sourceMappingURL=badge_only.css.map */ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/css/theme.css b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/css/theme.css new file mode 100644 index 0000000000000000000000000000000000000000..7be93399a4f530da7eb43e2c214ec42cea89a6c3 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/css/theme.css @@ -0,0 +1,5 @@ +*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}article,aside,details,figcaption,figure,footer,header,hgroup,nav,section{display:block}audio,canvas,video{display:inline-block;*display:inline;*zoom:1}audio:not([controls]){display:none}[hidden]{display:none}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:100%;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}body{margin:0}a:hover,a:active{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:bold}blockquote{margin:0}dfn{font-style:italic}ins{background:#ff9;color:#000;text-decoration:none}mark{background:#ff0;color:#000;font-style:italic;font-weight:bold}pre,code,.rst-content tt,.rst-content code,kbd,samp{font-family:monospace,serif;_font-family:"courier new",monospace;font-size:1em}pre{white-space:pre}q{quotes:none}q:before,q:after{content:"";content:none}small{font-size:85%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-0.5em}sub{bottom:-0.25em}ul,ol,dl{margin:0;padding:0;list-style:none;list-style-image:none}li{list-style:none}dd{margin:0}img{border:0;-ms-interpolation-mode:bicubic;vertical-align:middle;max-width:100%}svg:not(:root){overflow:hidden}figure{margin:0}form{margin:0}fieldset{border:0;margin:0;padding:0}label{cursor:pointer}legend{border:0;*margin-left:-7px;padding:0;white-space:normal}button,input,select,textarea{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle}button,input{line-height:normal}button,input[type="button"],input[type="reset"],input[type="submit"]{cursor:pointer;-webkit-appearance:button;*overflow:visible}button[disabled],input[disabled]{cursor:default}input[type="checkbox"],input[type="radio"]{box-sizing:border-box;padding:0;*width:13px;*height:13px}input[type="search"]{-webkit-appearance:textfield;-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box}input[type="search"]::-webkit-search-decoration,input[type="search"]::-webkit-search-cancel-button{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top;resize:vertical}table{border-collapse:collapse;border-spacing:0}td{vertical-align:top}.chromeframe{margin:0.2em 0;background:#ccc;color:#000;padding:0.2em 0}.ir{display:block;border:0;text-indent:-999em;overflow:hidden;background-color:transparent;background-repeat:no-repeat;text-align:left;direction:ltr;*line-height:0}.ir br{display:none}.hidden{display:none !important;visibility:hidden}.visuallyhidden{border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px}.visuallyhidden.focusable:active,.visuallyhidden.focusable:focus{clip:auto;height:auto;margin:0;overflow:visible;position:static;width:auto}.invisible{visibility:hidden}.relative{position:relative}big,small{font-size:100%}@media print{html,body,section{background:none !important}*{box-shadow:none !important;text-shadow:none !important;filter:none !important;-ms-filter:none !important}a,a:visited{text-decoration:underline}.ir a:after,a[href^="javascript:"]:after,a[href^="#"]:after{content:""}pre,blockquote{page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}img{max-width:100% !important}@page{margin:0.5cm}p,h2,.rst-content .toctree-wrapper p.caption,h3{orphans:3;widows:3}h2,.rst-content .toctree-wrapper p.caption,h3{page-break-after:avoid}}.fa:before,.wy-menu-vertical li span.toctree-expand:before,.wy-menu-vertical li.on a span.toctree-expand:before,.wy-menu-vertical li.current>a span.toctree-expand:before,.rst-content .admonition-title:before,.rst-content h1 .headerlink:before,.rst-content h2 .headerlink:before,.rst-content h3 .headerlink:before,.rst-content h4 .headerlink:before,.rst-content h5 .headerlink:before,.rst-content h6 .headerlink:before,.rst-content dl dt .headerlink:before,.rst-content p.caption .headerlink:before,.rst-content tt.download span:first-child:before,.rst-content code.download span:first-child:before,.icon:before,.wy-dropdown .caret:before,.wy-inline-validate.wy-inline-validate-success .wy-input-context:before,.wy-inline-validate.wy-inline-validate-danger .wy-input-context:before,.wy-inline-validate.wy-inline-validate-warning .wy-input-context:before,.wy-inline-validate.wy-inline-validate-info .wy-input-context:before,.wy-alert,.rst-content .note,.rst-content .attention,.rst-content .caution,.rst-content .danger,.rst-content .error,.rst-content .hint,.rst-content .important,.rst-content .tip,.rst-content .warning,.rst-content .seealso,.rst-content .admonition-todo,.btn,input[type="text"],input[type="password"],input[type="email"],input[type="url"],input[type="date"],input[type="month"],input[type="time"],input[type="datetime"],input[type="datetime-local"],input[type="week"],input[type="number"],input[type="search"],input[type="tel"],input[type="color"],select,textarea,.wy-menu-vertical li.on a,.wy-menu-vertical li.current>a,.wy-side-nav-search>a,.wy-side-nav-search .wy-dropdown>a,.wy-nav-top a{-webkit-font-smoothing:antialiased}.clearfix{*zoom:1}.clearfix:before,.clearfix:after{display:table;content:""}.clearfix:after{clear:both}/*! + * Font Awesome 4.2.0 by @davegandy - http://fontawesome.io - @fontawesome + * License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License) + */@font-face{font-family:'FontAwesome';src:url("../fonts/fontawesome-webfont.eot?v=4.2.0");src:url("../fonts/fontawesome-webfont.eot?#iefix&v=4.2.0") format("embedded-opentype"),url("../fonts/fontawesome-webfont.woff?v=4.2.0") format("woff"),url("../fonts/fontawesome-webfont.ttf?v=4.2.0") format("truetype"),url("../fonts/fontawesome-webfont.svg?v=4.2.0#fontawesomeregular") format("svg");font-weight:normal;font-style:normal}.fa,.wy-menu-vertical li span.toctree-expand,.wy-menu-vertical li.on a span.toctree-expand,.wy-menu-vertical li.current>a span.toctree-expand,.rst-content .admonition-title,.rst-content h1 .headerlink,.rst-content h2 .headerlink,.rst-content h3 .headerlink,.rst-content h4 .headerlink,.rst-content h5 .headerlink,.rst-content h6 .headerlink,.rst-content dl dt .headerlink,.rst-content p.caption .headerlink,.rst-content tt.download span:first-child,.rst-content code.download span:first-child,.icon{display:inline-block;font:normal normal normal 14px/1 FontAwesome;font-size:inherit;text-rendering:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.fa-lg{font-size:1.33333em;line-height:0.75em;vertical-align:-15%}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-fw{width:1.28571em;text-align:center}.fa-ul{padding-left:0;margin-left:2.14286em;list-style-type:none}.fa-ul>li{position:relative}.fa-li{position:absolute;left:-2.14286em;width:2.14286em;top:0.14286em;text-align:center}.fa-li.fa-lg{left:-1.85714em}.fa-border{padding:.2em .25em .15em;border:solid 0.08em #eee;border-radius:.1em}.pull-right{float:right}.pull-left{float:left}.fa.pull-left,.wy-menu-vertical li span.pull-left.toctree-expand,.wy-menu-vertical li.on a span.pull-left.toctree-expand,.wy-menu-vertical li.current>a span.pull-left.toctree-expand,.rst-content .pull-left.admonition-title,.rst-content h1 .pull-left.headerlink,.rst-content h2 .pull-left.headerlink,.rst-content h3 .pull-left.headerlink,.rst-content h4 .pull-left.headerlink,.rst-content h5 .pull-left.headerlink,.rst-content h6 .pull-left.headerlink,.rst-content dl dt .pull-left.headerlink,.rst-content p.caption .pull-left.headerlink,.rst-content tt.download span.pull-left:first-child,.rst-content code.download span.pull-left:first-child,.pull-left.icon{margin-right:.3em}.fa.pull-right,.wy-menu-vertical li span.pull-right.toctree-expand,.wy-menu-vertical li.on a span.pull-right.toctree-expand,.wy-menu-vertical li.current>a span.pull-right.toctree-expand,.rst-content .pull-right.admonition-title,.rst-content h1 .pull-right.headerlink,.rst-content h2 .pull-right.headerlink,.rst-content h3 .pull-right.headerlink,.rst-content h4 .pull-right.headerlink,.rst-content h5 .pull-right.headerlink,.rst-content h6 .pull-right.headerlink,.rst-content dl dt .pull-right.headerlink,.rst-content p.caption .pull-right.headerlink,.rst-content tt.download span.pull-right:first-child,.rst-content code.download span.pull-right:first-child,.pull-right.icon{margin-left:.3em}.fa-spin{-webkit-animation:fa-spin 2s infinite linear;animation:fa-spin 2s infinite linear}@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}.fa-rotate-90{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=1);-webkit-transform:rotate(90deg);-ms-transform:rotate(90deg);transform:rotate(90deg)}.fa-rotate-180{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=2);-webkit-transform:rotate(180deg);-ms-transform:rotate(180deg);transform:rotate(180deg)}.fa-rotate-270{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=3);-webkit-transform:rotate(270deg);-ms-transform:rotate(270deg);transform:rotate(270deg)}.fa-flip-horizontal{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=0);-webkit-transform:scale(-1, 1);-ms-transform:scale(-1, 1);transform:scale(-1, 1)}.fa-flip-vertical{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=2);-webkit-transform:scale(1, -1);-ms-transform:scale(1, -1);transform:scale(1, -1)}:root .fa-rotate-90,:root .fa-rotate-180,:root .fa-rotate-270,:root .fa-flip-horizontal,:root .fa-flip-vertical{filter:none}.fa-stack{position:relative;display:inline-block;width:2em;height:2em;line-height:2em;vertical-align:middle}.fa-stack-1x,.fa-stack-2x{position:absolute;left:0;width:100%;text-align:center}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa-glass:before{content:""}.fa-music:before{content:"ï€"}.fa-search:before,.icon-search:before{content:""}.fa-envelope-o:before{content:""}.fa-heart:before{content:""}.fa-star:before{content:""}.fa-star-o:before{content:""}.fa-user:before{content:""}.fa-film:before{content:""}.fa-th-large:before{content:""}.fa-th:before{content:""}.fa-th-list:before{content:""}.fa-check:before{content:""}.fa-remove:before,.fa-close:before,.fa-times:before{content:"ï€"}.fa-search-plus:before{content:""}.fa-search-minus:before{content:"ï€"}.fa-power-off:before{content:""}.fa-signal:before{content:""}.fa-gear:before,.fa-cog:before{content:""}.fa-trash-o:before{content:""}.fa-home:before,.icon-home:before{content:""}.fa-file-o:before{content:""}.fa-clock-o:before{content:""}.fa-road:before{content:""}.fa-download:before,.rst-content tt.download span:first-child:before,.rst-content code.download span:first-child:before{content:""}.fa-arrow-circle-o-down:before{content:""}.fa-arrow-circle-o-up:before{content:""}.fa-inbox:before{content:""}.fa-play-circle-o:before{content:"ï€"}.fa-rotate-right:before,.fa-repeat:before{content:""}.fa-refresh:before{content:""}.fa-list-alt:before{content:""}.fa-lock:before{content:""}.fa-flag:before{content:""}.fa-headphones:before{content:""}.fa-volume-off:before{content:""}.fa-volume-down:before{content:""}.fa-volume-up:before{content:""}.fa-qrcode:before{content:""}.fa-barcode:before{content:""}.fa-tag:before{content:""}.fa-tags:before{content:""}.fa-book:before,.icon-book:before{content:"ï€"}.fa-bookmark:before{content:""}.fa-print:before{content:""}.fa-camera:before{content:""}.fa-font:before{content:""}.fa-bold:before{content:""}.fa-italic:before{content:""}.fa-text-height:before{content:""}.fa-text-width:before{content:""}.fa-align-left:before{content:""}.fa-align-center:before{content:""}.fa-align-right:before{content:""}.fa-align-justify:before{content:""}.fa-list:before{content:""}.fa-dedent:before,.fa-outdent:before{content:""}.fa-indent:before{content:""}.fa-video-camera:before{content:""}.fa-photo:before,.fa-image:before,.fa-picture-o:before{content:""}.fa-pencil:before{content:"ï€"}.fa-map-marker:before{content:"ï"}.fa-adjust:before{content:"ï‚"}.fa-tint:before{content:"ïƒ"}.fa-edit:before,.fa-pencil-square-o:before{content:"ï„"}.fa-share-square-o:before{content:"ï…"}.fa-check-square-o:before{content:"ï†"}.fa-arrows:before{content:"ï‡"}.fa-step-backward:before{content:"ïˆ"}.fa-fast-backward:before{content:"ï‰"}.fa-backward:before{content:"ïŠ"}.fa-play:before{content:"ï‹"}.fa-pause:before{content:"ïŒ"}.fa-stop:before{content:"ï"}.fa-forward:before{content:"ïŽ"}.fa-fast-forward:before{content:"ï"}.fa-step-forward:before{content:"ï‘"}.fa-eject:before{content:"ï’"}.fa-chevron-left:before{content:"ï“"}.fa-chevron-right:before{content:"ï”"}.fa-plus-circle:before{content:"ï•"}.fa-minus-circle:before{content:"ï–"}.fa-times-circle:before,.wy-inline-validate.wy-inline-validate-danger .wy-input-context:before{content:"ï—"}.fa-check-circle:before,.wy-inline-validate.wy-inline-validate-success .wy-input-context:before{content:"ï˜"}.fa-question-circle:before{content:"ï™"}.fa-info-circle:before{content:"ïš"}.fa-crosshairs:before{content:"ï›"}.fa-times-circle-o:before{content:"ïœ"}.fa-check-circle-o:before{content:"ï"}.fa-ban:before{content:"ïž"}.fa-arrow-left:before{content:"ï "}.fa-arrow-right:before{content:"ï¡"}.fa-arrow-up:before{content:"ï¢"}.fa-arrow-down:before{content:"ï£"}.fa-mail-forward:before,.fa-share:before{content:"ï¤"}.fa-expand:before{content:"ï¥"}.fa-compress:before{content:"ï¦"}.fa-plus:before{content:"ï§"}.fa-minus:before{content:"ï¨"}.fa-asterisk:before{content:"ï©"}.fa-exclamation-circle:before,.wy-inline-validate.wy-inline-validate-warning .wy-input-context:before,.wy-inline-validate.wy-inline-validate-info .wy-input-context:before,.rst-content .admonition-title:before{content:"ïª"}.fa-gift:before{content:"ï«"}.fa-leaf:before{content:"ï¬"}.fa-fire:before,.icon-fire:before{content:"ï"}.fa-eye:before{content:"ï®"}.fa-eye-slash:before{content:"ï°"}.fa-warning:before,.fa-exclamation-triangle:before{content:"ï±"}.fa-plane:before{content:"ï²"}.fa-calendar:before{content:"ï³"}.fa-random:before{content:"ï´"}.fa-comment:before{content:"ïµ"}.fa-magnet:before{content:"ï¶"}.fa-chevron-up:before{content:"ï·"}.fa-chevron-down:before{content:"ï¸"}.fa-retweet:before{content:"ï¹"}.fa-shopping-cart:before{content:"ïº"}.fa-folder:before{content:"ï»"}.fa-folder-open:before{content:"ï¼"}.fa-arrows-v:before{content:"ï½"}.fa-arrows-h:before{content:"ï¾"}.fa-bar-chart-o:before,.fa-bar-chart:before{content:"ï‚€"}.fa-twitter-square:before{content:"ï‚"}.fa-facebook-square:before{content:"ï‚‚"}.fa-camera-retro:before{content:""}.fa-key:before{content:"ï‚„"}.fa-gears:before,.fa-cogs:before{content:"ï‚…"}.fa-comments:before{content:""}.fa-thumbs-o-up:before{content:""}.fa-thumbs-o-down:before{content:""}.fa-star-half:before{content:""}.fa-heart-o:before{content:"ï‚Š"}.fa-sign-out:before{content:"ï‚‹"}.fa-linkedin-square:before{content:"ï‚Œ"}.fa-thumb-tack:before{content:"ï‚"}.fa-external-link:before{content:"ï‚Ž"}.fa-sign-in:before{content:"ï‚"}.fa-trophy:before{content:"ï‚‘"}.fa-github-square:before{content:"ï‚’"}.fa-upload:before{content:"ï‚“"}.fa-lemon-o:before{content:"ï‚”"}.fa-phone:before{content:"ï‚•"}.fa-square-o:before{content:"ï‚–"}.fa-bookmark-o:before{content:"ï‚—"}.fa-phone-square:before{content:""}.fa-twitter:before{content:"ï‚™"}.fa-facebook:before{content:"ï‚š"}.fa-github:before,.icon-github:before{content:"ï‚›"}.fa-unlock:before{content:"ï‚œ"}.fa-credit-card:before{content:"ï‚"}.fa-rss:before{content:"ï‚ž"}.fa-hdd-o:before{content:"ï‚ "}.fa-bullhorn:before{content:"ï‚¡"}.fa-bell:before{content:""}.fa-certificate:before{content:"ï‚£"}.fa-hand-o-right:before{content:""}.fa-hand-o-left:before{content:"ï‚¥"}.fa-hand-o-up:before{content:""}.fa-hand-o-down:before{content:""}.fa-arrow-circle-left:before,.icon-circle-arrow-left:before{content:""}.fa-arrow-circle-right:before,.icon-circle-arrow-right:before{content:"ï‚©"}.fa-arrow-circle-up:before{content:""}.fa-arrow-circle-down:before{content:"ï‚«"}.fa-globe:before{content:""}.fa-wrench:before{content:"ï‚"}.fa-tasks:before{content:"ï‚®"}.fa-filter:before{content:"ï‚°"}.fa-briefcase:before{content:""}.fa-arrows-alt:before{content:""}.fa-group:before,.fa-users:before{content:""}.fa-chain:before,.fa-link:before,.icon-link:before{content:"ïƒ"}.fa-cloud:before{content:""}.fa-flask:before{content:""}.fa-cut:before,.fa-scissors:before{content:""}.fa-copy:before,.fa-files-o:before{content:""}.fa-paperclip:before{content:""}.fa-save:before,.fa-floppy-o:before{content:""}.fa-square:before{content:""}.fa-navicon:before,.fa-reorder:before,.fa-bars:before{content:""}.fa-list-ul:before{content:""}.fa-list-ol:before{content:""}.fa-strikethrough:before{content:""}.fa-underline:before{content:"ïƒ"}.fa-table:before{content:""}.fa-magic:before{content:"ïƒ"}.fa-truck:before{content:""}.fa-pinterest:before{content:""}.fa-pinterest-square:before{content:""}.fa-google-plus-square:before{content:""}.fa-google-plus:before{content:""}.fa-money:before{content:""}.fa-caret-down:before,.wy-dropdown .caret:before,.icon-caret-down:before{content:""}.fa-caret-up:before{content:""}.fa-caret-left:before{content:""}.fa-caret-right:before{content:""}.fa-columns:before{content:""}.fa-unsorted:before,.fa-sort:before{content:""}.fa-sort-down:before,.fa-sort-desc:before{content:"ïƒ"}.fa-sort-up:before,.fa-sort-asc:before{content:""}.fa-envelope:before{content:"ïƒ "}.fa-linkedin:before{content:""}.fa-rotate-left:before,.fa-undo:before{content:""}.fa-legal:before,.fa-gavel:before{content:""}.fa-dashboard:before,.fa-tachometer:before{content:""}.fa-comment-o:before{content:""}.fa-comments-o:before{content:""}.fa-flash:before,.fa-bolt:before{content:""}.fa-sitemap:before{content:""}.fa-umbrella:before{content:""}.fa-paste:before,.fa-clipboard:before{content:""}.fa-lightbulb-o:before{content:""}.fa-exchange:before{content:""}.fa-cloud-download:before{content:"ïƒ"}.fa-cloud-upload:before{content:""}.fa-user-md:before{content:""}.fa-stethoscope:before{content:""}.fa-suitcase:before{content:""}.fa-bell-o:before{content:"ï‚¢"}.fa-coffee:before{content:""}.fa-cutlery:before{content:""}.fa-file-text-o:before{content:""}.fa-building-o:before{content:""}.fa-hospital-o:before{content:""}.fa-ambulance:before{content:""}.fa-medkit:before{content:""}.fa-fighter-jet:before{content:""}.fa-beer:before{content:""}.fa-h-square:before{content:""}.fa-plus-square:before{content:""}.fa-angle-double-left:before{content:"ï„€"}.fa-angle-double-right:before{content:"ï„"}.fa-angle-double-up:before{content:"ï„‚"}.fa-angle-double-down:before{content:""}.fa-angle-left:before{content:"ï„„"}.fa-angle-right:before{content:"ï„…"}.fa-angle-up:before{content:""}.fa-angle-down:before{content:""}.fa-desktop:before{content:""}.fa-laptop:before{content:""}.fa-tablet:before{content:"ï„Š"}.fa-mobile-phone:before,.fa-mobile:before{content:"ï„‹"}.fa-circle-o:before{content:"ï„Œ"}.fa-quote-left:before{content:"ï„"}.fa-quote-right:before{content:"ï„Ž"}.fa-spinner:before{content:"ï„"}.fa-circle:before{content:"ï„‘"}.fa-mail-reply:before,.fa-reply:before{content:"ï„’"}.fa-github-alt:before{content:"ï„“"}.fa-folder-o:before{content:"ï„”"}.fa-folder-open-o:before{content:"ï„•"}.fa-smile-o:before{content:""}.fa-frown-o:before{content:"ï„™"}.fa-meh-o:before{content:"ï„š"}.fa-gamepad:before{content:"ï„›"}.fa-keyboard-o:before{content:"ï„œ"}.fa-flag-o:before{content:"ï„"}.fa-flag-checkered:before{content:"ï„ž"}.fa-terminal:before{content:"ï„ "}.fa-code:before{content:"ï„¡"}.fa-mail-reply-all:before,.fa-reply-all:before{content:"ï„¢"}.fa-star-half-empty:before,.fa-star-half-full:before,.fa-star-half-o:before{content:"ï„£"}.fa-location-arrow:before{content:""}.fa-crop:before{content:"ï„¥"}.fa-code-fork:before{content:""}.fa-unlink:before,.fa-chain-broken:before{content:""}.fa-question:before{content:""}.fa-info:before{content:"ï„©"}.fa-exclamation:before{content:""}.fa-superscript:before{content:"ï„«"}.fa-subscript:before{content:""}.fa-eraser:before{content:"ï„"}.fa-puzzle-piece:before{content:"ï„®"}.fa-microphone:before{content:"ï„°"}.fa-microphone-slash:before{content:""}.fa-shield:before{content:""}.fa-calendar-o:before{content:""}.fa-fire-extinguisher:before{content:"ï„´"}.fa-rocket:before{content:""}.fa-maxcdn:before{content:""}.fa-chevron-circle-left:before{content:"ï„·"}.fa-chevron-circle-right:before{content:""}.fa-chevron-circle-up:before{content:""}.fa-chevron-circle-down:before{content:""}.fa-html5:before{content:"ï„»"}.fa-css3:before{content:""}.fa-anchor:before{content:""}.fa-unlock-alt:before{content:""}.fa-bullseye:before{content:"ï…€"}.fa-ellipsis-h:before{content:"ï…"}.fa-ellipsis-v:before{content:"ï…‚"}.fa-rss-square:before{content:"ï…ƒ"}.fa-play-circle:before{content:"ï…„"}.fa-ticket:before{content:"ï……"}.fa-minus-square:before{content:"ï…†"}.fa-minus-square-o:before,.wy-menu-vertical li.on a span.toctree-expand:before,.wy-menu-vertical li.current>a span.toctree-expand:before{content:"ï…‡"}.fa-level-up:before{content:"ï…ˆ"}.fa-level-down:before{content:"ï…‰"}.fa-check-square:before{content:"ï…Š"}.fa-pencil-square:before{content:"ï…‹"}.fa-external-link-square:before{content:"ï…Œ"}.fa-share-square:before{content:"ï…"}.fa-compass:before{content:"ï…Ž"}.fa-toggle-down:before,.fa-caret-square-o-down:before{content:"ï…"}.fa-toggle-up:before,.fa-caret-square-o-up:before{content:"ï…‘"}.fa-toggle-right:before,.fa-caret-square-o-right:before{content:"ï…’"}.fa-euro:before,.fa-eur:before{content:"ï…“"}.fa-gbp:before{content:"ï…”"}.fa-dollar:before,.fa-usd:before{content:"ï…•"}.fa-rupee:before,.fa-inr:before{content:"ï…–"}.fa-cny:before,.fa-rmb:before,.fa-yen:before,.fa-jpy:before{content:"ï…—"}.fa-ruble:before,.fa-rouble:before,.fa-rub:before{content:"ï…˜"}.fa-won:before,.fa-krw:before{content:"ï…™"}.fa-bitcoin:before,.fa-btc:before{content:"ï…š"}.fa-file:before{content:"ï…›"}.fa-file-text:before{content:"ï…œ"}.fa-sort-alpha-asc:before{content:"ï…"}.fa-sort-alpha-desc:before{content:"ï…ž"}.fa-sort-amount-asc:before{content:"ï… "}.fa-sort-amount-desc:before{content:"ï…¡"}.fa-sort-numeric-asc:before{content:"ï…¢"}.fa-sort-numeric-desc:before{content:"ï…£"}.fa-thumbs-up:before{content:"ï…¤"}.fa-thumbs-down:before{content:"ï…¥"}.fa-youtube-square:before{content:"ï…¦"}.fa-youtube:before{content:"ï…§"}.fa-xing:before{content:"ï…¨"}.fa-xing-square:before{content:"ï…©"}.fa-youtube-play:before{content:"ï…ª"}.fa-dropbox:before{content:"ï…«"}.fa-stack-overflow:before{content:"ï…¬"}.fa-instagram:before{content:"ï…"}.fa-flickr:before{content:"ï…®"}.fa-adn:before{content:"ï…°"}.fa-bitbucket:before,.icon-bitbucket:before{content:"ï…±"}.fa-bitbucket-square:before{content:"ï…²"}.fa-tumblr:before{content:"ï…³"}.fa-tumblr-square:before{content:"ï…´"}.fa-long-arrow-down:before{content:"ï…µ"}.fa-long-arrow-up:before{content:"ï…¶"}.fa-long-arrow-left:before{content:"ï…·"}.fa-long-arrow-right:before{content:"ï…¸"}.fa-apple:before{content:"ï…¹"}.fa-windows:before{content:"ï…º"}.fa-android:before{content:"ï…»"}.fa-linux:before{content:"ï…¼"}.fa-dribbble:before{content:"ï…½"}.fa-skype:before{content:"ï…¾"}.fa-foursquare:before{content:""}.fa-trello:before{content:"ï†"}.fa-female:before{content:""}.fa-male:before{content:""}.fa-gittip:before{content:""}.fa-sun-o:before{content:""}.fa-moon-o:before{content:""}.fa-archive:before{content:""}.fa-bug:before{content:""}.fa-vk:before{content:""}.fa-weibo:before{content:""}.fa-renren:before{content:""}.fa-pagelines:before{content:""}.fa-stack-exchange:before{content:"ï†"}.fa-arrow-circle-o-right:before{content:""}.fa-arrow-circle-o-left:before{content:"ï†"}.fa-toggle-left:before,.fa-caret-square-o-left:before{content:""}.fa-dot-circle-o:before{content:""}.fa-wheelchair:before{content:""}.fa-vimeo-square:before{content:""}.fa-turkish-lira:before,.fa-try:before{content:""}.fa-plus-square-o:before,.wy-menu-vertical li span.toctree-expand:before{content:""}.fa-space-shuttle:before{content:""}.fa-slack:before{content:""}.fa-envelope-square:before{content:""}.fa-wordpress:before{content:""}.fa-openid:before{content:""}.fa-institution:before,.fa-bank:before,.fa-university:before{content:""}.fa-mortar-board:before,.fa-graduation-cap:before{content:"ï†"}.fa-yahoo:before{content:""}.fa-google:before{content:"ï† "}.fa-reddit:before{content:""}.fa-reddit-square:before{content:""}.fa-stumbleupon-circle:before{content:""}.fa-stumbleupon:before{content:""}.fa-delicious:before{content:""}.fa-digg:before{content:""}.fa-pied-piper:before{content:""}.fa-pied-piper-alt:before{content:""}.fa-drupal:before{content:""}.fa-joomla:before{content:""}.fa-language:before{content:""}.fa-fax:before{content:""}.fa-building:before{content:"ï†"}.fa-child:before{content:""}.fa-paw:before{content:""}.fa-spoon:before{content:""}.fa-cube:before{content:""}.fa-cubes:before{content:""}.fa-behance:before{content:""}.fa-behance-square:before{content:""}.fa-steam:before{content:""}.fa-steam-square:before{content:""}.fa-recycle:before{content:""}.fa-automobile:before,.fa-car:before{content:""}.fa-cab:before,.fa-taxi:before{content:""}.fa-tree:before{content:""}.fa-spotify:before{content:""}.fa-deviantart:before{content:""}.fa-soundcloud:before{content:""}.fa-database:before{content:""}.fa-file-pdf-o:before{content:"ï‡"}.fa-file-word-o:before{content:""}.fa-file-excel-o:before{content:""}.fa-file-powerpoint-o:before{content:""}.fa-file-photo-o:before,.fa-file-picture-o:before,.fa-file-image-o:before{content:""}.fa-file-zip-o:before,.fa-file-archive-o:before{content:""}.fa-file-sound-o:before,.fa-file-audio-o:before{content:""}.fa-file-movie-o:before,.fa-file-video-o:before{content:""}.fa-file-code-o:before{content:""}.fa-vine:before{content:""}.fa-codepen:before{content:""}.fa-jsfiddle:before{content:""}.fa-life-bouy:before,.fa-life-buoy:before,.fa-life-saver:before,.fa-support:before,.fa-life-ring:before{content:"ï‡"}.fa-circle-o-notch:before{content:""}.fa-ra:before,.fa-rebel:before{content:"ï‡"}.fa-ge:before,.fa-empire:before{content:""}.fa-git-square:before{content:""}.fa-git:before{content:""}.fa-hacker-news:before{content:""}.fa-tencent-weibo:before{content:""}.fa-qq:before{content:""}.fa-wechat:before,.fa-weixin:before{content:""}.fa-send:before,.fa-paper-plane:before{content:""}.fa-send-o:before,.fa-paper-plane-o:before{content:""}.fa-history:before{content:""}.fa-circle-thin:before{content:""}.fa-header:before{content:""}.fa-paragraph:before{content:"ï‡"}.fa-sliders:before{content:""}.fa-share-alt:before{content:"ï‡ "}.fa-share-alt-square:before{content:""}.fa-bomb:before{content:""}.fa-soccer-ball-o:before,.fa-futbol-o:before{content:""}.fa-tty:before{content:""}.fa-binoculars:before{content:""}.fa-plug:before{content:""}.fa-slideshare:before{content:""}.fa-twitch:before{content:""}.fa-yelp:before{content:""}.fa-newspaper-o:before{content:""}.fa-wifi:before{content:""}.fa-calculator:before{content:""}.fa-paypal:before{content:"ï‡"}.fa-google-wallet:before{content:""}.fa-cc-visa:before{content:""}.fa-cc-mastercard:before{content:""}.fa-cc-discover:before{content:""}.fa-cc-amex:before{content:""}.fa-cc-paypal:before{content:""}.fa-cc-stripe:before{content:""}.fa-bell-slash:before{content:""}.fa-bell-slash-o:before{content:""}.fa-trash:before{content:""}.fa-copyright:before{content:""}.fa-at:before{content:""}.fa-eyedropper:before{content:""}.fa-paint-brush:before{content:""}.fa-birthday-cake:before{content:""}.fa-area-chart:before{content:""}.fa-pie-chart:before{content:""}.fa-line-chart:before{content:"ïˆ"}.fa-lastfm:before{content:""}.fa-lastfm-square:before{content:""}.fa-toggle-off:before{content:""}.fa-toggle-on:before{content:""}.fa-bicycle:before{content:""}.fa-bus:before{content:""}.fa-ioxhost:before{content:""}.fa-angellist:before{content:""}.fa-cc:before{content:""}.fa-shekel:before,.fa-sheqel:before,.fa-ils:before{content:""}.fa-meanpath:before{content:""}.fa,.wy-menu-vertical li span.toctree-expand,.wy-menu-vertical li.on a span.toctree-expand,.wy-menu-vertical li.current>a span.toctree-expand,.rst-content .admonition-title,.rst-content h1 .headerlink,.rst-content h2 .headerlink,.rst-content h3 .headerlink,.rst-content h4 .headerlink,.rst-content h5 .headerlink,.rst-content h6 .headerlink,.rst-content dl dt .headerlink,.rst-content p.caption .headerlink,.rst-content tt.download span:first-child,.rst-content code.download span:first-child,.icon,.wy-dropdown .caret,.wy-inline-validate.wy-inline-validate-success .wy-input-context,.wy-inline-validate.wy-inline-validate-danger .wy-input-context,.wy-inline-validate.wy-inline-validate-warning .wy-input-context,.wy-inline-validate.wy-inline-validate-info .wy-input-context{font-family:inherit}.fa:before,.wy-menu-vertical li span.toctree-expand:before,.wy-menu-vertical li.on a span.toctree-expand:before,.wy-menu-vertical li.current>a span.toctree-expand:before,.rst-content .admonition-title:before,.rst-content h1 .headerlink:before,.rst-content h2 .headerlink:before,.rst-content h3 .headerlink:before,.rst-content h4 .headerlink:before,.rst-content h5 .headerlink:before,.rst-content h6 .headerlink:before,.rst-content dl dt .headerlink:before,.rst-content p.caption .headerlink:before,.rst-content tt.download span:first-child:before,.rst-content code.download span:first-child:before,.icon:before,.wy-dropdown .caret:before,.wy-inline-validate.wy-inline-validate-success .wy-input-context:before,.wy-inline-validate.wy-inline-validate-danger .wy-input-context:before,.wy-inline-validate.wy-inline-validate-warning .wy-input-context:before,.wy-inline-validate.wy-inline-validate-info .wy-input-context:before{font-family:"FontAwesome";display:inline-block;font-style:normal;font-weight:normal;line-height:1;text-decoration:inherit}a .fa,a .wy-menu-vertical li span.toctree-expand,.wy-menu-vertical li a span.toctree-expand,.wy-menu-vertical li.on a span.toctree-expand,.wy-menu-vertical li.current>a span.toctree-expand,a .rst-content .admonition-title,.rst-content a .admonition-title,a .rst-content h1 .headerlink,.rst-content h1 a .headerlink,a .rst-content h2 .headerlink,.rst-content h2 a .headerlink,a .rst-content h3 .headerlink,.rst-content h3 a .headerlink,a .rst-content h4 .headerlink,.rst-content h4 a .headerlink,a .rst-content h5 .headerlink,.rst-content h5 a .headerlink,a .rst-content h6 .headerlink,.rst-content h6 a .headerlink,a .rst-content dl dt .headerlink,.rst-content dl dt a .headerlink,a .rst-content p.caption .headerlink,.rst-content p.caption a .headerlink,a .rst-content tt.download span:first-child,.rst-content tt.download a span:first-child,a .rst-content code.download span:first-child,.rst-content code.download a span:first-child,a .icon{display:inline-block;text-decoration:inherit}.btn .fa,.btn .wy-menu-vertical li span.toctree-expand,.wy-menu-vertical li .btn span.toctree-expand,.btn .wy-menu-vertical li.on a span.toctree-expand,.wy-menu-vertical li.on a .btn span.toctree-expand,.btn .wy-menu-vertical li.current>a span.toctree-expand,.wy-menu-vertical li.current>a .btn span.toctree-expand,.btn .rst-content .admonition-title,.rst-content .btn .admonition-title,.btn .rst-content h1 .headerlink,.rst-content h1 .btn .headerlink,.btn .rst-content h2 .headerlink,.rst-content h2 .btn .headerlink,.btn .rst-content h3 .headerlink,.rst-content h3 .btn .headerlink,.btn .rst-content h4 .headerlink,.rst-content h4 .btn .headerlink,.btn .rst-content h5 .headerlink,.rst-content h5 .btn .headerlink,.btn .rst-content h6 .headerlink,.rst-content h6 .btn .headerlink,.btn .rst-content dl dt .headerlink,.rst-content dl dt .btn .headerlink,.btn .rst-content p.caption .headerlink,.rst-content p.caption .btn .headerlink,.btn .rst-content tt.download span:first-child,.rst-content tt.download .btn span:first-child,.btn .rst-content code.download span:first-child,.rst-content code.download .btn span:first-child,.btn .icon,.nav .fa,.nav .wy-menu-vertical li span.toctree-expand,.wy-menu-vertical li .nav span.toctree-expand,.nav .wy-menu-vertical li.on a span.toctree-expand,.wy-menu-vertical li.on a .nav span.toctree-expand,.nav .wy-menu-vertical li.current>a span.toctree-expand,.wy-menu-vertical li.current>a .nav span.toctree-expand,.nav .rst-content .admonition-title,.rst-content .nav .admonition-title,.nav .rst-content h1 .headerlink,.rst-content h1 .nav .headerlink,.nav .rst-content h2 .headerlink,.rst-content h2 .nav .headerlink,.nav .rst-content h3 .headerlink,.rst-content h3 .nav .headerlink,.nav .rst-content h4 .headerlink,.rst-content h4 .nav .headerlink,.nav .rst-content h5 .headerlink,.rst-content h5 .nav .headerlink,.nav .rst-content h6 .headerlink,.rst-content h6 .nav .headerlink,.nav .rst-content dl dt .headerlink,.rst-content dl dt .nav .headerlink,.nav .rst-content p.caption .headerlink,.rst-content p.caption .nav .headerlink,.nav .rst-content tt.download span:first-child,.rst-content tt.download .nav span:first-child,.nav .rst-content code.download span:first-child,.rst-content code.download .nav span:first-child,.nav .icon{display:inline}.btn .fa.fa-large,.btn .wy-menu-vertical li span.fa-large.toctree-expand,.wy-menu-vertical li .btn span.fa-large.toctree-expand,.btn .rst-content .fa-large.admonition-title,.rst-content .btn .fa-large.admonition-title,.btn .rst-content h1 .fa-large.headerlink,.rst-content h1 .btn .fa-large.headerlink,.btn .rst-content h2 .fa-large.headerlink,.rst-content h2 .btn .fa-large.headerlink,.btn .rst-content h3 .fa-large.headerlink,.rst-content h3 .btn .fa-large.headerlink,.btn .rst-content h4 .fa-large.headerlink,.rst-content h4 .btn .fa-large.headerlink,.btn .rst-content h5 .fa-large.headerlink,.rst-content h5 .btn .fa-large.headerlink,.btn .rst-content h6 .fa-large.headerlink,.rst-content h6 .btn .fa-large.headerlink,.btn .rst-content dl dt .fa-large.headerlink,.rst-content dl dt .btn .fa-large.headerlink,.btn .rst-content p.caption .fa-large.headerlink,.rst-content p.caption .btn .fa-large.headerlink,.btn .rst-content tt.download span.fa-large:first-child,.rst-content tt.download .btn span.fa-large:first-child,.btn .rst-content code.download span.fa-large:first-child,.rst-content code.download .btn span.fa-large:first-child,.btn .fa-large.icon,.nav .fa.fa-large,.nav .wy-menu-vertical li span.fa-large.toctree-expand,.wy-menu-vertical li .nav span.fa-large.toctree-expand,.nav .rst-content .fa-large.admonition-title,.rst-content .nav .fa-large.admonition-title,.nav .rst-content h1 .fa-large.headerlink,.rst-content h1 .nav .fa-large.headerlink,.nav .rst-content h2 .fa-large.headerlink,.rst-content h2 .nav .fa-large.headerlink,.nav .rst-content h3 .fa-large.headerlink,.rst-content h3 .nav .fa-large.headerlink,.nav .rst-content h4 .fa-large.headerlink,.rst-content h4 .nav .fa-large.headerlink,.nav .rst-content h5 .fa-large.headerlink,.rst-content h5 .nav .fa-large.headerlink,.nav .rst-content h6 .fa-large.headerlink,.rst-content h6 .nav .fa-large.headerlink,.nav .rst-content dl dt .fa-large.headerlink,.rst-content dl dt .nav .fa-large.headerlink,.nav .rst-content p.caption .fa-large.headerlink,.rst-content p.caption .nav .fa-large.headerlink,.nav .rst-content tt.download span.fa-large:first-child,.rst-content tt.download .nav span.fa-large:first-child,.nav .rst-content code.download span.fa-large:first-child,.rst-content code.download .nav span.fa-large:first-child,.nav .fa-large.icon{line-height:0.9em}.btn .fa.fa-spin,.btn .wy-menu-vertical li span.fa-spin.toctree-expand,.wy-menu-vertical li .btn span.fa-spin.toctree-expand,.btn .rst-content .fa-spin.admonition-title,.rst-content .btn .fa-spin.admonition-title,.btn .rst-content h1 .fa-spin.headerlink,.rst-content h1 .btn .fa-spin.headerlink,.btn .rst-content h2 .fa-spin.headerlink,.rst-content h2 .btn .fa-spin.headerlink,.btn .rst-content h3 .fa-spin.headerlink,.rst-content h3 .btn .fa-spin.headerlink,.btn .rst-content h4 .fa-spin.headerlink,.rst-content h4 .btn .fa-spin.headerlink,.btn .rst-content h5 .fa-spin.headerlink,.rst-content h5 .btn .fa-spin.headerlink,.btn .rst-content h6 .fa-spin.headerlink,.rst-content h6 .btn .fa-spin.headerlink,.btn .rst-content dl dt .fa-spin.headerlink,.rst-content dl dt .btn .fa-spin.headerlink,.btn .rst-content p.caption .fa-spin.headerlink,.rst-content p.caption .btn .fa-spin.headerlink,.btn .rst-content tt.download span.fa-spin:first-child,.rst-content tt.download .btn span.fa-spin:first-child,.btn .rst-content code.download span.fa-spin:first-child,.rst-content code.download .btn span.fa-spin:first-child,.btn .fa-spin.icon,.nav .fa.fa-spin,.nav .wy-menu-vertical li span.fa-spin.toctree-expand,.wy-menu-vertical li .nav span.fa-spin.toctree-expand,.nav .rst-content .fa-spin.admonition-title,.rst-content .nav .fa-spin.admonition-title,.nav .rst-content h1 .fa-spin.headerlink,.rst-content h1 .nav .fa-spin.headerlink,.nav .rst-content h2 .fa-spin.headerlink,.rst-content h2 .nav .fa-spin.headerlink,.nav .rst-content h3 .fa-spin.headerlink,.rst-content h3 .nav .fa-spin.headerlink,.nav .rst-content h4 .fa-spin.headerlink,.rst-content h4 .nav .fa-spin.headerlink,.nav .rst-content h5 .fa-spin.headerlink,.rst-content h5 .nav .fa-spin.headerlink,.nav .rst-content h6 .fa-spin.headerlink,.rst-content h6 .nav .fa-spin.headerlink,.nav .rst-content dl dt .fa-spin.headerlink,.rst-content dl dt .nav .fa-spin.headerlink,.nav .rst-content p.caption .fa-spin.headerlink,.rst-content p.caption .nav .fa-spin.headerlink,.nav .rst-content tt.download span.fa-spin:first-child,.rst-content tt.download .nav span.fa-spin:first-child,.nav .rst-content code.download span.fa-spin:first-child,.rst-content code.download .nav span.fa-spin:first-child,.nav .fa-spin.icon{display:inline-block}.btn.fa:before,.wy-menu-vertical li span.btn.toctree-expand:before,.rst-content .btn.admonition-title:before,.rst-content h1 .btn.headerlink:before,.rst-content h2 .btn.headerlink:before,.rst-content h3 .btn.headerlink:before,.rst-content h4 .btn.headerlink:before,.rst-content h5 .btn.headerlink:before,.rst-content h6 .btn.headerlink:before,.rst-content dl dt .btn.headerlink:before,.rst-content p.caption .btn.headerlink:before,.rst-content tt.download span.btn:first-child:before,.rst-content code.download span.btn:first-child:before,.btn.icon:before{opacity:0.5;-webkit-transition:opacity 0.05s ease-in;-moz-transition:opacity 0.05s ease-in;transition:opacity 0.05s ease-in}.btn.fa:hover:before,.wy-menu-vertical li span.btn.toctree-expand:hover:before,.rst-content .btn.admonition-title:hover:before,.rst-content h1 .btn.headerlink:hover:before,.rst-content h2 .btn.headerlink:hover:before,.rst-content h3 .btn.headerlink:hover:before,.rst-content h4 .btn.headerlink:hover:before,.rst-content h5 .btn.headerlink:hover:before,.rst-content h6 .btn.headerlink:hover:before,.rst-content dl dt .btn.headerlink:hover:before,.rst-content p.caption .btn.headerlink:hover:before,.rst-content tt.download span.btn:first-child:hover:before,.rst-content code.download span.btn:first-child:hover:before,.btn.icon:hover:before{opacity:1}.btn-mini .fa:before,.btn-mini .wy-menu-vertical li span.toctree-expand:before,.wy-menu-vertical li .btn-mini span.toctree-expand:before,.btn-mini .rst-content .admonition-title:before,.rst-content .btn-mini .admonition-title:before,.btn-mini .rst-content h1 .headerlink:before,.rst-content h1 .btn-mini .headerlink:before,.btn-mini .rst-content h2 .headerlink:before,.rst-content h2 .btn-mini .headerlink:before,.btn-mini .rst-content h3 .headerlink:before,.rst-content h3 .btn-mini .headerlink:before,.btn-mini .rst-content h4 .headerlink:before,.rst-content h4 .btn-mini .headerlink:before,.btn-mini .rst-content h5 .headerlink:before,.rst-content h5 .btn-mini .headerlink:before,.btn-mini .rst-content h6 .headerlink:before,.rst-content h6 .btn-mini .headerlink:before,.btn-mini .rst-content dl dt .headerlink:before,.rst-content dl dt .btn-mini .headerlink:before,.btn-mini .rst-content p.caption .headerlink:before,.rst-content p.caption .btn-mini .headerlink:before,.btn-mini .rst-content tt.download span:first-child:before,.rst-content tt.download .btn-mini span:first-child:before,.btn-mini .rst-content code.download span:first-child:before,.rst-content code.download .btn-mini span:first-child:before,.btn-mini .icon:before{font-size:14px;vertical-align:-15%}.wy-alert,.rst-content .note,.rst-content .attention,.rst-content .caution,.rst-content .danger,.rst-content .error,.rst-content .hint,.rst-content .important,.rst-content .tip,.rst-content .warning,.rst-content .seealso,.rst-content .admonition-todo{padding:12px;line-height:24px;margin-bottom:24px;background:#e7f2fa}.wy-alert-title,.rst-content .admonition-title{color:#fff;font-weight:bold;display:block;color:#fff;background:#6ab0de;margin:-12px;padding:6px 12px;margin-bottom:12px}.wy-alert.wy-alert-danger,.rst-content .wy-alert-danger.note,.rst-content .wy-alert-danger.attention,.rst-content .wy-alert-danger.caution,.rst-content .danger,.rst-content .error,.rst-content .wy-alert-danger.hint,.rst-content .wy-alert-danger.important,.rst-content .wy-alert-danger.tip,.rst-content .wy-alert-danger.warning,.rst-content .wy-alert-danger.seealso,.rst-content .wy-alert-danger.admonition-todo{background:#fdf3f2}.wy-alert.wy-alert-danger .wy-alert-title,.rst-content .wy-alert-danger.note .wy-alert-title,.rst-content .wy-alert-danger.attention .wy-alert-title,.rst-content .wy-alert-danger.caution .wy-alert-title,.rst-content .danger .wy-alert-title,.rst-content .error .wy-alert-title,.rst-content .wy-alert-danger.hint .wy-alert-title,.rst-content .wy-alert-danger.important .wy-alert-title,.rst-content .wy-alert-danger.tip .wy-alert-title,.rst-content .wy-alert-danger.warning .wy-alert-title,.rst-content .wy-alert-danger.seealso .wy-alert-title,.rst-content .wy-alert-danger.admonition-todo .wy-alert-title,.wy-alert.wy-alert-danger .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-danger .admonition-title,.rst-content .wy-alert-danger.note .admonition-title,.rst-content .wy-alert-danger.attention .admonition-title,.rst-content .wy-alert-danger.caution .admonition-title,.rst-content .danger .admonition-title,.rst-content .error .admonition-title,.rst-content .wy-alert-danger.hint .admonition-title,.rst-content .wy-alert-danger.important .admonition-title,.rst-content .wy-alert-danger.tip .admonition-title,.rst-content .wy-alert-danger.warning .admonition-title,.rst-content .wy-alert-danger.seealso .admonition-title,.rst-content .wy-alert-danger.admonition-todo .admonition-title{background:#f29f97}.wy-alert.wy-alert-warning,.rst-content .wy-alert-warning.note,.rst-content .attention,.rst-content .caution,.rst-content .wy-alert-warning.danger,.rst-content .wy-alert-warning.error,.rst-content .wy-alert-warning.hint,.rst-content .wy-alert-warning.important,.rst-content .wy-alert-warning.tip,.rst-content .warning,.rst-content .wy-alert-warning.seealso,.rst-content .admonition-todo{background:#ffedcc}.wy-alert.wy-alert-warning .wy-alert-title,.rst-content .wy-alert-warning.note .wy-alert-title,.rst-content .attention .wy-alert-title,.rst-content .caution .wy-alert-title,.rst-content .wy-alert-warning.danger .wy-alert-title,.rst-content .wy-alert-warning.error .wy-alert-title,.rst-content .wy-alert-warning.hint .wy-alert-title,.rst-content .wy-alert-warning.important .wy-alert-title,.rst-content .wy-alert-warning.tip .wy-alert-title,.rst-content .warning .wy-alert-title,.rst-content .wy-alert-warning.seealso .wy-alert-title,.rst-content .admonition-todo .wy-alert-title,.wy-alert.wy-alert-warning .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-warning .admonition-title,.rst-content .wy-alert-warning.note .admonition-title,.rst-content .attention .admonition-title,.rst-content .caution .admonition-title,.rst-content .wy-alert-warning.danger .admonition-title,.rst-content .wy-alert-warning.error .admonition-title,.rst-content .wy-alert-warning.hint .admonition-title,.rst-content .wy-alert-warning.important .admonition-title,.rst-content .wy-alert-warning.tip .admonition-title,.rst-content .warning .admonition-title,.rst-content .wy-alert-warning.seealso .admonition-title,.rst-content .admonition-todo .admonition-title{background:#f0b37e}.wy-alert.wy-alert-info,.rst-content .note,.rst-content .wy-alert-info.attention,.rst-content .wy-alert-info.caution,.rst-content .wy-alert-info.danger,.rst-content .wy-alert-info.error,.rst-content .wy-alert-info.hint,.rst-content .wy-alert-info.important,.rst-content .wy-alert-info.tip,.rst-content .wy-alert-info.warning,.rst-content .seealso,.rst-content .wy-alert-info.admonition-todo{background:#e7f2fa}.wy-alert.wy-alert-info .wy-alert-title,.rst-content .note .wy-alert-title,.rst-content .wy-alert-info.attention .wy-alert-title,.rst-content .wy-alert-info.caution .wy-alert-title,.rst-content .wy-alert-info.danger .wy-alert-title,.rst-content .wy-alert-info.error .wy-alert-title,.rst-content .wy-alert-info.hint .wy-alert-title,.rst-content .wy-alert-info.important .wy-alert-title,.rst-content .wy-alert-info.tip .wy-alert-title,.rst-content .wy-alert-info.warning .wy-alert-title,.rst-content .seealso .wy-alert-title,.rst-content .wy-alert-info.admonition-todo .wy-alert-title,.wy-alert.wy-alert-info .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-info .admonition-title,.rst-content .note .admonition-title,.rst-content .wy-alert-info.attention .admonition-title,.rst-content .wy-alert-info.caution .admonition-title,.rst-content .wy-alert-info.danger .admonition-title,.rst-content .wy-alert-info.error .admonition-title,.rst-content .wy-alert-info.hint .admonition-title,.rst-content .wy-alert-info.important .admonition-title,.rst-content .wy-alert-info.tip .admonition-title,.rst-content .wy-alert-info.warning .admonition-title,.rst-content .seealso .admonition-title,.rst-content .wy-alert-info.admonition-todo .admonition-title{background:#6ab0de}.wy-alert.wy-alert-success,.rst-content .wy-alert-success.note,.rst-content .wy-alert-success.attention,.rst-content .wy-alert-success.caution,.rst-content .wy-alert-success.danger,.rst-content .wy-alert-success.error,.rst-content .hint,.rst-content .important,.rst-content .tip,.rst-content .wy-alert-success.warning,.rst-content .wy-alert-success.seealso,.rst-content .wy-alert-success.admonition-todo{background:#dbfaf4}.wy-alert.wy-alert-success .wy-alert-title,.rst-content .wy-alert-success.note .wy-alert-title,.rst-content .wy-alert-success.attention .wy-alert-title,.rst-content .wy-alert-success.caution .wy-alert-title,.rst-content .wy-alert-success.danger .wy-alert-title,.rst-content .wy-alert-success.error .wy-alert-title,.rst-content .hint .wy-alert-title,.rst-content .important .wy-alert-title,.rst-content .tip .wy-alert-title,.rst-content .wy-alert-success.warning .wy-alert-title,.rst-content .wy-alert-success.seealso .wy-alert-title,.rst-content .wy-alert-success.admonition-todo .wy-alert-title,.wy-alert.wy-alert-success .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-success .admonition-title,.rst-content .wy-alert-success.note .admonition-title,.rst-content .wy-alert-success.attention .admonition-title,.rst-content .wy-alert-success.caution .admonition-title,.rst-content .wy-alert-success.danger .admonition-title,.rst-content .wy-alert-success.error .admonition-title,.rst-content .hint .admonition-title,.rst-content .important .admonition-title,.rst-content .tip .admonition-title,.rst-content .wy-alert-success.warning .admonition-title,.rst-content .wy-alert-success.seealso .admonition-title,.rst-content .wy-alert-success.admonition-todo .admonition-title{background:#1abc9c}.wy-alert.wy-alert-neutral,.rst-content .wy-alert-neutral.note,.rst-content .wy-alert-neutral.attention,.rst-content .wy-alert-neutral.caution,.rst-content .wy-alert-neutral.danger,.rst-content .wy-alert-neutral.error,.rst-content .wy-alert-neutral.hint,.rst-content .wy-alert-neutral.important,.rst-content .wy-alert-neutral.tip,.rst-content .wy-alert-neutral.warning,.rst-content .wy-alert-neutral.seealso,.rst-content .wy-alert-neutral.admonition-todo{background:#f3f6f6}.wy-alert.wy-alert-neutral .wy-alert-title,.rst-content .wy-alert-neutral.note .wy-alert-title,.rst-content .wy-alert-neutral.attention .wy-alert-title,.rst-content .wy-alert-neutral.caution .wy-alert-title,.rst-content .wy-alert-neutral.danger .wy-alert-title,.rst-content .wy-alert-neutral.error .wy-alert-title,.rst-content .wy-alert-neutral.hint .wy-alert-title,.rst-content .wy-alert-neutral.important .wy-alert-title,.rst-content .wy-alert-neutral.tip .wy-alert-title,.rst-content .wy-alert-neutral.warning .wy-alert-title,.rst-content .wy-alert-neutral.seealso .wy-alert-title,.rst-content .wy-alert-neutral.admonition-todo .wy-alert-title,.wy-alert.wy-alert-neutral .rst-content .admonition-title,.rst-content .wy-alert.wy-alert-neutral .admonition-title,.rst-content .wy-alert-neutral.note .admonition-title,.rst-content .wy-alert-neutral.attention .admonition-title,.rst-content .wy-alert-neutral.caution .admonition-title,.rst-content .wy-alert-neutral.danger .admonition-title,.rst-content .wy-alert-neutral.error .admonition-title,.rst-content .wy-alert-neutral.hint .admonition-title,.rst-content .wy-alert-neutral.important .admonition-title,.rst-content .wy-alert-neutral.tip .admonition-title,.rst-content .wy-alert-neutral.warning .admonition-title,.rst-content .wy-alert-neutral.seealso .admonition-title,.rst-content .wy-alert-neutral.admonition-todo .admonition-title{color:#404040;background:#e1e4e5}.wy-alert.wy-alert-neutral a,.rst-content .wy-alert-neutral.note a,.rst-content .wy-alert-neutral.attention a,.rst-content .wy-alert-neutral.caution a,.rst-content .wy-alert-neutral.danger a,.rst-content .wy-alert-neutral.error a,.rst-content .wy-alert-neutral.hint a,.rst-content .wy-alert-neutral.important a,.rst-content .wy-alert-neutral.tip a,.rst-content .wy-alert-neutral.warning a,.rst-content .wy-alert-neutral.seealso a,.rst-content .wy-alert-neutral.admonition-todo a{color:#2980B9}.wy-alert p:last-child,.rst-content .note p:last-child,.rst-content .attention p:last-child,.rst-content .caution p:last-child,.rst-content .danger p:last-child,.rst-content .error p:last-child,.rst-content .hint p:last-child,.rst-content .important p:last-child,.rst-content .tip p:last-child,.rst-content .warning p:last-child,.rst-content .seealso p:last-child,.rst-content .admonition-todo p:last-child{margin-bottom:0}.wy-tray-container{position:fixed;bottom:0px;left:0;z-index:600}.wy-tray-container li{display:block;width:300px;background:transparent;color:#fff;text-align:center;box-shadow:0 5px 5px 0 rgba(0,0,0,0.1);padding:0 24px;min-width:20%;opacity:0;height:0;line-height:56px;overflow:hidden;-webkit-transition:all 0.3s ease-in;-moz-transition:all 0.3s ease-in;transition:all 0.3s ease-in}.wy-tray-container li.wy-tray-item-success{background:#27AE60}.wy-tray-container li.wy-tray-item-info{background:#2980B9}.wy-tray-container li.wy-tray-item-warning{background:#E67E22}.wy-tray-container li.wy-tray-item-danger{background:#E74C3C}.wy-tray-container li.on{opacity:1;height:56px}@media screen and (max-width: 768px){.wy-tray-container{bottom:auto;top:0;width:100%}.wy-tray-container li{width:100%}}button{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle;cursor:pointer;line-height:normal;-webkit-appearance:button;*overflow:visible}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}button[disabled]{cursor:default}.btn{display:inline-block;border-radius:2px;line-height:normal;white-space:nowrap;text-align:center;cursor:pointer;font-size:100%;padding:6px 12px 8px 12px;color:#fff;border:1px solid rgba(0,0,0,0.1);background-color:#27AE60;text-decoration:none;font-weight:normal;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;box-shadow:0px 1px 2px -1px rgba(255,255,255,0.5) inset,0px -2px 0px 0px rgba(0,0,0,0.1) inset;outline-none:false;vertical-align:middle;*display:inline;zoom:1;-webkit-user-drag:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;-webkit-transition:all 0.1s linear;-moz-transition:all 0.1s linear;transition:all 0.1s linear}.btn-hover{background:#2e8ece;color:#fff}.btn:hover{background:#2cc36b;color:#fff}.btn:focus{background:#2cc36b;outline:0}.btn:active{box-shadow:0px -1px 0px 0px rgba(0,0,0,0.05) inset,0px 2px 0px 0px rgba(0,0,0,0.1) inset;padding:8px 12px 6px 12px}.btn:visited{color:#fff}.btn:disabled{background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=40);opacity:0.4;cursor:not-allowed;box-shadow:none}.btn-disabled{background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=40);opacity:0.4;cursor:not-allowed;box-shadow:none}.btn-disabled:hover,.btn-disabled:focus,.btn-disabled:active{background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled = false);filter:alpha(opacity=40);opacity:0.4;cursor:not-allowed;box-shadow:none}.btn::-moz-focus-inner{padding:0;border:0}.btn-small{font-size:80%}.btn-info{background-color:#2980B9 !important}.btn-info:hover{background-color:#2e8ece !important}.btn-neutral{background-color:#f3f6f6 !important;color:#404040 !important}.btn-neutral:hover{background-color:#e5ebeb !important;color:#404040}.btn-neutral:visited{color:#404040 !important}.btn-success{background-color:#27AE60 !important}.btn-success:hover{background-color:#295 !important}.btn-danger{background-color:#E74C3C !important}.btn-danger:hover{background-color:#ea6153 !important}.btn-warning{background-color:#E67E22 !important}.btn-warning:hover{background-color:#e98b39 !important}.btn-invert{background-color:#222}.btn-invert:hover{background-color:#2f2f2f !important}.btn-link{background-color:transparent !important;color:#2980B9;box-shadow:none;border-color:transparent !important}.btn-link:hover{background-color:transparent !important;color:#409ad5 !important;box-shadow:none}.btn-link:active{background-color:transparent !important;color:#409ad5 !important;box-shadow:none}.btn-link:visited{color:#9B59B6}.wy-btn-group .btn,.wy-control .btn{vertical-align:middle}.wy-btn-group{margin-bottom:24px;*zoom:1}.wy-btn-group:before,.wy-btn-group:after{display:table;content:""}.wy-btn-group:after{clear:both}.wy-dropdown{position:relative;display:inline-block}.wy-dropdown-active .wy-dropdown-menu{display:block}.wy-dropdown-menu{position:absolute;left:0;display:none;float:left;top:100%;min-width:100%;background:#fcfcfc;z-index:100;border:solid 1px #cfd7dd;box-shadow:0 2px 2px 0 rgba(0,0,0,0.1);padding:12px}.wy-dropdown-menu>dd>a{display:block;clear:both;color:#404040;white-space:nowrap;font-size:90%;padding:0 12px;cursor:pointer}.wy-dropdown-menu>dd>a:hover{background:#2980B9;color:#fff}.wy-dropdown-menu>dd.divider{border-top:solid 1px #cfd7dd;margin:6px 0}.wy-dropdown-menu>dd.search{padding-bottom:12px}.wy-dropdown-menu>dd.search input[type="search"]{width:100%}.wy-dropdown-menu>dd.call-to-action{background:#e3e3e3;text-transform:uppercase;font-weight:500;font-size:80%}.wy-dropdown-menu>dd.call-to-action:hover{background:#e3e3e3}.wy-dropdown-menu>dd.call-to-action .btn{color:#fff}.wy-dropdown.wy-dropdown-up .wy-dropdown-menu{bottom:100%;top:auto;left:auto;right:0}.wy-dropdown.wy-dropdown-bubble .wy-dropdown-menu{background:#fcfcfc;margin-top:2px}.wy-dropdown.wy-dropdown-bubble .wy-dropdown-menu a{padding:6px 12px}.wy-dropdown.wy-dropdown-bubble .wy-dropdown-menu a:hover{background:#2980B9;color:#fff}.wy-dropdown.wy-dropdown-left .wy-dropdown-menu{right:0;left:auto;text-align:right}.wy-dropdown-arrow:before{content:" ";border-bottom:5px solid #f5f5f5;border-left:5px solid transparent;border-right:5px solid transparent;position:absolute;display:block;top:-4px;left:50%;margin-left:-3px}.wy-dropdown-arrow.wy-dropdown-arrow-left:before{left:11px}.wy-form-stacked select{display:block}.wy-form-aligned input,.wy-form-aligned textarea,.wy-form-aligned select,.wy-form-aligned .wy-help-inline,.wy-form-aligned label{display:inline-block;*display:inline;*zoom:1;vertical-align:middle}.wy-form-aligned .wy-control-group>label{display:inline-block;vertical-align:middle;width:10em;margin:6px 12px 0 0;float:left}.wy-form-aligned .wy-control{float:left}.wy-form-aligned .wy-control label{display:block}.wy-form-aligned .wy-control select{margin-top:6px}fieldset{border:0;margin:0;padding:0}legend{display:block;width:100%;border:0;padding:0;white-space:normal;margin-bottom:24px;font-size:150%;*margin-left:-7px}label{display:block;margin:0 0 0.3125em 0;color:#333;font-size:90%}input,select,textarea{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle}.wy-control-group{margin-bottom:24px;*zoom:1;max-width:68em;margin-left:auto;margin-right:auto;*zoom:1}.wy-control-group:before,.wy-control-group:after{display:table;content:""}.wy-control-group:after{clear:both}.wy-control-group:before,.wy-control-group:after{display:table;content:""}.wy-control-group:after{clear:both}.wy-control-group.wy-control-group-required>label:after{content:" *";color:#E74C3C}.wy-control-group .wy-form-full,.wy-control-group .wy-form-halves,.wy-control-group .wy-form-thirds{padding-bottom:12px}.wy-control-group .wy-form-full select,.wy-control-group .wy-form-halves select,.wy-control-group .wy-form-thirds select{width:100%}.wy-control-group .wy-form-full input[type="text"],.wy-control-group .wy-form-full input[type="password"],.wy-control-group .wy-form-full input[type="email"],.wy-control-group .wy-form-full input[type="url"],.wy-control-group .wy-form-full input[type="date"],.wy-control-group .wy-form-full input[type="month"],.wy-control-group .wy-form-full input[type="time"],.wy-control-group .wy-form-full input[type="datetime"],.wy-control-group .wy-form-full input[type="datetime-local"],.wy-control-group .wy-form-full input[type="week"],.wy-control-group .wy-form-full input[type="number"],.wy-control-group .wy-form-full input[type="search"],.wy-control-group .wy-form-full input[type="tel"],.wy-control-group .wy-form-full input[type="color"],.wy-control-group .wy-form-halves input[type="text"],.wy-control-group .wy-form-halves input[type="password"],.wy-control-group .wy-form-halves input[type="email"],.wy-control-group .wy-form-halves input[type="url"],.wy-control-group .wy-form-halves input[type="date"],.wy-control-group .wy-form-halves input[type="month"],.wy-control-group .wy-form-halves input[type="time"],.wy-control-group .wy-form-halves input[type="datetime"],.wy-control-group .wy-form-halves input[type="datetime-local"],.wy-control-group .wy-form-halves input[type="week"],.wy-control-group .wy-form-halves input[type="number"],.wy-control-group .wy-form-halves input[type="search"],.wy-control-group .wy-form-halves input[type="tel"],.wy-control-group .wy-form-halves input[type="color"],.wy-control-group .wy-form-thirds input[type="text"],.wy-control-group .wy-form-thirds input[type="password"],.wy-control-group .wy-form-thirds input[type="email"],.wy-control-group .wy-form-thirds input[type="url"],.wy-control-group .wy-form-thirds input[type="date"],.wy-control-group .wy-form-thirds input[type="month"],.wy-control-group .wy-form-thirds input[type="time"],.wy-control-group .wy-form-thirds input[type="datetime"],.wy-control-group .wy-form-thirds input[type="datetime-local"],.wy-control-group .wy-form-thirds input[type="week"],.wy-control-group .wy-form-thirds input[type="number"],.wy-control-group .wy-form-thirds input[type="search"],.wy-control-group .wy-form-thirds input[type="tel"],.wy-control-group .wy-form-thirds input[type="color"]{width:100%}.wy-control-group .wy-form-full{float:left;display:block;margin-right:2.35765%;width:100%;margin-right:0}.wy-control-group .wy-form-full:last-child{margin-right:0}.wy-control-group .wy-form-halves{float:left;display:block;margin-right:2.35765%;width:48.82117%}.wy-control-group .wy-form-halves:last-child{margin-right:0}.wy-control-group .wy-form-halves:nth-of-type(2n){margin-right:0}.wy-control-group .wy-form-halves:nth-of-type(2n+1){clear:left}.wy-control-group .wy-form-thirds{float:left;display:block;margin-right:2.35765%;width:31.76157%}.wy-control-group .wy-form-thirds:last-child{margin-right:0}.wy-control-group .wy-form-thirds:nth-of-type(3n){margin-right:0}.wy-control-group .wy-form-thirds:nth-of-type(3n+1){clear:left}.wy-control-group.wy-control-group-no-input .wy-control{margin:6px 0 0 0;font-size:90%}.wy-control-no-input{display:inline-block;margin:6px 0 0 0;font-size:90%}.wy-control-group.fluid-input input[type="text"],.wy-control-group.fluid-input input[type="password"],.wy-control-group.fluid-input input[type="email"],.wy-control-group.fluid-input input[type="url"],.wy-control-group.fluid-input input[type="date"],.wy-control-group.fluid-input input[type="month"],.wy-control-group.fluid-input input[type="time"],.wy-control-group.fluid-input input[type="datetime"],.wy-control-group.fluid-input input[type="datetime-local"],.wy-control-group.fluid-input input[type="week"],.wy-control-group.fluid-input input[type="number"],.wy-control-group.fluid-input input[type="search"],.wy-control-group.fluid-input input[type="tel"],.wy-control-group.fluid-input input[type="color"]{width:100%}.wy-form-message-inline{display:inline-block;padding-left:0.3em;color:#666;vertical-align:middle;font-size:90%}.wy-form-message{display:block;color:#999;font-size:70%;margin-top:0.3125em;font-style:italic}.wy-form-message p{font-size:inherit;font-style:italic;margin-bottom:6px}.wy-form-message p:last-child{margin-bottom:0}input{line-height:normal}input[type="button"],input[type="reset"],input[type="submit"]{-webkit-appearance:button;cursor:pointer;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;*overflow:visible}input[type="text"],input[type="password"],input[type="email"],input[type="url"],input[type="date"],input[type="month"],input[type="time"],input[type="datetime"],input[type="datetime-local"],input[type="week"],input[type="number"],input[type="search"],input[type="tel"],input[type="color"]{-webkit-appearance:none;padding:6px;display:inline-block;border:1px solid #ccc;font-size:80%;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;box-shadow:inset 0 1px 3px #ddd;border-radius:0;-webkit-transition:border 0.3s linear;-moz-transition:border 0.3s linear;transition:border 0.3s linear}input[type="datetime-local"]{padding:0.34375em 0.625em}input[disabled]{cursor:default}input[type="checkbox"],input[type="radio"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0;margin-right:0.3125em;*height:13px;*width:13px}input[type="search"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}input[type="search"]::-webkit-search-cancel-button,input[type="search"]::-webkit-search-decoration{-webkit-appearance:none}input[type="text"]:focus,input[type="password"]:focus,input[type="email"]:focus,input[type="url"]:focus,input[type="date"]:focus,input[type="month"]:focus,input[type="time"]:focus,input[type="datetime"]:focus,input[type="datetime-local"]:focus,input[type="week"]:focus,input[type="number"]:focus,input[type="search"]:focus,input[type="tel"]:focus,input[type="color"]:focus{outline:0;outline:thin dotted \9;border-color:#333}input.no-focus:focus{border-color:#ccc !important}input[type="file"]:focus,input[type="radio"]:focus,input[type="checkbox"]:focus{outline:thin dotted #333;outline:1px auto #129FEA}input[type="text"][disabled],input[type="password"][disabled],input[type="email"][disabled],input[type="url"][disabled],input[type="date"][disabled],input[type="month"][disabled],input[type="time"][disabled],input[type="datetime"][disabled],input[type="datetime-local"][disabled],input[type="week"][disabled],input[type="number"][disabled],input[type="search"][disabled],input[type="tel"][disabled],input[type="color"][disabled]{cursor:not-allowed;background-color:#fafafa}input:focus:invalid,textarea:focus:invalid,select:focus:invalid{color:#E74C3C;border:1px solid #E74C3C}input:focus:invalid:focus,textarea:focus:invalid:focus,select:focus:invalid:focus{border-color:#E74C3C}input[type="file"]:focus:invalid:focus,input[type="radio"]:focus:invalid:focus,input[type="checkbox"]:focus:invalid:focus{outline-color:#E74C3C}input.wy-input-large{padding:12px;font-size:100%}textarea{overflow:auto;vertical-align:top;width:100%;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif}select,textarea{padding:0.5em 0.625em;display:inline-block;border:1px solid #ccc;font-size:80%;box-shadow:inset 0 1px 3px #ddd;-webkit-transition:border 0.3s linear;-moz-transition:border 0.3s linear;transition:border 0.3s linear}select{border:1px solid #ccc;background-color:#fff}select[multiple]{height:auto}select:focus,textarea:focus{outline:0}select[disabled],textarea[disabled],input[readonly],select[readonly],textarea[readonly]{cursor:not-allowed;background-color:#fafafa}input[type="radio"][disabled],input[type="checkbox"][disabled]{cursor:not-allowed}.wy-checkbox,.wy-radio{margin:6px 0;color:#404040;display:block}.wy-checkbox input,.wy-radio input{vertical-align:baseline}.wy-form-message-inline{display:inline-block;*display:inline;*zoom:1;vertical-align:middle}.wy-input-prefix,.wy-input-suffix{white-space:nowrap;padding:6px}.wy-input-prefix .wy-input-context,.wy-input-suffix .wy-input-context{line-height:27px;padding:0 8px;display:inline-block;font-size:80%;background-color:#f3f6f6;border:solid 1px #ccc;color:#999}.wy-input-suffix .wy-input-context{border-left:0}.wy-input-prefix .wy-input-context{border-right:0}.wy-switch{width:36px;height:12px;margin:12px 0;position:relative;border-radius:4px;background:#ccc;cursor:pointer;-webkit-transition:all 0.2s ease-in-out;-moz-transition:all 0.2s ease-in-out;transition:all 0.2s ease-in-out}.wy-switch:before{position:absolute;content:"";display:block;width:18px;height:18px;border-radius:4px;background:#999;left:-3px;top:-3px;-webkit-transition:all 0.2s ease-in-out;-moz-transition:all 0.2s ease-in-out;transition:all 0.2s ease-in-out}.wy-switch:after{content:"false";position:absolute;left:48px;display:block;font-size:12px;color:#ccc}.wy-switch.active{background:#1e8449}.wy-switch.active:before{left:24px;background:#27AE60}.wy-switch.active:after{content:"true"}.wy-switch.disabled,.wy-switch.active.disabled{cursor:not-allowed}.wy-control-group.wy-control-group-error .wy-form-message,.wy-control-group.wy-control-group-error>label{color:#E74C3C}.wy-control-group.wy-control-group-error input[type="text"],.wy-control-group.wy-control-group-error input[type="password"],.wy-control-group.wy-control-group-error input[type="email"],.wy-control-group.wy-control-group-error input[type="url"],.wy-control-group.wy-control-group-error input[type="date"],.wy-control-group.wy-control-group-error input[type="month"],.wy-control-group.wy-control-group-error input[type="time"],.wy-control-group.wy-control-group-error input[type="datetime"],.wy-control-group.wy-control-group-error input[type="datetime-local"],.wy-control-group.wy-control-group-error input[type="week"],.wy-control-group.wy-control-group-error input[type="number"],.wy-control-group.wy-control-group-error input[type="search"],.wy-control-group.wy-control-group-error input[type="tel"],.wy-control-group.wy-control-group-error input[type="color"]{border:solid 1px #E74C3C}.wy-control-group.wy-control-group-error textarea{border:solid 1px #E74C3C}.wy-inline-validate{white-space:nowrap}.wy-inline-validate .wy-input-context{padding:0.5em 0.625em;display:inline-block;font-size:80%}.wy-inline-validate.wy-inline-validate-success .wy-input-context{color:#27AE60}.wy-inline-validate.wy-inline-validate-danger .wy-input-context{color:#E74C3C}.wy-inline-validate.wy-inline-validate-warning .wy-input-context{color:#E67E22}.wy-inline-validate.wy-inline-validate-info .wy-input-context{color:#2980B9}.rotate-90{-webkit-transform:rotate(90deg);-moz-transform:rotate(90deg);-ms-transform:rotate(90deg);-o-transform:rotate(90deg);transform:rotate(90deg)}.rotate-180{-webkit-transform:rotate(180deg);-moz-transform:rotate(180deg);-ms-transform:rotate(180deg);-o-transform:rotate(180deg);transform:rotate(180deg)}.rotate-270{-webkit-transform:rotate(270deg);-moz-transform:rotate(270deg);-ms-transform:rotate(270deg);-o-transform:rotate(270deg);transform:rotate(270deg)}.mirror{-webkit-transform:scaleX(-1);-moz-transform:scaleX(-1);-ms-transform:scaleX(-1);-o-transform:scaleX(-1);transform:scaleX(-1)}.mirror.rotate-90{-webkit-transform:scaleX(-1) rotate(90deg);-moz-transform:scaleX(-1) rotate(90deg);-ms-transform:scaleX(-1) rotate(90deg);-o-transform:scaleX(-1) rotate(90deg);transform:scaleX(-1) rotate(90deg)}.mirror.rotate-180{-webkit-transform:scaleX(-1) rotate(180deg);-moz-transform:scaleX(-1) rotate(180deg);-ms-transform:scaleX(-1) rotate(180deg);-o-transform:scaleX(-1) rotate(180deg);transform:scaleX(-1) rotate(180deg)}.mirror.rotate-270{-webkit-transform:scaleX(-1) rotate(270deg);-moz-transform:scaleX(-1) rotate(270deg);-ms-transform:scaleX(-1) rotate(270deg);-o-transform:scaleX(-1) rotate(270deg);transform:scaleX(-1) rotate(270deg)}@media only screen and (max-width: 480px){.wy-form button[type="submit"]{margin:0.7em 0 0}.wy-form input[type="text"],.wy-form input[type="password"],.wy-form input[type="email"],.wy-form input[type="url"],.wy-form input[type="date"],.wy-form input[type="month"],.wy-form input[type="time"],.wy-form input[type="datetime"],.wy-form input[type="datetime-local"],.wy-form input[type="week"],.wy-form input[type="number"],.wy-form input[type="search"],.wy-form input[type="tel"],.wy-form input[type="color"]{margin-bottom:0.3em;display:block}.wy-form label{margin-bottom:0.3em;display:block}.wy-form input[type="password"],.wy-form input[type="email"],.wy-form input[type="url"],.wy-form input[type="date"],.wy-form input[type="month"],.wy-form input[type="time"],.wy-form input[type="datetime"],.wy-form input[type="datetime-local"],.wy-form input[type="week"],.wy-form input[type="number"],.wy-form input[type="search"],.wy-form input[type="tel"],.wy-form input[type="color"]{margin-bottom:0}.wy-form-aligned .wy-control-group label{margin-bottom:0.3em;text-align:left;display:block;width:100%}.wy-form-aligned .wy-control{margin:1.5em 0 0 0}.wy-form .wy-help-inline,.wy-form-message-inline,.wy-form-message{display:block;font-size:80%;padding:6px 0}}@media screen and (max-width: 768px){.tablet-hide{display:none}}@media screen and (max-width: 480px){.mobile-hide{display:none}}.float-left{float:left}.float-right{float:right}.full-width{width:100%}.wy-table,.rst-content table.docutils,.rst-content table.field-list{border-collapse:collapse;border-spacing:0;empty-cells:show;margin-bottom:24px}.wy-table caption,.rst-content table.docutils caption,.rst-content table.field-list caption{color:#000;font:italic 85%/1 arial,sans-serif;padding:1em 0;text-align:center}.wy-table td,.rst-content table.docutils td,.rst-content table.field-list td,.wy-table th,.rst-content table.docutils th,.rst-content table.field-list th{font-size:90%;margin:0;overflow:visible;padding:8px 16px}.wy-table td:first-child,.rst-content table.docutils td:first-child,.rst-content table.field-list td:first-child,.wy-table th:first-child,.rst-content table.docutils th:first-child,.rst-content table.field-list th:first-child{border-left-width:0}.wy-table thead,.rst-content table.docutils thead,.rst-content table.field-list thead{color:#000;text-align:left;vertical-align:bottom;white-space:nowrap}.wy-table thead th,.rst-content table.docutils thead th,.rst-content table.field-list thead th{font-weight:bold;border-bottom:solid 2px #e1e4e5}.wy-table td,.rst-content table.docutils td,.rst-content table.field-list td{background-color:transparent;vertical-align:middle}.wy-table td p,.rst-content table.docutils td p,.rst-content table.field-list td p{line-height:18px}.wy-table td p:last-child,.rst-content table.docutils td p:last-child,.rst-content table.field-list td p:last-child{margin-bottom:0}.wy-table .wy-table-cell-min,.rst-content table.docutils .wy-table-cell-min,.rst-content table.field-list .wy-table-cell-min{width:1%;padding-right:0}.wy-table .wy-table-cell-min input[type=checkbox],.rst-content table.docutils .wy-table-cell-min input[type=checkbox],.rst-content table.field-list .wy-table-cell-min input[type=checkbox],.wy-table .wy-table-cell-min input[type=checkbox],.rst-content table.docutils .wy-table-cell-min input[type=checkbox],.rst-content table.field-list .wy-table-cell-min input[type=checkbox]{margin:0}.wy-table-secondary{color:gray;font-size:90%}.wy-table-tertiary{color:gray;font-size:80%}.wy-table-odd td,.wy-table-striped tr:nth-child(2n-1) td,.rst-content table.docutils:not(.field-list) tr:nth-child(2n-1) td{background-color:#f3f6f6}.wy-table-backed{background-color:#f3f6f6}.wy-table-bordered-all,.rst-content table.docutils{border:1px solid #e1e4e5}.wy-table-bordered-all td,.rst-content table.docutils td{border-bottom:1px solid #e1e4e5;border-left:1px solid #e1e4e5}.wy-table-bordered-all tbody>tr:last-child td,.rst-content table.docutils tbody>tr:last-child td{border-bottom-width:0}.wy-table-bordered{border:1px solid #e1e4e5}.wy-table-bordered-rows td{border-bottom:1px solid #e1e4e5}.wy-table-bordered-rows tbody>tr:last-child td{border-bottom-width:0}.wy-table-horizontal tbody>tr:last-child td{border-bottom-width:0}.wy-table-horizontal td,.wy-table-horizontal th{border-width:0 0 1px 0;border-bottom:1px solid #e1e4e5}.wy-table-horizontal tbody>tr:last-child td{border-bottom-width:0}.wy-table-responsive{margin-bottom:24px;max-width:100%;overflow:auto}.wy-table-responsive table{margin-bottom:0 !important}.wy-table-responsive table td,.wy-table-responsive table th{white-space:nowrap}a{color:#2980B9;text-decoration:none;cursor:pointer}a:hover{color:#3091d1}a:visited{color:#9B59B6}html{height:100%;overflow-x:hidden}body{font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;font-weight:normal;color:#404040;min-height:100%;overflow-x:hidden;background:#edf0f2}.wy-text-left{text-align:left}.wy-text-center{text-align:center}.wy-text-right{text-align:right}.wy-text-large{font-size:120%}.wy-text-normal{font-size:100%}.wy-text-small,small{font-size:80%}.wy-text-strike{text-decoration:line-through}.wy-text-warning{color:#E67E22 !important}a.wy-text-warning:hover{color:#eb9950 !important}.wy-text-info{color:#2980B9 !important}a.wy-text-info:hover{color:#409ad5 !important}.wy-text-success{color:#27AE60 !important}a.wy-text-success:hover{color:#36d278 !important}.wy-text-danger{color:#E74C3C !important}a.wy-text-danger:hover{color:#ed7669 !important}.wy-text-neutral{color:#404040 !important}a.wy-text-neutral:hover{color:#595959 !important}h1,h2,.rst-content .toctree-wrapper p.caption,h3,h4,h5,h6,legend{margin-top:0;font-weight:700;font-family:"Roboto Slab","ff-tisa-web-pro","Georgia",Arial,sans-serif}p{line-height:24px;margin:0;font-size:16px;margin-bottom:24px}h1{font-size:175%}h2,.rst-content .toctree-wrapper p.caption{font-size:150%}h3{font-size:125%}h4{font-size:115%}h5{font-size:110%}h6{font-size:100%}hr{display:block;height:1px;border:0;border-top:1px solid #e1e4e5;margin:24px 0;padding:0}code,.rst-content tt,.rst-content code{white-space:nowrap;max-width:100%;background:#fff;border:solid 1px #e1e4e5;font-size:75%;padding:0 5px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;color:#E74C3C;overflow-x:auto}code.code-large,.rst-content tt.code-large{font-size:90%}.wy-plain-list-disc,.rst-content .section ul,.rst-content .toctree-wrapper ul,article ul{list-style:disc;line-height:24px;margin-bottom:24px}.wy-plain-list-disc li,.rst-content .section ul li,.rst-content .toctree-wrapper ul li,article ul li{list-style:disc;margin-left:24px}.wy-plain-list-disc li p:last-child,.rst-content .section ul li p:last-child,.rst-content .toctree-wrapper ul li p:last-child,article ul li p:last-child{margin-bottom:0}.wy-plain-list-disc li ul,.rst-content .section ul li ul,.rst-content .toctree-wrapper ul li ul,article ul li ul{margin-bottom:0}.wy-plain-list-disc li li,.rst-content .section ul li li,.rst-content .toctree-wrapper ul li li,article ul li li{list-style:circle}.wy-plain-list-disc li li li,.rst-content .section ul li li li,.rst-content .toctree-wrapper ul li li li,article ul li li li{list-style:square}.wy-plain-list-disc li ol li,.rst-content .section ul li ol li,.rst-content .toctree-wrapper ul li ol li,article ul li ol li{list-style:decimal}.wy-plain-list-decimal,.rst-content .section ol,.rst-content ol.arabic,article ol{list-style:decimal;line-height:24px;margin-bottom:24px}.wy-plain-list-decimal li,.rst-content .section ol li,.rst-content ol.arabic li,article ol li{list-style:decimal;margin-left:24px}.wy-plain-list-decimal li p:last-child,.rst-content .section ol li p:last-child,.rst-content ol.arabic li p:last-child,article ol li p:last-child{margin-bottom:0}.wy-plain-list-decimal li ul,.rst-content .section ol li ul,.rst-content ol.arabic li ul,article ol li ul{margin-bottom:0}.wy-plain-list-decimal li ul li,.rst-content .section ol li ul li,.rst-content ol.arabic li ul li,article ol li ul li{list-style:disc}.codeblock-example{border:1px solid #e1e4e5;border-bottom:none;padding:24px;padding-top:48px;font-weight:500;background:#fff;position:relative}.codeblock-example:after{content:"Example";position:absolute;top:0px;left:0px;background:#9B59B6;color:#fff;padding:6px 12px}.codeblock-example.prettyprint-example-only{border:1px solid #e1e4e5;margin-bottom:24px}.codeblock,pre.literal-block,.rst-content .literal-block,.rst-content pre.literal-block,div[class^='highlight']{border:1px solid #e1e4e5;padding:0px;overflow-x:auto;background:#fff;margin:1px 0 24px 0}.codeblock div[class^='highlight'],pre.literal-block div[class^='highlight'],.rst-content .literal-block div[class^='highlight'],div[class^='highlight'] div[class^='highlight']{border:none;background:none;margin:0}div[class^='highlight'] td.code{width:100%}.linenodiv pre{border-right:solid 1px #e6e9ea;margin:0;padding:12px 12px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;font-size:12px;line-height:1.5;color:#d9d9d9}div[class^='highlight'] pre{white-space:pre;margin:0;padding:12px 12px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;font-size:12px;line-height:1.5;display:block;overflow:auto;color:#404040}@media print{.codeblock,pre.literal-block,.rst-content .literal-block,.rst-content pre.literal-block,div[class^='highlight'],div[class^='highlight'] pre{white-space:pre-wrap}}.hll{background-color:#ffc;margin:0 -12px;padding:0 12px;display:block}.c{color:#998;font-style:italic}.err{color:#a61717;background-color:#e3d2d2}.k{font-weight:bold}.o{font-weight:bold}.cm{color:#998;font-style:italic}.cp{color:#999;font-weight:bold}.c1{color:#998;font-style:italic}.cs{color:#999;font-weight:bold;font-style:italic}.gd{color:#000;background-color:#fdd}.gd .x{color:#000;background-color:#faa}.ge{font-style:italic}.gr{color:#a00}.gh{color:#999}.gi{color:#000;background-color:#dfd}.gi .x{color:#000;background-color:#afa}.go{color:#888}.gp{color:#555}.gs{font-weight:bold}.gu{color:purple;font-weight:bold}.gt{color:#a00}.kc{font-weight:bold}.kd{font-weight:bold}.kn{font-weight:bold}.kp{font-weight:bold}.kr{font-weight:bold}.kt{color:#458;font-weight:bold}.m{color:#099}.s{color:#d14}.n{color:#333}.na{color:teal}.nb{color:#0086b3}.nc{color:#458;font-weight:bold}.no{color:teal}.ni{color:purple}.ne{color:#900;font-weight:bold}.nf{color:#900;font-weight:bold}.nn{color:#555}.nt{color:navy}.nv{color:teal}.ow{font-weight:bold}.w{color:#bbb}.mf{color:#099}.mh{color:#099}.mi{color:#099}.mo{color:#099}.sb{color:#d14}.sc{color:#d14}.sd{color:#d14}.s2{color:#d14}.se{color:#d14}.sh{color:#d14}.si{color:#d14}.sx{color:#d14}.sr{color:#009926}.s1{color:#d14}.ss{color:#990073}.bp{color:#999}.vc{color:teal}.vg{color:teal}.vi{color:teal}.il{color:#099}.gc{color:#999;background-color:#EAF2F5}.wy-breadcrumbs li{display:inline-block}.wy-breadcrumbs li.wy-breadcrumbs-aside{float:right}.wy-breadcrumbs li a{display:inline-block;padding:5px}.wy-breadcrumbs li a:first-child{padding-left:0}.wy-breadcrumbs li code,.wy-breadcrumbs li .rst-content tt,.rst-content .wy-breadcrumbs li tt{padding:5px;border:none;background:none}.wy-breadcrumbs li code.literal,.wy-breadcrumbs li .rst-content tt.literal,.rst-content .wy-breadcrumbs li tt.literal{color:#404040}.wy-breadcrumbs-extra{margin-bottom:0;color:#b3b3b3;font-size:80%;display:inline-block}@media screen and (max-width: 480px){.wy-breadcrumbs-extra{display:none}.wy-breadcrumbs li.wy-breadcrumbs-aside{display:none}}@media print{.wy-breadcrumbs li.wy-breadcrumbs-aside{display:none}}.wy-affix{position:fixed;top:1.618em}.wy-menu a:hover{text-decoration:none}.wy-menu-horiz{*zoom:1}.wy-menu-horiz:before,.wy-menu-horiz:after{display:table;content:""}.wy-menu-horiz:after{clear:both}.wy-menu-horiz ul,.wy-menu-horiz li{display:inline-block}.wy-menu-horiz li:hover{background:rgba(255,255,255,0.1)}.wy-menu-horiz li.divide-left{border-left:solid 1px #404040}.wy-menu-horiz li.divide-right{border-right:solid 1px #404040}.wy-menu-horiz a{height:32px;display:inline-block;line-height:32px;padding:0 16px}.wy-menu-vertical{width:300px}.wy-menu-vertical header,.wy-menu-vertical p.caption{height:32px;display:inline-block;line-height:32px;padding:0 1.618em;margin-bottom:0;display:block;font-weight:bold;text-transform:uppercase;font-size:80%;color:#555;white-space:nowrap}.wy-menu-vertical ul{margin-bottom:0}.wy-menu-vertical li.divide-top{border-top:solid 1px #404040}.wy-menu-vertical li.divide-bottom{border-bottom:solid 1px #404040}.wy-menu-vertical li.current{background:#e3e3e3}.wy-menu-vertical li.current a{color:gray;border-right:solid 1px #c9c9c9;padding:0.4045em 2.427em}.wy-menu-vertical li.current a:hover{background:#d6d6d6}.wy-menu-vertical li code,.wy-menu-vertical li .rst-content tt,.rst-content .wy-menu-vertical li tt{border:none;background:inherit;color:inherit;padding-left:0;padding-right:0}.wy-menu-vertical li span.toctree-expand{display:block;float:left;margin-left:-1.2em;font-size:0.8em;line-height:1.6em;color:#4d4d4d}.wy-menu-vertical li.on a,.wy-menu-vertical li.current>a{color:#404040;padding:0.4045em 1.618em;font-weight:bold;position:relative;background:#fcfcfc;border:none;border-bottom:solid 1px #c9c9c9;border-top:solid 1px #c9c9c9;padding-left:1.618em -4px}.wy-menu-vertical li.on a:hover,.wy-menu-vertical li.current>a:hover{background:#fcfcfc}.wy-menu-vertical li.on a:hover span.toctree-expand,.wy-menu-vertical li.current>a:hover span.toctree-expand{color:gray}.wy-menu-vertical li.on a span.toctree-expand,.wy-menu-vertical li.current>a span.toctree-expand{display:block;font-size:0.8em;line-height:1.6em;color:#333}.wy-menu-vertical li.toctree-l1.current li.toctree-l2>ul,.wy-menu-vertical li.toctree-l2.current li.toctree-l3>ul{display:none}.wy-menu-vertical li.toctree-l1.current li.toctree-l2.current>ul,.wy-menu-vertical li.toctree-l2.current li.toctree-l3.current>ul{display:block}.wy-menu-vertical li.toctree-l2.current>a{background:#c9c9c9;padding:0.4045em 2.427em}.wy-menu-vertical li.toctree-l2.current li.toctree-l3>a{display:block;background:#c9c9c9;padding:0.4045em 4.045em}.wy-menu-vertical li.toctree-l2 a:hover span.toctree-expand{color:gray}.wy-menu-vertical li.toctree-l2 span.toctree-expand{color:#a3a3a3}.wy-menu-vertical li.toctree-l3{font-size:0.9em}.wy-menu-vertical li.toctree-l3.current>a{background:#bdbdbd;padding:0.4045em 4.045em}.wy-menu-vertical li.toctree-l3.current li.toctree-l4>a{display:block;background:#bdbdbd;padding:0.4045em 5.663em;border-top:none;border-bottom:none}.wy-menu-vertical li.toctree-l3 a:hover span.toctree-expand{color:gray}.wy-menu-vertical li.toctree-l3 span.toctree-expand{color:#969696}.wy-menu-vertical li.toctree-l4{font-size:0.9em}.wy-menu-vertical li.current ul{display:block}.wy-menu-vertical li ul{margin-bottom:0;display:none}.wy-menu-vertical .local-toc li ul{display:block}.wy-menu-vertical li ul li a{margin-bottom:0;color:#b3b3b3;font-weight:normal}.wy-menu-vertical a{display:inline-block;line-height:18px;padding:0.4045em 1.618em;display:block;position:relative;font-size:90%;color:#b3b3b3}.wy-menu-vertical a:hover{background-color:#4e4a4a;cursor:pointer}.wy-menu-vertical a:hover span.toctree-expand{color:#b3b3b3}.wy-menu-vertical a:active{background-color:#2980B9;cursor:pointer;color:#fff}.wy-menu-vertical a:active span.toctree-expand{color:#fff}.wy-side-nav-search{display:block;width:300px;padding:0.809em;margin-bottom:0.809em;z-index:200;background-color:#2980B9;text-align:center;padding:0.809em;display:block;color:#fcfcfc;margin-bottom:0.809em}.wy-side-nav-search input[type=text]{width:100%;border-radius:50px;padding:6px 12px;border-color:#2472a4}.wy-side-nav-search img{display:block;margin:auto auto 0.809em auto;height:45px;width:45px;background-color:#2980B9;padding:5px;border-radius:100%}.wy-side-nav-search>a,.wy-side-nav-search .wy-dropdown>a{color:#fcfcfc;font-size:100%;font-weight:bold;display:inline-block;padding:4px 6px;margin-bottom:0.809em}.wy-side-nav-search>a:hover,.wy-side-nav-search .wy-dropdown>a:hover{background:rgba(255,255,255,0.1)}.wy-side-nav-search>a img.logo,.wy-side-nav-search .wy-dropdown>a img.logo{display:block;margin:0 auto;height:auto;width:auto;border-radius:0;max-width:100%;background:transparent}.wy-side-nav-search>a.icon img.logo,.wy-side-nav-search .wy-dropdown>a.icon img.logo{margin-top:0.85em}.wy-side-nav-search>div.version{margin-top:-0.4045em;margin-bottom:0.809em;font-weight:normal;color:rgba(255,255,255,0.3)}.wy-nav .wy-menu-vertical header{color:#2980B9}.wy-nav .wy-menu-vertical a{color:#b3b3b3}.wy-nav .wy-menu-vertical a:hover{background-color:#2980B9;color:#fff}[data-menu-wrap]{-webkit-transition:all 0.2s ease-in;-moz-transition:all 0.2s ease-in;transition:all 0.2s ease-in;position:absolute;opacity:1;width:100%;opacity:0}[data-menu-wrap].move-center{left:0;right:auto;opacity:1}[data-menu-wrap].move-left{right:auto;left:-100%;opacity:0}[data-menu-wrap].move-right{right:-100%;left:auto;opacity:0}.wy-body-for-nav{background:left repeat-y #fcfcfc;background-image:url();background-size:300px 1px}.wy-grid-for-nav{position:absolute;width:100%;height:100%}.wy-nav-side{position:fixed;top:0;bottom:0;left:0;padding-bottom:2em;width:300px;overflow-x:hidden;overflow-y:hidden;min-height:100%;background:#343131;z-index:200}.wy-side-scroll{width:320px;position:relative;overflow-x:hidden;overflow-y:scroll;height:100%}.wy-nav-top{display:none;background:#2980B9;color:#fff;padding:0.4045em 0.809em;position:relative;line-height:50px;text-align:center;font-size:100%;*zoom:1}.wy-nav-top:before,.wy-nav-top:after{display:table;content:""}.wy-nav-top:after{clear:both}.wy-nav-top a{color:#fff;font-weight:bold}.wy-nav-top img{margin-right:12px;height:45px;width:45px;background-color:#2980B9;padding:5px;border-radius:100%}.wy-nav-top i{font-size:30px;float:left;cursor:pointer}.wy-nav-content-wrap{margin-left:300px;background:#fcfcfc;min-height:100%}.wy-nav-content{padding:1.618em 3.236em;height:100%;max-width:800px;margin:auto}.wy-body-mask{position:fixed;width:100%;height:100%;background:rgba(0,0,0,0.2);display:none;z-index:499}.wy-body-mask.on{display:block}footer{color:#999}footer p{margin-bottom:12px}footer span.commit code,footer span.commit .rst-content tt,.rst-content footer span.commit tt{padding:0px;font-family:Consolas,"Andale Mono WT","Andale Mono","Lucida Console","Lucida Sans Typewriter","DejaVu Sans Mono","Bitstream Vera Sans Mono","Liberation Mono","Nimbus Mono L",Monaco,"Courier New",Courier,monospace;font-size:1em;background:none;border:none;color:#999}.rst-footer-buttons{*zoom:1}.rst-footer-buttons:before,.rst-footer-buttons:after{display:table;content:""}.rst-footer-buttons:after{clear:both}#search-results .search li{margin-bottom:24px;border-bottom:solid 1px #e1e4e5;padding-bottom:24px}#search-results .search li:first-child{border-top:solid 1px #e1e4e5;padding-top:24px}#search-results .search li a{font-size:120%;margin-bottom:12px;display:inline-block}#search-results .context{color:gray;font-size:90%}@media screen and (max-width: 768px){.wy-body-for-nav{background:#fcfcfc}.wy-nav-top{display:block}.wy-nav-side{left:-300px}.wy-nav-side.shift{width:85%;left:0}.wy-side-scroll{width:auto}.wy-side-nav-search{width:auto}.wy-menu.wy-menu-vertical{width:auto}.wy-nav-content-wrap{margin-left:0}.wy-nav-content-wrap .wy-nav-content{padding:1.618em}.wy-nav-content-wrap.shift{position:fixed;min-width:100%;left:85%;top:0;height:100%;overflow:hidden}}@media screen and (min-width: 1400px){.wy-nav-content-wrap{background:rgba(0,0,0,0.05)}.wy-nav-content{margin:0;background:#fcfcfc}}@media print{.rst-versions,footer,.wy-nav-side{display:none}.wy-nav-content-wrap{margin-left:0}}.rst-versions{position:fixed;bottom:0;left:0;width:300px;color:#fcfcfc;background:#1f1d1d;border-top:solid 10px #343131;font-family:"Lato","proxima-nova","Helvetica Neue",Arial,sans-serif;z-index:400}.rst-versions a{color:#2980B9;text-decoration:none}.rst-versions .rst-badge-small{display:none}.rst-versions .rst-current-version{padding:12px;background-color:#272525;display:block;text-align:right;font-size:90%;cursor:pointer;color:#27AE60;*zoom:1}.rst-versions .rst-current-version:before,.rst-versions .rst-current-version:after{display:table;content:""}.rst-versions .rst-current-version:after{clear:both}.rst-versions .rst-current-version .fa,.rst-versions .rst-current-version .wy-menu-vertical li span.toctree-expand,.wy-menu-vertical li .rst-versions .rst-current-version span.toctree-expand,.rst-versions .rst-current-version .rst-content .admonition-title,.rst-content .rst-versions .rst-current-version .admonition-title,.rst-versions .rst-current-version .rst-content h1 .headerlink,.rst-content h1 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h2 .headerlink,.rst-content h2 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h3 .headerlink,.rst-content h3 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h4 .headerlink,.rst-content h4 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h5 .headerlink,.rst-content h5 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content h6 .headerlink,.rst-content h6 .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content dl dt .headerlink,.rst-content dl dt .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content p.caption .headerlink,.rst-content p.caption .rst-versions .rst-current-version .headerlink,.rst-versions .rst-current-version .rst-content tt.download span:first-child,.rst-content tt.download .rst-versions .rst-current-version span:first-child,.rst-versions .rst-current-version .rst-content code.download span:first-child,.rst-content code.download .rst-versions .rst-current-version span:first-child,.rst-versions .rst-current-version .icon{color:#fcfcfc}.rst-versions .rst-current-version .fa-book,.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version .icon-book{float:left}.rst-versions .rst-current-version.rst-out-of-date{background-color:#E74C3C;color:#fff}.rst-versions .rst-current-version.rst-active-old-version{background-color:#F1C40F;color:#000}.rst-versions.shift-up .rst-other-versions{display:block}.rst-versions .rst-other-versions{font-size:90%;padding:12px;color:gray;display:none}.rst-versions .rst-other-versions hr{display:block;height:1px;border:0;margin:20px 0;padding:0;border-top:solid 1px #413d3d}.rst-versions .rst-other-versions dd{display:inline-block;margin:0}.rst-versions .rst-other-versions dd a{display:inline-block;padding:6px;color:#fcfcfc}.rst-versions.rst-badge{width:auto;bottom:20px;right:20px;left:auto;border:none;max-width:300px}.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge .fa-book,.rst-versions.rst-badge .icon-book{float:none}.rst-versions.rst-badge.shift-up .rst-current-version{text-align:right}.rst-versions.rst-badge.shift-up .rst-current-version .fa-book,.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge.shift-up .rst-current-version .icon-book{float:left}.rst-versions.rst-badge .rst-current-version{width:auto;height:30px;line-height:30px;padding:0 6px;display:block;text-align:center}@media screen and (max-width: 768px){.rst-versions{width:85%;display:none}.rst-versions.shift{display:block}img{width:100%;height:auto}}.rst-content img{max-width:100%;height:auto !important}.rst-content div.figure{margin-bottom:24px}.rst-content div.figure p.caption{font-style:italic}.rst-content div.figure.align-center{text-align:center}.rst-content .section>img,.rst-content .section>a>img{margin-bottom:24px}.rst-content blockquote{margin-left:24px;line-height:24px;margin-bottom:24px}.rst-content .note .last,.rst-content .attention .last,.rst-content .caution .last,.rst-content .danger .last,.rst-content .error .last,.rst-content .hint .last,.rst-content .important .last,.rst-content .tip .last,.rst-content .warning .last,.rst-content .seealso .last,.rst-content .admonition-todo .last{margin-bottom:0}.rst-content .admonition-title:before{margin-right:4px}.rst-content .admonition table{border-color:rgba(0,0,0,0.1)}.rst-content .admonition table td,.rst-content .admonition table th{background:transparent !important;border-color:rgba(0,0,0,0.1) !important}.rst-content .section ol.loweralpha,.rst-content .section ol.loweralpha li{list-style:lower-alpha}.rst-content .section ol.upperalpha,.rst-content .section ol.upperalpha li{list-style:upper-alpha}.rst-content .section ol p,.rst-content .section ul p{margin-bottom:12px}.rst-content .line-block{margin-left:24px}.rst-content .topic-title{font-weight:bold;margin-bottom:12px}.rst-content .toc-backref{color:#404040}.rst-content .align-right{float:right;margin:0px 0px 24px 24px}.rst-content .align-left{float:left;margin:0px 24px 24px 0px}.rst-content .align-center{margin:auto;display:block}.rst-content h1 .headerlink,.rst-content h2 .headerlink,.rst-content .toctree-wrapper p.caption .headerlink,.rst-content h3 .headerlink,.rst-content h4 .headerlink,.rst-content h5 .headerlink,.rst-content h6 .headerlink,.rst-content dl dt .headerlink,.rst-content p.caption .headerlink{display:none;visibility:hidden;font-size:14px}.rst-content h1 .headerlink:after,.rst-content h2 .headerlink:after,.rst-content .toctree-wrapper p.caption .headerlink:after,.rst-content h3 .headerlink:after,.rst-content h4 .headerlink:after,.rst-content h5 .headerlink:after,.rst-content h6 .headerlink:after,.rst-content dl dt .headerlink:after,.rst-content p.caption .headerlink:after{visibility:visible;content:"ïƒ";font-family:FontAwesome;display:inline-block}.rst-content h1:hover .headerlink,.rst-content h2:hover .headerlink,.rst-content .toctree-wrapper p.caption:hover .headerlink,.rst-content h3:hover .headerlink,.rst-content h4:hover .headerlink,.rst-content h5:hover .headerlink,.rst-content h6:hover .headerlink,.rst-content dl dt:hover .headerlink,.rst-content p.caption:hover .headerlink{display:inline-block}.rst-content .sidebar{float:right;width:40%;display:block;margin:0 0 24px 24px;padding:24px;background:#f3f6f6;border:solid 1px #e1e4e5}.rst-content .sidebar p,.rst-content .sidebar ul,.rst-content .sidebar dl{font-size:90%}.rst-content .sidebar .last{margin-bottom:0}.rst-content .sidebar .sidebar-title{display:block;font-family:"Roboto Slab","ff-tisa-web-pro","Georgia",Arial,sans-serif;font-weight:bold;background:#e1e4e5;padding:6px 12px;margin:-24px;margin-bottom:24px;font-size:100%}.rst-content .highlighted{background:#F1C40F;display:inline-block;font-weight:bold;padding:0 6px}.rst-content .footnote-reference,.rst-content .citation-reference{vertical-align:super;font-size:90%}.rst-content table.docutils.citation,.rst-content table.docutils.footnote{background:none;border:none;color:#999}.rst-content table.docutils.citation td,.rst-content table.docutils.citation tr,.rst-content table.docutils.footnote td,.rst-content table.docutils.footnote tr{border:none;background-color:transparent !important;white-space:normal}.rst-content table.docutils.citation td.label,.rst-content table.docutils.footnote td.label{padding-left:0;padding-right:0;vertical-align:top}.rst-content table.docutils.citation tt,.rst-content table.docutils.citation code,.rst-content table.docutils.footnote tt,.rst-content table.docutils.footnote code{color:#555}.rst-content table.field-list{border:none}.rst-content table.field-list td{border:none;padding-top:5px}.rst-content table.field-list td>strong{display:inline-block;margin-top:3px}.rst-content table.field-list .field-name{padding-right:10px;text-align:left;white-space:nowrap}.rst-content table.field-list .field-body{text-align:left;padding-left:0}.rst-content tt,.rst-content tt,.rst-content code{color:#000;padding:2px 5px}.rst-content tt big,.rst-content tt em,.rst-content tt big,.rst-content code big,.rst-content tt em,.rst-content code em{font-size:100% !important;line-height:normal}.rst-content tt.literal,.rst-content tt.literal,.rst-content code.literal{color:#E74C3C}.rst-content tt.xref,a .rst-content tt,.rst-content tt.xref,.rst-content code.xref,a .rst-content tt,a .rst-content code{font-weight:bold;color:#404040}.rst-content a tt,.rst-content a tt,.rst-content a code{color:#2980B9}.rst-content dl{margin-bottom:24px}.rst-content dl dt{font-weight:bold}.rst-content dl p,.rst-content dl table,.rst-content dl ul,.rst-content dl ol{margin-bottom:12px !important}.rst-content dl dd{margin:0 0 12px 24px}.rst-content dl:not(.docutils){margin-bottom:24px}.rst-content dl:not(.docutils) dt{display:inline-block;margin:6px 0;font-size:90%;line-height:normal;background:#e7f2fa;color:#2980B9;border-top:solid 3px #6ab0de;padding:6px;position:relative}.rst-content dl:not(.docutils) dt:before{color:#6ab0de}.rst-content dl:not(.docutils) dt .headerlink{color:#404040;font-size:100% !important}.rst-content dl:not(.docutils) dl dt{margin-bottom:6px;border:none;border-left:solid 3px #ccc;background:#f0f0f0;color:#555}.rst-content dl:not(.docutils) dl dt .headerlink{color:#404040;font-size:100% !important}.rst-content dl:not(.docutils) dt:first-child{margin-top:0}.rst-content dl:not(.docutils) tt,.rst-content dl:not(.docutils) tt,.rst-content dl:not(.docutils) code{font-weight:bold}.rst-content dl:not(.docutils) tt.descname,.rst-content dl:not(.docutils) tt.descclassname,.rst-content dl:not(.docutils) tt.descname,.rst-content dl:not(.docutils) code.descname,.rst-content dl:not(.docutils) tt.descclassname,.rst-content dl:not(.docutils) code.descclassname{background-color:transparent;border:none;padding:0;font-size:100% !important}.rst-content dl:not(.docutils) tt.descname,.rst-content dl:not(.docutils) tt.descname,.rst-content dl:not(.docutils) code.descname{font-weight:bold}.rst-content dl:not(.docutils) .optional{display:inline-block;padding:0 4px;color:#000;font-weight:bold}.rst-content dl:not(.docutils) .property{display:inline-block;padding-right:8px}.rst-content .viewcode-link,.rst-content .viewcode-back{display:inline-block;color:#27AE60;font-size:80%;padding-left:24px}.rst-content .viewcode-back{display:block;float:right}.rst-content p.rubric{margin-bottom:12px;font-weight:bold}.rst-content tt.download,.rst-content code.download{background:inherit;padding:inherit;font-family:inherit;font-size:inherit;color:inherit;border:inherit;white-space:inherit}.rst-content tt.download span:first-child:before,.rst-content code.download span:first-child:before{margin-right:4px}@media screen and (max-width: 480px){.rst-content .sidebar{width:100%}}span[id*='MathJax-Span']{color:#404040}.math{text-align:center}@font-face{font-family:"Inconsolata";font-style:normal;font-weight:400;src:local("Inconsolata"),local("Inconsolata-Regular"),url(../fonts/Inconsolata-Regular.ttf) format("truetype")}@font-face{font-family:"Inconsolata";font-style:normal;font-weight:700;src:local("Inconsolata Bold"),local("Inconsolata-Bold"),url(../fonts/Inconsolata-Bold.ttf) format("truetype")}@font-face{font-family:"Lato";font-style:normal;font-weight:400;src:local("Lato Regular"),local("Lato-Regular"),url(../fonts/Lato-Regular.ttf) format("truetype")}@font-face{font-family:"Lato";font-style:normal;font-weight:700;src:local("Lato Bold"),local("Lato-Bold"),url(../fonts/Lato-Bold.ttf) format("truetype")}@font-face{font-family:"Roboto Slab";font-style:normal;font-weight:400;src:local("Roboto Slab Regular"),local("RobotoSlab-Regular"),url(../fonts/RobotoSlab-Regular.ttf) format("truetype")}@font-face{font-family:"Roboto Slab";font-style:normal;font-weight:700;src:local("Roboto Slab Bold"),local("RobotoSlab-Bold"),url(../fonts/RobotoSlab-Bold.ttf) format("truetype")} +/*# sourceMappingURL=theme.css.map */ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/FontAwesome.otf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/FontAwesome.otf new file mode 100644 index 0000000000000000000000000000000000000000..81c9ad949b47f64afeca5642ee2494b6e3147f44 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/FontAwesome.otf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Inconsolata-Bold.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Inconsolata-Bold.ttf new file mode 100644 index 0000000000000000000000000000000000000000..58c9fef3a01c899867e280f49283fbb8e57d631d Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Inconsolata-Bold.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Inconsolata-Regular.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Inconsolata-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..a87ffba6bef48195c8cf4e3ccb42ea77034f7cbc Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Inconsolata-Regular.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Lato-Bold.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Lato-Bold.ttf new file mode 100644 index 0000000000000000000000000000000000000000..74343694e2b2114272f38b1124813b972cb592e5 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Lato-Bold.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Lato-Regular.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Lato-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..04ea8efb1367727b081dea87e63818be0a4d02f0 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/Lato-Regular.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/RobotoSlab-Bold.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/RobotoSlab-Bold.ttf new file mode 100644 index 0000000000000000000000000000000000000000..df5d1df2730433013f41bf2698cbe249b075aa02 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/RobotoSlab-Bold.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/RobotoSlab-Regular.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/RobotoSlab-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..eb52a7907362cc3392eb74892883f5d9e260b638 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/RobotoSlab-Regular.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.eot b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.eot new file mode 100644 index 0000000000000000000000000000000000000000..84677bc0c5f37f1fac9d87548c4554b5c91717cf Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.eot differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.svg b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.svg new file mode 100644 index 0000000000000000000000000000000000000000..d907b25ae60ec7e3d32e4027aa6e6b7595de97af --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.svg @@ -0,0 +1,520 @@ +<?xml version="1.0" standalone="no"?> +<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" > +<svg xmlns="http://www.w3.org/2000/svg"> +<metadata></metadata> +<defs> +<font id="fontawesomeregular" horiz-adv-x="1536" > +<font-face units-per-em="1792" ascent="1536" descent="-256" /> +<missing-glyph horiz-adv-x="448" /> +<glyph unicode=" " horiz-adv-x="448" /> +<glyph unicode="	" horiz-adv-x="448" /> +<glyph unicode=" " horiz-adv-x="448" /> +<glyph unicode="¨" horiz-adv-x="1792" /> +<glyph unicode="©" horiz-adv-x="1792" /> +<glyph unicode="®" horiz-adv-x="1792" /> +<glyph unicode="´" horiz-adv-x="1792" /> +<glyph unicode="Æ" horiz-adv-x="1792" /> +<glyph unicode="Ø" horiz-adv-x="1792" /> +<glyph unicode=" " horiz-adv-x="768" /> +<glyph unicode=" " horiz-adv-x="1537" /> +<glyph unicode=" " horiz-adv-x="768" /> +<glyph unicode=" " horiz-adv-x="1537" /> +<glyph unicode=" " horiz-adv-x="512" /> +<glyph unicode=" " horiz-adv-x="384" /> +<glyph unicode=" " horiz-adv-x="256" /> +<glyph unicode=" " horiz-adv-x="256" /> +<glyph unicode=" " horiz-adv-x="192" /> +<glyph unicode=" " horiz-adv-x="307" /> +<glyph unicode=" " horiz-adv-x="85" /> +<glyph unicode=" " horiz-adv-x="307" /> +<glyph unicode=" " horiz-adv-x="384" /> +<glyph unicode="™" horiz-adv-x="1792" /> +<glyph unicode="∞" horiz-adv-x="1792" /> +<glyph unicode="≠" horiz-adv-x="1792" /> +<glyph unicode="◼" horiz-adv-x="500" d="M0 0z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1699 1350q0 -35 -43 -78l-632 -632v-768h320q26 0 45 -19t19 -45t-19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45t45 19h320v768l-632 632q-43 43 -43 78q0 23 18 36.5t38 17.5t43 4h1408q23 0 43 -4t38 -17.5t18 -36.5z" /> +<glyph unicode="" d="M1536 1312v-1120q0 -50 -34 -89t-86 -60.5t-103.5 -32t-96.5 -10.5t-96.5 10.5t-103.5 32t-86 60.5t-34 89t34 89t86 60.5t103.5 32t96.5 10.5q105 0 192 -39v537l-768 -237v-709q0 -50 -34 -89t-86 -60.5t-103.5 -32t-96.5 -10.5t-96.5 10.5t-103.5 32t-86 60.5t-34 89 t34 89t86 60.5t103.5 32t96.5 10.5q105 0 192 -39v967q0 31 19 56.5t49 35.5l832 256q12 4 28 4q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5zM1664 -128q0 -52 -38 -90t-90 -38q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5 t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1664 32v768q-32 -36 -69 -66q-268 -206 -426 -338q-51 -43 -83 -67t-86.5 -48.5t-102.5 -24.5h-1h-1q-48 0 -102.5 24.5t-86.5 48.5t-83 67q-158 132 -426 338q-37 30 -69 66v-768q0 -13 9.5 -22.5t22.5 -9.5h1472q13 0 22.5 9.5t9.5 22.5zM1664 1083v11v13.5t-0.5 13 t-3 12.5t-5.5 9t-9 7.5t-14 2.5h-1472q-13 0 -22.5 -9.5t-9.5 -22.5q0 -168 147 -284q193 -152 401 -317q6 -5 35 -29.5t46 -37.5t44.5 -31.5t50.5 -27.5t43 -9h1h1q20 0 43 9t50.5 27.5t44.5 31.5t46 37.5t35 29.5q208 165 401 317q54 43 100.5 115.5t46.5 131.5z M1792 1120v-1088q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1472q66 0 113 -47t47 -113z" /> +<glyph unicode="" horiz-adv-x="1792" d="M896 -128q-26 0 -44 18l-624 602q-10 8 -27.5 26t-55.5 65.5t-68 97.5t-53.5 121t-23.5 138q0 220 127 344t351 124q62 0 126.5 -21.5t120 -58t95.5 -68.5t76 -68q36 36 76 68t95.5 68.5t120 58t126.5 21.5q224 0 351 -124t127 -344q0 -221 -229 -450l-623 -600 q-18 -18 -44 -18z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1664 889q0 -22 -26 -48l-363 -354l86 -500q1 -7 1 -20q0 -21 -10.5 -35.5t-30.5 -14.5q-19 0 -40 12l-449 236l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41t49 -41l225 -455 l502 -73q56 -9 56 -46z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1137 532l306 297l-422 62l-189 382l-189 -382l-422 -62l306 -297l-73 -421l378 199l377 -199zM1664 889q0 -22 -26 -48l-363 -354l86 -500q1 -7 1 -20q0 -50 -41 -50q-19 0 -40 12l-449 236l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500 l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41t49 -41l225 -455l502 -73q56 -9 56 -46z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1408 131q0 -120 -73 -189.5t-194 -69.5h-874q-121 0 -194 69.5t-73 189.5q0 53 3.5 103.5t14 109t26.5 108.5t43 97.5t62 81t85.5 53.5t111.5 20q9 0 42 -21.5t74.5 -48t108 -48t133.5 -21.5t133.5 21.5t108 48t74.5 48t42 21.5q61 0 111.5 -20t85.5 -53.5t62 -81 t43 -97.5t26.5 -108.5t14 -109t3.5 -103.5zM1088 1024q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5t271.5 -112.5t112.5 -271.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M384 -64v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM384 320v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM384 704v128q0 26 -19 45t-45 19h-128 q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1408 -64v512q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-512q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM384 1088v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45 t45 -19h128q26 0 45 19t19 45zM1792 -64v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1408 704v512q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-512q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM1792 320v128 q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1792 704v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1792 1088v128q0 26 -19 45t-45 19h-128q-26 0 -45 -19 t-19 -45v-128q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1920 1248v-1344q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1344q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> +<glyph unicode="" horiz-adv-x="1664" d="M768 512v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM768 1280v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM1664 512v-384q0 -52 -38 -90t-90 -38 h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90zM1664 1280v-384q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h512q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="1792" d="M512 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 288v-192q0 -40 -28 -68t-68 -28h-320 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28 h320q40 0 68 -28t28 -68zM1792 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1152 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 800v-192 q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="1792" d="M512 288v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM512 800v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 288v-192q0 -40 -28 -68t-68 -28h-960 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h960q40 0 68 -28t28 -68zM512 1312v-192q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h320q40 0 68 -28t28 -68zM1792 800v-192q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v192q0 40 28 68t68 28 h960q40 0 68 -28t28 -68zM1792 1312v-192q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h960q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1671 970q0 -40 -28 -68l-724 -724l-136 -136q-28 -28 -68 -28t-68 28l-136 136l-362 362q-28 28 -28 68t28 68l136 136q28 28 68 28t68 -28l294 -295l656 657q28 28 68 28t68 -28l136 -136q28 -28 28 -68z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1298 214q0 -40 -28 -68l-136 -136q-28 -28 -68 -28t-68 28l-294 294l-294 -294q-28 -28 -68 -28t-68 28l-136 136q-28 28 -28 68t28 68l294 294l-294 294q-28 28 -28 68t28 68l136 136q28 28 68 28t68 -28l294 -294l294 294q28 28 68 28t68 -28l136 -136q28 -28 28 -68 t-28 -68l-294 -294l294 -294q28 -28 28 -68z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1024 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-224v-224q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v224h-224q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h224v224q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5v-224h224 q13 0 22.5 -9.5t9.5 -22.5zM1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5zM1664 -128q0 -53 -37.5 -90.5t-90.5 -37.5q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5 t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1024 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-576q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h576q13 0 22.5 -9.5t9.5 -22.5zM1152 704q0 185 -131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5t316.5 131.5t131.5 316.5z M1664 -128q0 -53 -37.5 -90.5t-90.5 -37.5q-54 0 -90 38l-343 342q-179 -124 -399 -124q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5t273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -220 -124 -399l343 -343q37 -37 37 -90z " /> +<glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61t-298 61t-245 164t-164 245t-61 298q0 182 80.5 343t226.5 270q43 32 95.5 25t83.5 -50q32 -42 24.5 -94.5t-49.5 -84.5q-98 -74 -151.5 -181t-53.5 -228q0 -104 40.5 -198.5t109.5 -163.5t163.5 -109.5 t198.5 -40.5t198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5q0 121 -53.5 228t-151.5 181q-42 32 -49.5 84.5t24.5 94.5q31 43 84 50t95 -25q146 -109 226.5 -270t80.5 -343zM896 1408v-640q0 -52 -38 -90t-90 -38t-90 38t-38 90v640q0 52 38 90t90 38t90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="1792" d="M256 96v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM640 224v-320q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v320q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1024 480v-576q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23 v576q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1408 864v-960q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v960q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1792 1376v-1472q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v1472q0 14 9 23t23 9h192q14 0 23 -9t9 -23z" /> +<glyph unicode="" d="M1024 640q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1536 749v-222q0 -12 -8 -23t-20 -13l-185 -28q-19 -54 -39 -91q35 -50 107 -138q10 -12 10 -25t-9 -23q-27 -37 -99 -108t-94 -71q-12 0 -26 9l-138 108q-44 -23 -91 -38 q-16 -136 -29 -186q-7 -28 -36 -28h-222q-14 0 -24.5 8.5t-11.5 21.5l-28 184q-49 16 -90 37l-141 -107q-10 -9 -25 -9q-14 0 -25 11q-126 114 -165 168q-7 10 -7 23q0 12 8 23q15 21 51 66.5t54 70.5q-27 50 -41 99l-183 27q-13 2 -21 12.5t-8 23.5v222q0 12 8 23t19 13 l186 28q14 46 39 92q-40 57 -107 138q-10 12 -10 24q0 10 9 23q26 36 98.5 107.5t94.5 71.5q13 0 26 -10l138 -107q44 23 91 38q16 136 29 186q7 28 36 28h222q14 0 24.5 -8.5t11.5 -21.5l28 -184q49 -16 90 -37l142 107q9 9 24 9q13 0 25 -10q129 -119 165 -170q7 -8 7 -22 q0 -12 -8 -23q-15 -21 -51 -66.5t-54 -70.5q26 -50 41 -98l183 -28q13 -2 21 -12.5t8 -23.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M512 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM768 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1024 800v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576 q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1152 76v948h-896v-948q0 -22 7 -40.5t14.5 -27t10.5 -8.5h832q3 0 10.5 8.5t14.5 27t7 40.5zM480 1152h448l-48 117q-7 9 -17 11h-317q-10 -2 -17 -11zM1408 1120v-64q0 -14 -9 -23t-23 -9h-96v-948q0 -83 -47 -143.5t-113 -60.5h-832 q-66 0 -113 58.5t-47 141.5v952h-96q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h309l70 167q15 37 54 63t79 26h320q40 0 79 -26t54 -63l70 -167h309q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1408 544v-480q0 -26 -19 -45t-45 -19h-384v384h-256v-384h-384q-26 0 -45 19t-19 45v480q0 1 0.5 3t0.5 3l575 474l575 -474q1 -2 1 -6zM1631 613l-62 -74q-8 -9 -21 -11h-3q-13 0 -21 7l-692 577l-692 -577q-12 -8 -24 -7q-13 2 -21 11l-62 74q-8 10 -7 23.5t11 21.5 l719 599q32 26 76 26t76 -26l244 -204v195q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-408l219 -182q10 -8 11 -21.5t-7 -23.5z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z " /> +<glyph unicode="" d="M896 992v-448q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h224v352q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1111 540v4l-24 320q-1 13 -11 22.5t-23 9.5h-186q-13 0 -23 -9.5t-11 -22.5l-24 -320v-4q-1 -12 8 -20t21 -8h244q12 0 21 8t8 20zM1870 73q0 -73 -46 -73h-704q13 0 22 9.5t8 22.5l-20 256q-1 13 -11 22.5t-23 9.5h-272q-13 0 -23 -9.5t-11 -22.5l-20 -256 q-1 -13 8 -22.5t22 -9.5h-704q-46 0 -46 73q0 54 26 116l417 1044q8 19 26 33t38 14h339q-13 0 -23 -9.5t-11 -22.5l-15 -192q-1 -14 8 -23t22 -9h166q13 0 22 9t8 23l-15 192q-1 13 -11 22.5t-23 9.5h339q20 0 38 -14t26 -33l417 -1044q26 -62 26 -116z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1280 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 416v-320q0 -40 -28 -68t-68 -28h-1472q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h465l135 -136 q58 -56 136 -56t136 56l136 136h464q40 0 68 -28t28 -68zM1339 985q17 -41 -14 -70l-448 -448q-18 -19 -45 -19t-45 19l-448 448q-31 29 -14 70q17 39 59 39h256v448q0 26 19 45t45 19h256q26 0 45 -19t19 -45v-448h256q42 0 59 -39z" /> +<glyph unicode="" d="M1120 608q0 -12 -10 -24l-319 -319q-11 -9 -23 -9t-23 9l-320 320q-15 16 -7 35q8 20 30 20h192v352q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-352h192q14 0 23 -9t9 -23zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273 t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1118 660q-8 -20 -30 -20h-192v-352q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v352h-192q-14 0 -23 9t-9 23q0 12 10 24l319 319q11 9 23 9t23 -9l320 -320q15 -16 7 -35zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198 t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1023 576h316q-1 3 -2.5 8t-2.5 8l-212 496h-708l-212 -496q-1 -2 -2.5 -8t-2.5 -8h316l95 -192h320zM1536 546v-482q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v482q0 62 25 123l238 552q10 25 36.5 42t52.5 17h832q26 0 52.5 -17t36.5 -42l238 -552 q25 -61 25 -123z" /> +<glyph unicode="" d="M1184 640q0 -37 -32 -55l-544 -320q-15 -9 -32 -9q-16 0 -32 8q-32 19 -32 56v640q0 37 32 56q33 18 64 -1l544 -320q32 -18 32 -55zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1536 1280v-448q0 -26 -19 -45t-45 -19h-448q-42 0 -59 40q-17 39 14 69l138 138q-148 137 -349 137q-104 0 -198.5 -40.5t-163.5 -109.5t-109.5 -163.5t-40.5 -198.5t40.5 -198.5t109.5 -163.5t163.5 -109.5t198.5 -40.5q119 0 225 52t179 147q7 10 23 12q14 0 25 -9 l137 -138q9 -8 9.5 -20.5t-7.5 -22.5q-109 -132 -264 -204.5t-327 -72.5q-156 0 -298 61t-245 164t-164 245t-61 298t61 298t164 245t245 164t298 61q147 0 284.5 -55.5t244.5 -156.5l130 129q29 31 70 14q39 -17 39 -59z" /> +<glyph unicode="" d="M1511 480q0 -5 -1 -7q-64 -268 -268 -434.5t-478 -166.5q-146 0 -282.5 55t-243.5 157l-129 -129q-19 -19 -45 -19t-45 19t-19 45v448q0 26 19 45t45 19h448q26 0 45 -19t19 -45t-19 -45l-137 -137q71 -66 161 -102t187 -36q134 0 250 65t186 179q11 17 53 117 q8 23 30 23h192q13 0 22.5 -9.5t9.5 -22.5zM1536 1280v-448q0 -26 -19 -45t-45 -19h-448q-26 0 -45 19t-19 45t19 45l138 138q-148 137 -349 137q-134 0 -250 -65t-186 -179q-11 -17 -53 -117q-8 -23 -30 -23h-199q-13 0 -22.5 9.5t-9.5 22.5v7q65 268 270 434.5t480 166.5 q146 0 284 -55.5t245 -156.5l130 129q19 19 45 19t45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M384 352v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 608v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M384 864v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1536 352v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5t9.5 -22.5z M1536 608v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5t9.5 -22.5zM1536 864v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h960q13 0 22.5 -9.5 t9.5 -22.5zM1664 160v832q0 13 -9.5 22.5t-22.5 9.5h-1472q-13 0 -22.5 -9.5t-9.5 -22.5v-832q0 -13 9.5 -22.5t22.5 -9.5h1472q13 0 22.5 9.5t9.5 22.5zM1792 1248v-1088q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1472q66 0 113 -47 t47 -113z" /> +<glyph unicode="" horiz-adv-x="1152" d="M320 768h512v192q0 106 -75 181t-181 75t-181 -75t-75 -181v-192zM1152 672v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h32v192q0 184 132 316t316 132t316 -132t132 -316v-192h32q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="1792" d="M320 1280q0 -72 -64 -110v-1266q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v1266q-64 38 -64 110q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -25 -12.5 -38.5t-39.5 -27.5q-215 -116 -369 -116q-61 0 -123.5 22t-108.5 48 t-115.5 48t-142.5 22q-192 0 -464 -146q-17 -9 -33 -9q-26 0 -45 19t-19 45v742q0 32 31 55q21 14 79 43q236 120 421 120q107 0 200 -29t219 -88q38 -19 88 -19q54 0 117.5 21t110 47t88 47t54.5 21q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1664 650q0 -166 -60 -314l-20 -49l-185 -33q-22 -83 -90.5 -136.5t-156.5 -53.5v-32q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v576q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-32q71 0 130 -35.5t93 -95.5l68 12q29 95 29 193q0 148 -88 279t-236.5 209t-315.5 78 t-315.5 -78t-236.5 -209t-88 -279q0 -98 29 -193l68 -12q34 60 93 95.5t130 35.5v32q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-576q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v32q-88 0 -156.5 53.5t-90.5 136.5l-185 33l-20 49q-60 148 -60 314q0 151 67 291t179 242.5 t266 163.5t320 61t320 -61t266 -163.5t179 -242.5t67 -291z" /> +<glyph unicode="" horiz-adv-x="768" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1152" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45zM1152 640q0 -76 -42.5 -141.5t-112.5 -93.5q-10 -5 -25 -5q-26 0 -45 18.5t-19 45.5q0 21 12 35.5t29 25t34 23t29 35.5 t12 57t-12 57t-29 35.5t-34 23t-29 25t-12 35.5q0 27 19 45.5t45 18.5q15 0 25 -5q70 -27 112.5 -93t42.5 -142z" /> +<glyph unicode="" horiz-adv-x="1664" d="M768 1184v-1088q0 -26 -19 -45t-45 -19t-45 19l-333 333h-262q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h262l333 333q19 19 45 19t45 -19t19 -45zM1152 640q0 -76 -42.5 -141.5t-112.5 -93.5q-10 -5 -25 -5q-26 0 -45 18.5t-19 45.5q0 21 12 35.5t29 25t34 23t29 35.5 t12 57t-12 57t-29 35.5t-34 23t-29 25t-12 35.5q0 27 19 45.5t45 18.5q15 0 25 -5q70 -27 112.5 -93t42.5 -142zM1408 640q0 -153 -85 -282.5t-225 -188.5q-13 -5 -25 -5q-27 0 -46 19t-19 45q0 39 39 59q56 29 76 44q74 54 115.5 135.5t41.5 173.5t-41.5 173.5 t-115.5 135.5q-20 15 -76 44q-39 20 -39 59q0 26 19 45t45 19q13 0 26 -5q140 -59 225 -188.5t85 -282.5zM1664 640q0 -230 -127 -422.5t-338 -283.5q-13 -5 -26 -5q-26 0 -45 19t-19 45q0 36 39 59q7 4 22.5 10.5t22.5 10.5q46 25 82 51q123 91 192 227t69 289t-69 289 t-192 227q-36 26 -82 51q-7 4 -22.5 10.5t-22.5 10.5q-39 23 -39 59q0 26 19 45t45 19q13 0 26 -5q211 -91 338 -283.5t127 -422.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M384 384v-128h-128v128h128zM384 1152v-128h-128v128h128zM1152 1152v-128h-128v128h128zM128 129h384v383h-384v-383zM128 896h384v384h-384v-384zM896 896h384v384h-384v-384zM640 640v-640h-640v640h640zM1152 128v-128h-128v128h128zM1408 128v-128h-128v128h128z M1408 640v-384h-384v128h-128v-384h-128v640h384v-128h128v128h128zM640 1408v-640h-640v640h640zM1408 1408v-640h-640v640h640z" /> +<glyph unicode="" horiz-adv-x="1792" d="M63 0h-63v1408h63v-1408zM126 1h-32v1407h32v-1407zM220 1h-31v1407h31v-1407zM377 1h-31v1407h31v-1407zM534 1h-62v1407h62v-1407zM660 1h-31v1407h31v-1407zM723 1h-31v1407h31v-1407zM786 1h-31v1407h31v-1407zM943 1h-63v1407h63v-1407zM1100 1h-63v1407h63v-1407z M1226 1h-63v1407h63v-1407zM1352 1h-63v1407h63v-1407zM1446 1h-63v1407h63v-1407zM1635 1h-94v1407h94v-1407zM1698 1h-32v1407h32v-1407zM1792 0h-63v1408h63v-1408z" /> +<glyph unicode="" d="M448 1088q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1515 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-53 0 -90 37l-715 716q-38 37 -64.5 101t-26.5 117v416q0 52 38 90t90 38h416q53 0 117 -26.5t102 -64.5 l715 -714q37 -39 37 -91z" /> +<glyph unicode="" horiz-adv-x="1920" d="M448 1088q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1515 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-53 0 -90 37l-715 716q-38 37 -64.5 101t-26.5 117v416q0 52 38 90t90 38h416q53 0 117 -26.5t102 -64.5 l715 -714q37 -39 37 -91zM1899 512q0 -53 -37 -90l-491 -492q-39 -37 -91 -37q-36 0 -59 14t-53 45l470 470q37 37 37 90q0 52 -37 91l-715 714q-38 38 -102 64.5t-117 26.5h224q53 0 117 -26.5t102 -64.5l715 -714q37 -39 37 -91z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1639 1058q40 -57 18 -129l-275 -906q-19 -64 -76.5 -107.5t-122.5 -43.5h-923q-77 0 -148.5 53.5t-99.5 131.5q-24 67 -2 127q0 4 3 27t4 37q1 8 -3 21.5t-3 19.5q2 11 8 21t16.5 23.5t16.5 23.5q23 38 45 91.5t30 91.5q3 10 0.5 30t-0.5 28q3 11 17 28t17 23 q21 36 42 92t25 90q1 9 -2.5 32t0.5 28q4 13 22 30.5t22 22.5q19 26 42.5 84.5t27.5 96.5q1 8 -3 25.5t-2 26.5q2 8 9 18t18 23t17 21q8 12 16.5 30.5t15 35t16 36t19.5 32t26.5 23.5t36 11.5t47.5 -5.5l-1 -3q38 9 51 9h761q74 0 114 -56t18 -130l-274 -906 q-36 -119 -71.5 -153.5t-128.5 -34.5h-869q-27 0 -38 -15q-11 -16 -1 -43q24 -70 144 -70h923q29 0 56 15.5t35 41.5l300 987q7 22 5 57q38 -15 59 -43zM575 1056q-4 -13 2 -22.5t20 -9.5h608q13 0 25.5 9.5t16.5 22.5l21 64q4 13 -2 22.5t-20 9.5h-608q-13 0 -25.5 -9.5 t-16.5 -22.5zM492 800q-4 -13 2 -22.5t20 -9.5h608q13 0 25.5 9.5t16.5 22.5l21 64q4 13 -2 22.5t-20 9.5h-608q-13 0 -25.5 -9.5t-16.5 -22.5z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1164 1408q23 0 44 -9q33 -13 52.5 -41t19.5 -62v-1289q0 -34 -19.5 -62t-52.5 -41q-19 -8 -44 -8q-48 0 -83 32l-441 424l-441 -424q-36 -33 -83 -33q-23 0 -44 9q-33 13 -52.5 41t-19.5 62v1289q0 34 19.5 62t52.5 41q21 9 44 9h1048z" /> +<glyph unicode="" horiz-adv-x="1664" d="M384 0h896v256h-896v-256zM384 640h896v384h-160q-40 0 -68 28t-28 68v160h-640v-640zM1536 576q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 576v-416q0 -13 -9.5 -22.5t-22.5 -9.5h-224v-160q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68 v160h-224q-13 0 -22.5 9.5t-9.5 22.5v416q0 79 56.5 135.5t135.5 56.5h64v544q0 40 28 68t68 28h672q40 0 88 -20t76 -48l152 -152q28 -28 48 -76t20 -88v-256h64q79 0 135.5 -56.5t56.5 -135.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M960 864q119 0 203.5 -84.5t84.5 -203.5t-84.5 -203.5t-203.5 -84.5t-203.5 84.5t-84.5 203.5t84.5 203.5t203.5 84.5zM1664 1280q106 0 181 -75t75 -181v-896q0 -106 -75 -181t-181 -75h-1408q-106 0 -181 75t-75 181v896q0 106 75 181t181 75h224l51 136 q19 49 69.5 84.5t103.5 35.5h512q53 0 103.5 -35.5t69.5 -84.5l51 -136h224zM960 128q185 0 316.5 131.5t131.5 316.5t-131.5 316.5t-316.5 131.5t-316.5 -131.5t-131.5 -316.5t131.5 -316.5t316.5 -131.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M725 977l-170 -450q33 0 136.5 -2t160.5 -2q19 0 57 2q-87 253 -184 452zM0 -128l2 79q23 7 56 12.5t57 10.5t49.5 14.5t44.5 29t31 50.5l237 616l280 724h75h53q8 -14 11 -21l205 -480q33 -78 106 -257.5t114 -274.5q15 -34 58 -144.5t72 -168.5q20 -45 35 -57 q19 -15 88 -29.5t84 -20.5q6 -38 6 -57q0 -4 -0.5 -13t-0.5 -13q-63 0 -190 8t-191 8q-76 0 -215 -7t-178 -8q0 43 4 78l131 28q1 0 12.5 2.5t15.5 3.5t14.5 4.5t15 6.5t11 8t9 11t2.5 14q0 16 -31 96.5t-72 177.5t-42 100l-450 2q-26 -58 -76.5 -195.5t-50.5 -162.5 q0 -22 14 -37.5t43.5 -24.5t48.5 -13.5t57 -8.5t41 -4q1 -19 1 -58q0 -9 -2 -27q-58 0 -174.5 10t-174.5 10q-8 0 -26.5 -4t-21.5 -4q-80 -14 -188 -14z" /> +<glyph unicode="" horiz-adv-x="1408" d="M555 15q74 -32 140 -32q376 0 376 335q0 114 -41 180q-27 44 -61.5 74t-67.5 46.5t-80.5 25t-84 10.5t-94.5 2q-73 0 -101 -10q0 -53 -0.5 -159t-0.5 -158q0 -8 -1 -67.5t-0.5 -96.5t4.5 -83.5t12 -66.5zM541 761q42 -7 109 -7q82 0 143 13t110 44.5t74.5 89.5t25.5 142 q0 70 -29 122.5t-79 82t-108 43.5t-124 14q-50 0 -130 -13q0 -50 4 -151t4 -152q0 -27 -0.5 -80t-0.5 -79q0 -46 1 -69zM0 -128l2 94q15 4 85 16t106 27q7 12 12.5 27t8.5 33.5t5.5 32.5t3 37.5t0.5 34v35.5v30q0 982 -22 1025q-4 8 -22 14.5t-44.5 11t-49.5 7t-48.5 4.5 t-30.5 3l-4 83q98 2 340 11.5t373 9.5q23 0 68.5 -0.5t67.5 -0.5q70 0 136.5 -13t128.5 -42t108 -71t74 -104.5t28 -137.5q0 -52 -16.5 -95.5t-39 -72t-64.5 -57.5t-73 -45t-84 -40q154 -35 256.5 -134t102.5 -248q0 -100 -35 -179.5t-93.5 -130.5t-138 -85.5t-163.5 -48.5 t-176 -14q-44 0 -132 3t-132 3q-106 0 -307 -11t-231 -12z" /> +<glyph unicode="" horiz-adv-x="1024" d="M0 -126l17 85q6 2 81.5 21.5t111.5 37.5q28 35 41 101q1 7 62 289t114 543.5t52 296.5v25q-24 13 -54.5 18.5t-69.5 8t-58 5.5l19 103q33 -2 120 -6.5t149.5 -7t120.5 -2.5q48 0 98.5 2.5t121 7t98.5 6.5q-5 -39 -19 -89q-30 -10 -101.5 -28.5t-108.5 -33.5 q-8 -19 -14 -42.5t-9 -40t-7.5 -45.5t-6.5 -42q-27 -148 -87.5 -419.5t-77.5 -355.5q-2 -9 -13 -58t-20 -90t-16 -83.5t-6 -57.5l1 -18q17 -4 185 -31q-3 -44 -16 -99q-11 0 -32.5 -1.5t-32.5 -1.5q-29 0 -87 10t-86 10q-138 2 -206 2q-51 0 -143 -9t-121 -11z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1744 128q33 0 42 -18.5t-11 -44.5l-126 -162q-20 -26 -49 -26t-49 26l-126 162q-20 26 -11 44.5t42 18.5h80v1024h-80q-33 0 -42 18.5t11 44.5l126 162q20 26 49 26t49 -26l126 -162q20 -26 11 -44.5t-42 -18.5h-80v-1024h80zM81 1407l54 -27q12 -5 211 -5q44 0 132 2 t132 2q36 0 107.5 -0.5t107.5 -0.5h293q6 0 21 -0.5t20.5 0t16 3t17.5 9t15 17.5l42 1q4 0 14 -0.5t14 -0.5q2 -112 2 -336q0 -80 -5 -109q-39 -14 -68 -18q-25 44 -54 128q-3 9 -11 48t-14.5 73.5t-7.5 35.5q-6 8 -12 12.5t-15.5 6t-13 2.5t-18 0.5t-16.5 -0.5 q-17 0 -66.5 0.5t-74.5 0.5t-64 -2t-71 -6q-9 -81 -8 -136q0 -94 2 -388t2 -455q0 -16 -2.5 -71.5t0 -91.5t12.5 -69q40 -21 124 -42.5t120 -37.5q5 -40 5 -50q0 -14 -3 -29l-34 -1q-76 -2 -218 8t-207 10q-50 0 -151 -9t-152 -9q-3 51 -3 52v9q17 27 61.5 43t98.5 29t78 27 q19 42 19 383q0 101 -3 303t-3 303v117q0 2 0.5 15.5t0.5 25t-1 25.5t-3 24t-5 14q-11 12 -162 12q-33 0 -93 -12t-80 -26q-19 -13 -34 -72.5t-31.5 -111t-42.5 -53.5q-42 26 -56 44v383z" /> +<glyph unicode="" d="M81 1407l54 -27q12 -5 211 -5q44 0 132 2t132 2q70 0 246.5 1t304.5 0.5t247 -4.5q33 -1 56 31l42 1q4 0 14 -0.5t14 -0.5q2 -112 2 -336q0 -80 -5 -109q-39 -14 -68 -18q-25 44 -54 128q-3 9 -11 47.5t-15 73.5t-7 36q-10 13 -27 19q-5 2 -66 2q-30 0 -93 1t-103 1 t-94 -2t-96 -7q-9 -81 -8 -136l1 -152v52q0 -55 1 -154t1.5 -180t0.5 -153q0 -16 -2.5 -71.5t0 -91.5t12.5 -69q40 -21 124 -42.5t120 -37.5q5 -40 5 -50q0 -14 -3 -29l-34 -1q-76 -2 -218 8t-207 10q-50 0 -151 -9t-152 -9q-3 51 -3 52v9q17 27 61.5 43t98.5 29t78 27 q7 16 11.5 74t6 145.5t1.5 155t-0.5 153.5t-0.5 89q0 7 -2.5 21.5t-2.5 22.5q0 7 0.5 44t1 73t0 76.5t-3 67.5t-6.5 32q-11 12 -162 12q-41 0 -163 -13.5t-138 -24.5q-19 -12 -34 -71.5t-31.5 -111.5t-42.5 -54q-42 26 -56 44v383zM1310 125q12 0 42 -19.5t57.5 -41.5 t59.5 -49t36 -30q26 -21 26 -49t-26 -49q-4 -3 -36 -30t-59.5 -49t-57.5 -41.5t-42 -19.5q-13 0 -20.5 10.5t-10 28.5t-2.5 33.5t1.5 33t1.5 19.5h-1024q0 -2 1.5 -19.5t1.5 -33t-2.5 -33.5t-10 -28.5t-20.5 -10.5q-12 0 -42 19.5t-57.5 41.5t-59.5 49t-36 30q-26 21 -26 49 t26 49q4 3 36 30t59.5 49t57.5 41.5t42 19.5q13 0 20.5 -10.5t10 -28.5t2.5 -33.5t-1.5 -33t-1.5 -19.5h1024q0 2 -1.5 19.5t-1.5 33t2.5 33.5t10 28.5t20.5 10.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1408 576v-128q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1280q26 0 45 -19t19 -45zM1664 960v-128q0 -26 -19 -45 t-45 -19h-1536q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1536q26 0 45 -19t19 -45zM1280 1344v-128q0 -26 -19 -45t-45 -19h-1152q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1408 576v-128q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h896q26 0 45 -19t19 -45zM1664 960v-128q0 -26 -19 -45t-45 -19 h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1280 1344v-128q0 -26 -19 -45t-45 -19h-640q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h640q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 576v-128q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1280q26 0 45 -19t19 -45zM1792 960v-128q0 -26 -19 -45 t-45 -19h-1536q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1536q26 0 45 -19t19 -45zM1792 1344v-128q0 -26 -19 -45t-45 -19h-1152q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 192v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 576v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 960v-128q0 -26 -19 -45 t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 1344v-128q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1664q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M256 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM256 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5 t9.5 -22.5zM256 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1344 q13 0 22.5 -9.5t9.5 -22.5zM256 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-192q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h192q13 0 22.5 -9.5t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5 t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v192 q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M384 992v-576q0 -13 -9.5 -22.5t-22.5 -9.5q-14 0 -23 9l-288 288q-9 9 -9 23t9 23l288 288q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5 t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088 q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5t9.5 -22.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M352 704q0 -14 -9 -23l-288 -288q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v576q0 13 9.5 22.5t22.5 9.5q14 0 23 -9l288 -288q9 -9 9 -23zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5 t9.5 -22.5zM1792 608v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088q13 0 22.5 -9.5t9.5 -22.5zM1792 992v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1088q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1088 q13 0 22.5 -9.5t9.5 -22.5zM1792 1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1728q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1728q13 0 22.5 -9.5t9.5 -22.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 1184v-1088q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-403 403v-166q0 -119 -84.5 -203.5t-203.5 -84.5h-704q-119 0 -203.5 84.5t-84.5 203.5v704q0 119 84.5 203.5t203.5 84.5h704q119 0 203.5 -84.5t84.5 -203.5v-165l403 402q18 19 45 19q12 0 25 -5 q39 -17 39 -59z" /> +<glyph unicode="" horiz-adv-x="1920" d="M640 960q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1664 576v-448h-1408v192l320 320l160 -160l512 512zM1760 1280h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-1216q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5v1216 q0 13 -9.5 22.5t-22.5 9.5zM1920 1248v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> +<glyph unicode="" d="M363 0l91 91l-235 235l-91 -91v-107h128v-128h107zM886 928q0 22 -22 22q-10 0 -17 -7l-542 -542q-7 -7 -7 -17q0 -22 22 -22q10 0 17 7l542 542q7 7 7 17zM832 1120l416 -416l-832 -832h-416v416zM1515 1024q0 -53 -37 -90l-166 -166l-416 416l166 165q36 38 90 38 q53 0 91 -38l235 -234q37 -39 37 -91z" /> +<glyph unicode="" horiz-adv-x="1024" d="M768 896q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1024 896q0 -109 -33 -179l-364 -774q-16 -33 -47.5 -52t-67.5 -19t-67.5 19t-46.5 52l-365 774q-33 70 -33 179q0 212 150 362t362 150t362 -150t150 -362z" /> +<glyph unicode="" d="M768 96v1088q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M512 384q0 36 -20 69q-1 1 -15.5 22.5t-25.5 38t-25 44t-21 50.5q-4 16 -21 16t-21 -16q-7 -23 -21 -50.5t-25 -44t-25.5 -38t-15.5 -22.5q-20 -33 -20 -69q0 -53 37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1024 512q0 -212 -150 -362t-362 -150t-362 150t-150 362 q0 145 81 275q6 9 62.5 90.5t101 151t99.5 178t83 201.5q9 30 34 47t51 17t51.5 -17t33.5 -47q28 -93 83 -201.5t99.5 -178t101 -151t62.5 -90.5q81 -127 81 -275z" /> +<glyph unicode="" horiz-adv-x="1792" d="M888 352l116 116l-152 152l-116 -116v-56h96v-96h56zM1328 1072q-16 16 -33 -1l-350 -350q-17 -17 -1 -33t33 1l350 350q17 17 1 33zM1408 478v-190q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832 q63 0 117 -25q15 -7 18 -23q3 -17 -9 -29l-49 -49q-14 -14 -32 -8q-23 6 -45 6h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v126q0 13 9 22l64 64q15 15 35 7t20 -29zM1312 1216l288 -288l-672 -672h-288v288zM1756 1084l-92 -92 l-288 288l92 92q28 28 68 28t68 -28l152 -152q28 -28 28 -68t-28 -68z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1408 547v-259q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h255v0q13 0 22.5 -9.5t9.5 -22.5q0 -27 -26 -32q-77 -26 -133 -60q-10 -4 -16 -4h-112q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832 q66 0 113 47t47 113v214q0 19 18 29q28 13 54 37q16 16 35 8q21 -9 21 -29zM1645 1043l-384 -384q-18 -19 -45 -19q-12 0 -25 5q-39 17 -39 59v192h-160q-323 0 -438 -131q-119 -137 -74 -473q3 -23 -20 -34q-8 -2 -12 -2q-16 0 -26 13q-10 14 -21 31t-39.5 68.5t-49.5 99.5 t-38.5 114t-17.5 122q0 49 3.5 91t14 90t28 88t47 81.5t68.5 74t94.5 61.5t124.5 48.5t159.5 30.5t196.5 11h160v192q0 42 39 59q13 5 25 5q26 0 45 -19l384 -384q19 -19 19 -45t-19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1408 606v-318q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q63 0 117 -25q15 -7 18 -23q3 -17 -9 -29l-49 -49q-10 -10 -23 -10q-3 0 -9 2q-23 6 -45 6h-832q-66 0 -113 -47t-47 -113v-832 q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v254q0 13 9 22l64 64q10 10 23 10q6 0 12 -3q20 -8 20 -29zM1639 1095l-814 -814q-24 -24 -57 -24t-57 24l-430 430q-24 24 -24 57t24 57l110 110q24 24 57 24t57 -24l263 -263l647 647q24 24 57 24t57 -24l110 -110 q24 -24 24 -57t-24 -57z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -26 -19 -45l-256 -256q-19 -19 -45 -19t-45 19t-19 45v128h-384v-384h128q26 0 45 -19t19 -45t-19 -45l-256 -256q-19 -19 -45 -19t-45 19l-256 256q-19 19 -19 45t19 45t45 19h128v384h-384v-128q0 -26 -19 -45t-45 -19t-45 19l-256 256q-19 19 -19 45 t19 45l256 256q19 19 45 19t45 -19t19 -45v-128h384v384h-128q-26 0 -45 19t-19 45t19 45l256 256q19 19 45 19t45 -19l256 -256q19 -19 19 -45t-19 -45t-45 -19h-128v-384h384v128q0 26 19 45t45 19t45 -19l256 -256q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="1024" d="M979 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-678q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-678q4 11 13 19z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1747 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-710q0 -26 -13 -32t-32 13l-710 710q-9 9 -13 19v-678q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-678q4 11 13 19l710 710 q19 19 32 13t13 -32v-710q4 11 13 19z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1619 1395q19 19 32 13t13 -32v-1472q0 -26 -13 -32t-32 13l-710 710q-8 9 -13 19v-710q0 -26 -13 -32t-32 13l-710 710q-19 19 -19 45t19 45l710 710q19 19 32 13t13 -32v-710q5 11 13 19z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1384 609l-1328 -738q-23 -13 -39.5 -3t-16.5 36v1472q0 26 16.5 36t39.5 -3l1328 -738q23 -13 23 -31t-23 -31z" /> +<glyph unicode="" d="M1536 1344v-1408q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h512q26 0 45 -19t19 -45zM640 1344v-1408q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h512q26 0 45 -19t19 -45z" /> +<glyph unicode="" d="M1536 1344v-1408q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v710q0 26 13 32t32 -13l710 -710q19 -19 19 -45t-19 -45l-710 -710q-19 -19 -32 -13t-13 32v710q-5 -10 -13 -19z" /> +<glyph unicode="" horiz-adv-x="1792" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v710q0 26 13 32t32 -13l710 -710q8 -8 13 -19v678q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-1408q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v678q-5 -10 -13 -19l-710 -710 q-19 -19 -32 -13t-13 32v710q-5 -10 -13 -19z" /> +<glyph unicode="" horiz-adv-x="1024" d="M45 -115q-19 -19 -32 -13t-13 32v1472q0 26 13 32t32 -13l710 -710q8 -8 13 -19v678q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-1408q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v678q-5 -10 -13 -19z" /> +<glyph unicode="" horiz-adv-x="1538" d="M14 557l710 710q19 19 45 19t45 -19l710 -710q19 -19 13 -32t-32 -13h-1472q-26 0 -32 13t13 32zM1473 0h-1408q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1408q26 0 45 -19t19 -45v-256q0 -26 -19 -45t-45 -19z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1171 1235l-531 -531l531 -531q19 -19 19 -45t-19 -45l-166 -166q-19 -19 -45 -19t-45 19l-742 742q-19 19 -19 45t19 45l742 742q19 19 45 19t45 -19l166 -166q19 -19 19 -45t-19 -45z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1107 659l-742 -742q-19 -19 -45 -19t-45 19l-166 166q-19 19 -19 45t19 45l531 531l-531 531q-19 19 -19 45t19 45l166 166q19 19 45 19t45 -19l742 -742q19 -19 19 -45t-19 -45z" /> +<glyph unicode="" d="M1216 576v128q0 26 -19 45t-45 19h-256v256q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-256h-256q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h256v-256q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v256h256q26 0 45 19t19 45zM1536 640q0 -209 -103 -385.5 t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1216 576v128q0 26 -19 45t-45 19h-768q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h768q26 0 45 19t19 45zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5 t103 -385.5z" /> +<glyph unicode="" d="M1149 414q0 26 -19 45l-181 181l181 181q19 19 19 45q0 27 -19 46l-90 90q-19 19 -46 19q-26 0 -45 -19l-181 -181l-181 181q-19 19 -45 19q-27 0 -46 -19l-90 -90q-19 -19 -19 -46q0 -26 19 -45l181 -181l-181 -181q-19 -19 -19 -45q0 -27 19 -46l90 -90q19 -19 46 -19 q26 0 45 19l181 181l181 -181q19 -19 45 -19q27 0 46 19l90 90q19 19 19 46zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1284 802q0 28 -18 46l-91 90q-19 19 -45 19t-45 -19l-408 -407l-226 226q-19 19 -45 19t-45 -19l-91 -90q-18 -18 -18 -46q0 -27 18 -45l362 -362q19 -19 45 -19q27 0 46 19l543 543q18 18 18 45zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M896 160v192q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h192q14 0 23 9t9 23zM1152 832q0 88 -55.5 163t-138.5 116t-170 41q-243 0 -371 -213q-15 -24 8 -42l132 -100q7 -6 19 -6q16 0 25 12q53 68 86 92q34 24 86 24q48 0 85.5 -26t37.5 -59 q0 -38 -20 -61t-68 -45q-63 -28 -115.5 -86.5t-52.5 -125.5v-36q0 -14 9 -23t23 -9h192q14 0 23 9t9 23q0 19 21.5 49.5t54.5 49.5q32 18 49 28.5t46 35t44.5 48t28 60.5t12.5 81zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1024 160v160q0 14 -9 23t-23 9h-96v512q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23t23 -9h96v-320h-96q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23t23 -9h448q14 0 23 9t9 23zM896 1056v160q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-160q0 -14 9 -23 t23 -9h192q14 0 23 9t9 23zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1197 512h-109q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h109q-32 108 -112.5 188.5t-188.5 112.5v-109q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v109q-108 -32 -188.5 -112.5t-112.5 -188.5h109q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-109 q32 -108 112.5 -188.5t188.5 -112.5v109q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-109q108 32 188.5 112.5t112.5 188.5zM1536 704v-128q0 -26 -19 -45t-45 -19h-143q-37 -161 -154.5 -278.5t-278.5 -154.5v-143q0 -26 -19 -45t-45 -19h-128q-26 0 -45 19t-19 45v143 q-161 37 -278.5 154.5t-154.5 278.5h-143q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h143q37 161 154.5 278.5t278.5 154.5v143q0 26 19 45t45 19h128q26 0 45 -19t19 -45v-143q161 -37 278.5 -154.5t154.5 -278.5h143q26 0 45 -19t19 -45z" /> +<glyph unicode="" d="M1097 457l-146 -146q-10 -10 -23 -10t-23 10l-137 137l-137 -137q-10 -10 -23 -10t-23 10l-146 146q-10 10 -10 23t10 23l137 137l-137 137q-10 10 -10 23t10 23l146 146q10 10 23 10t23 -10l137 -137l137 137q10 10 23 10t23 -10l146 -146q10 -10 10 -23t-10 -23 l-137 -137l137 -137q10 -10 10 -23t-10 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5 t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1171 723l-422 -422q-19 -19 -45 -19t-45 19l-294 294q-19 19 -19 45t19 45l102 102q19 19 45 19t45 -19l147 -147l275 275q19 19 45 19t45 -19l102 -102q19 -19 19 -45t-19 -45zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198 t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1312 643q0 161 -87 295l-754 -753q137 -89 297 -89q111 0 211.5 43.5t173.5 116.5t116 174.5t43 212.5zM313 344l755 754q-135 91 -300 91q-148 0 -273 -73t-198 -199t-73 -274q0 -162 89 -299zM1536 643q0 -157 -61 -300t-163.5 -246t-245 -164t-298.5 -61t-298.5 61 t-245 164t-163.5 246t-61 300t61 299.5t163.5 245.5t245 164t298.5 61t298.5 -61t245 -164t163.5 -245.5t61 -299.5z" /> +<glyph unicode="" d="M1536 640v-128q0 -53 -32.5 -90.5t-84.5 -37.5h-704l293 -294q38 -36 38 -90t-38 -90l-75 -76q-37 -37 -90 -37q-52 0 -91 37l-651 652q-37 37 -37 90q0 52 37 91l651 650q38 38 91 38q52 0 90 -38l75 -74q38 -38 38 -91t-38 -91l-293 -293h704q52 0 84.5 -37.5 t32.5 -90.5z" /> +<glyph unicode="" d="M1472 576q0 -54 -37 -91l-651 -651q-39 -37 -91 -37q-51 0 -90 37l-75 75q-38 38 -38 91t38 91l293 293h-704q-52 0 -84.5 37.5t-32.5 90.5v128q0 53 32.5 90.5t84.5 37.5h704l-293 294q-38 36 -38 90t38 90l75 75q38 38 90 38q53 0 91 -38l651 -651q37 -35 37 -90z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1611 565q0 -51 -37 -90l-75 -75q-38 -38 -91 -38q-54 0 -90 38l-294 293v-704q0 -52 -37.5 -84.5t-90.5 -32.5h-128q-53 0 -90.5 32.5t-37.5 84.5v704l-294 -293q-36 -38 -90 -38t-90 38l-75 75q-38 38 -38 90q0 53 38 91l651 651q35 37 90 37q54 0 91 -37l651 -651 q37 -39 37 -91z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1611 704q0 -53 -37 -90l-651 -652q-39 -37 -91 -37q-53 0 -90 37l-651 652q-38 36 -38 90q0 53 38 91l74 75q39 37 91 37q53 0 90 -37l294 -294v704q0 52 38 90t90 38h128q52 0 90 -38t38 -90v-704l294 294q37 37 90 37q52 0 91 -37l75 -75q37 -39 37 -91z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 896q0 -26 -19 -45l-512 -512q-19 -19 -45 -19t-45 19t-19 45v256h-224q-98 0 -175.5 -6t-154 -21.5t-133 -42.5t-105.5 -69.5t-80 -101t-48.5 -138.5t-17.5 -181q0 -55 5 -123q0 -6 2.5 -23.5t2.5 -26.5q0 -15 -8.5 -25t-23.5 -10q-16 0 -28 17q-7 9 -13 22 t-13.5 30t-10.5 24q-127 285 -127 451q0 199 53 333q162 403 875 403h224v256q0 26 19 45t45 19t45 -19l512 -512q19 -19 19 -45z" /> +<glyph unicode="" d="M755 480q0 -13 -10 -23l-332 -332l144 -144q19 -19 19 -45t-19 -45t-45 -19h-448q-26 0 -45 19t-19 45v448q0 26 19 45t45 19t45 -19l144 -144l332 332q10 10 23 10t23 -10l114 -114q10 -10 10 -23zM1536 1344v-448q0 -26 -19 -45t-45 -19t-45 19l-144 144l-332 -332 q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l332 332l-144 144q-19 19 -19 45t19 45t45 19h448q26 0 45 -19t19 -45z" /> +<glyph unicode="" d="M768 576v-448q0 -26 -19 -45t-45 -19t-45 19l-144 144l-332 -332q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l332 332l-144 144q-19 19 -19 45t19 45t45 19h448q26 0 45 -19t19 -45zM1523 1248q0 -13 -10 -23l-332 -332l144 -144q19 -19 19 -45t-19 -45 t-45 -19h-448q-26 0 -45 19t-19 45v448q0 26 19 45t45 19t45 -19l144 -144l332 332q10 10 23 10t23 -10l114 -114q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1408 800v-192q0 -40 -28 -68t-68 -28h-416v-416q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v416h-416q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h416v416q0 40 28 68t68 28h192q40 0 68 -28t28 -68v-416h416q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1408 800v-192q0 -40 -28 -68t-68 -28h-1216q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h1216q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1482 486q46 -26 59.5 -77.5t-12.5 -97.5l-64 -110q-26 -46 -77.5 -59.5t-97.5 12.5l-266 153v-307q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v307l-266 -153q-46 -26 -97.5 -12.5t-77.5 59.5l-64 110q-26 46 -12.5 97.5t59.5 77.5l266 154l-266 154 q-46 26 -59.5 77.5t12.5 97.5l64 110q26 46 77.5 59.5t97.5 -12.5l266 -153v307q0 52 38 90t90 38h128q52 0 90 -38t38 -90v-307l266 153q46 26 97.5 12.5t77.5 -59.5l64 -110q26 -46 12.5 -97.5t-59.5 -77.5l-266 -154z" /> +<glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM896 161v190q0 14 -9 23.5t-22 9.5h-192q-13 0 -23 -10t-10 -23v-190q0 -13 10 -23t23 -10h192 q13 0 22 9.5t9 23.5zM894 505l18 621q0 12 -10 18q-10 8 -24 8h-220q-14 0 -24 -8q-10 -6 -10 -18l17 -621q0 -10 10 -17.5t24 -7.5h185q14 0 23.5 7.5t10.5 17.5z" /> +<glyph unicode="" d="M928 180v56v468v192h-320v-192v-468v-56q0 -25 18 -38.5t46 -13.5h192q28 0 46 13.5t18 38.5zM472 1024h195l-126 161q-26 31 -69 31q-40 0 -68 -28t-28 -68t28 -68t68 -28zM1160 1120q0 40 -28 68t-68 28q-43 0 -69 -31l-125 -161h194q40 0 68 28t28 68zM1536 864v-320 q0 -14 -9 -23t-23 -9h-96v-416q0 -40 -28 -68t-68 -28h-1088q-40 0 -68 28t-28 68v416h-96q-14 0 -23 9t-9 23v320q0 14 9 23t23 9h440q-93 0 -158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5q107 0 168 -77l128 -165l128 165q61 77 168 77q93 0 158.5 -65.5t65.5 -158.5 t-65.5 -158.5t-158.5 -65.5h440q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1280 832q0 26 -19 45t-45 19q-172 0 -318 -49.5t-259.5 -134t-235.5 -219.5q-19 -21 -19 -45q0 -26 19 -45t45 -19q24 0 45 19q27 24 74 71t67 66q137 124 268.5 176t313.5 52q26 0 45 19t19 45zM1792 1030q0 -95 -20 -193q-46 -224 -184.5 -383t-357.5 -268 q-214 -108 -438 -108q-148 0 -286 47q-15 5 -88 42t-96 37q-16 0 -39.5 -32t-45 -70t-52.5 -70t-60 -32q-30 0 -51 11t-31 24t-27 42q-2 4 -6 11t-5.5 10t-3 9.5t-1.5 13.5q0 35 31 73.5t68 65.5t68 56t31 48q0 4 -14 38t-16 44q-9 51 -9 104q0 115 43.5 220t119 184.5 t170.5 139t204 95.5q55 18 145 25.5t179.5 9t178.5 6t163.5 24t113.5 56.5l29.5 29.5t29.5 28t27 20t36.5 16t43.5 4.5q39 0 70.5 -46t47.5 -112t24 -124t8 -96z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1408 -160v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-1344q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h1344q13 0 22.5 -9.5t9.5 -22.5zM1152 896q0 -78 -24.5 -144t-64 -112.5t-87.5 -88t-96 -77.5t-87.5 -72t-64 -81.5t-24.5 -96.5q0 -96 67 -224l-4 1l1 -1 q-90 41 -160 83t-138.5 100t-113.5 122.5t-72.5 150.5t-27.5 184q0 78 24.5 144t64 112.5t87.5 88t96 77.5t87.5 72t64 81.5t24.5 96.5q0 94 -66 224l3 -1l-1 1q90 -41 160 -83t138.5 -100t113.5 -122.5t72.5 -150.5t27.5 -184z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1664 576q-152 236 -381 353q61 -104 61 -225q0 -185 -131.5 -316.5t-316.5 -131.5t-316.5 131.5t-131.5 316.5q0 121 61 225q-229 -117 -381 -353q133 -205 333.5 -326.5t434.5 -121.5t434.5 121.5t333.5 326.5zM944 960q0 20 -14 34t-34 14q-125 0 -214.5 -89.5 t-89.5 -214.5q0 -20 14 -34t34 -14t34 14t14 34q0 86 61 147t147 61q20 0 34 14t14 34zM1792 576q0 -34 -20 -69q-140 -230 -376.5 -368.5t-499.5 -138.5t-499.5 139t-376.5 368q-20 35 -20 69t20 69q140 229 376.5 368t499.5 139t499.5 -139t376.5 -368q20 -35 20 -69z" /> +<glyph unicode="" horiz-adv-x="1792" d="M555 201l78 141q-87 63 -136 159t-49 203q0 121 61 225q-229 -117 -381 -353q167 -258 427 -375zM944 960q0 20 -14 34t-34 14q-125 0 -214.5 -89.5t-89.5 -214.5q0 -20 14 -34t34 -14t34 14t14 34q0 86 61 147t147 61q20 0 34 14t14 34zM1307 1151q0 -7 -1 -9 q-105 -188 -315 -566t-316 -567l-49 -89q-10 -16 -28 -16q-12 0 -134 70q-16 10 -16 28q0 12 44 87q-143 65 -263.5 173t-208.5 245q-20 31 -20 69t20 69q153 235 380 371t496 136q89 0 180 -17l54 97q10 16 28 16q5 0 18 -6t31 -15.5t33 -18.5t31.5 -18.5t19.5 -11.5 q16 -10 16 -27zM1344 704q0 -139 -79 -253.5t-209 -164.5l280 502q8 -45 8 -84zM1792 576q0 -35 -20 -69q-39 -64 -109 -145q-150 -172 -347.5 -267t-419.5 -95l74 132q212 18 392.5 137t301.5 307q-115 179 -282 294l63 112q95 -64 182.5 -153t144.5 -184q20 -34 20 -69z " /> +<glyph unicode="" horiz-adv-x="1792" d="M1024 161v190q0 14 -9.5 23.5t-22.5 9.5h-192q-13 0 -22.5 -9.5t-9.5 -23.5v-190q0 -14 9.5 -23.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 23.5zM1022 535l18 459q0 12 -10 19q-13 11 -24 11h-220q-11 0 -24 -11q-10 -7 -10 -21l17 -457q0 -10 10 -16.5t24 -6.5h185 q14 0 23.5 6.5t10.5 16.5zM1008 1469l768 -1408q35 -63 -2 -126q-17 -29 -46.5 -46t-63.5 -17h-1536q-34 0 -63.5 17t-46.5 46q-37 63 -2 126l768 1408q17 31 47 49t65 18t65 -18t47 -49z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1376 1376q44 -52 12 -148t-108 -172l-161 -161l160 -696q5 -19 -12 -33l-128 -96q-7 -6 -19 -6q-4 0 -7 1q-15 3 -21 16l-279 508l-259 -259l53 -194q5 -17 -8 -31l-96 -96q-9 -9 -23 -9h-2q-15 2 -24 13l-189 252l-252 189q-11 7 -13 23q-1 13 9 25l96 97q9 9 23 9 q6 0 8 -1l194 -53l259 259l-508 279q-14 8 -17 24q-2 16 9 27l128 128q14 13 30 8l665 -159l160 160q76 76 172 108t148 -12z" /> +<glyph unicode="" horiz-adv-x="1664" d="M128 -128h288v288h-288v-288zM480 -128h320v288h-320v-288zM128 224h288v320h-288v-320zM480 224h320v320h-320v-320zM128 608h288v288h-288v-288zM864 -128h320v288h-320v-288zM480 608h320v288h-320v-288zM1248 -128h288v288h-288v-288zM864 224h320v320h-320v-320z M512 1088v288q0 13 -9.5 22.5t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-288q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1248 224h288v320h-288v-320zM864 608h320v288h-320v-288zM1248 608h288v288h-288v-288zM1280 1088v288q0 13 -9.5 22.5t-22.5 9.5h-64 q-13 0 -22.5 -9.5t-9.5 -22.5v-288q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1664 1152v-1280q0 -52 -38 -90t-90 -38h-1408q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h128v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h384v96q0 66 47 113t113 47 h64q66 0 113 -47t47 -113v-96h128q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="1792" d="M666 1055q-60 -92 -137 -273q-22 45 -37 72.5t-40.5 63.5t-51 56.5t-63 35t-81.5 14.5h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224q250 0 410 -225zM1792 256q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v192q-32 0 -85 -0.5t-81 -1t-73 1 t-71 5t-64 10.5t-63 18.5t-58 28.5t-59 40t-55 53.5t-56 69.5q59 93 136 273q22 -45 37 -72.5t40.5 -63.5t51 -56.5t63 -35t81.5 -14.5h256v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23zM1792 1152q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5 v192h-256q-48 0 -87 -15t-69 -45t-51 -61.5t-45 -77.5q-32 -62 -78 -171q-29 -66 -49.5 -111t-54 -105t-64 -100t-74 -83t-90 -68.5t-106.5 -42t-128 -16.5h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224q48 0 87 15t69 45t51 61.5t45 77.5q32 62 78 171q29 66 49.5 111 t54 105t64 100t74 83t90 68.5t106.5 42t128 16.5h256v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -174 -120 -321.5t-326 -233t-450 -85.5q-70 0 -145 8q-198 -175 -460 -242q-49 -14 -114 -22q-17 -2 -30.5 9t-17.5 29v1q-3 4 -0.5 12t2 10t4.5 9.5l6 9t7 8.5t8 9q7 8 31 34.5t34.5 38t31 39.5t32.5 51t27 59t26 76q-157 89 -247.5 220t-90.5 281 q0 130 71 248.5t191 204.5t286 136.5t348 50.5q244 0 450 -85.5t326 -233t120 -321.5z" /> +<glyph unicode="" d="M1536 704v-128q0 -201 -98.5 -362t-274 -251.5t-395.5 -90.5t-395.5 90.5t-274 251.5t-98.5 362v128q0 26 19 45t45 19h384q26 0 45 -19t19 -45v-128q0 -52 23.5 -90t53.5 -57t71 -30t64 -13t44 -2t44 2t64 13t71 30t53.5 57t23.5 90v128q0 26 19 45t45 19h384 q26 0 45 -19t19 -45zM512 1344v-384q0 -26 -19 -45t-45 -19h-384q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h384q26 0 45 -19t19 -45zM1536 1344v-384q0 -26 -19 -45t-45 -19h-384q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h384q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1683 205l-166 -165q-19 -19 -45 -19t-45 19l-531 531l-531 -531q-19 -19 -45 -19t-45 19l-166 165q-19 19 -19 45.5t19 45.5l742 741q19 19 45 19t45 -19l742 -741q19 -19 19 -45.5t-19 -45.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1683 728l-742 -741q-19 -19 -45 -19t-45 19l-742 741q-19 19 -19 45.5t19 45.5l166 165q19 19 45 19t45 -19l531 -531l531 531q19 19 45 19t45 -19l166 -165q19 -19 19 -45.5t-19 -45.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1280 32q0 -13 -9.5 -22.5t-22.5 -9.5h-960q-8 0 -13.5 2t-9 7t-5.5 8t-3 11.5t-1 11.5v13v11v160v416h-192q-26 0 -45 19t-19 45q0 24 15 41l320 384q19 22 49 22t49 -22l320 -384q15 -17 15 -41q0 -26 -19 -45t-45 -19h-192v-384h576q16 0 25 -11l160 -192q7 -11 7 -21 zM1920 448q0 -24 -15 -41l-320 -384q-20 -23 -49 -23t-49 23l-320 384q-15 17 -15 41q0 26 19 45t45 19h192v384h-576q-16 0 -25 12l-160 192q-7 9 -7 20q0 13 9.5 22.5t22.5 9.5h960q8 0 13.5 -2t9 -7t5.5 -8t3 -11.5t1 -11.5v-13v-11v-160v-416h192q26 0 45 -19t19 -45z " /> +<glyph unicode="" horiz-adv-x="1664" d="M640 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1536 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1664 1088v-512q0 -24 -16 -42.5t-41 -21.5 l-1044 -122q1 -7 4.5 -21.5t6 -26.5t2.5 -22q0 -16 -24 -64h920q26 0 45 -19t19 -45t-19 -45t-45 -19h-1024q-26 0 -45 19t-19 45q0 14 11 39.5t29.5 59.5t20.5 38l-177 823h-204q-26 0 -45 19t-19 45t19 45t45 19h256q16 0 28.5 -6.5t20 -15.5t13 -24.5t7.5 -26.5 t5.5 -29.5t4.5 -25.5h1201q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1664 928v-704q0 -92 -66 -158t-158 -66h-1216q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h672q92 0 158 -66t66 -158z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1879 584q0 -31 -31 -66l-336 -396q-43 -51 -120.5 -86.5t-143.5 -35.5h-1088q-34 0 -60.5 13t-26.5 43q0 31 31 66l336 396q43 51 120.5 86.5t143.5 35.5h1088q34 0 60.5 -13t26.5 -43zM1536 928v-160h-832q-94 0 -197 -47.5t-164 -119.5l-337 -396l-5 -6q0 4 -0.5 12.5 t-0.5 12.5v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h544q92 0 158 -66t66 -158z" /> +<glyph unicode="" horiz-adv-x="768" d="M704 1216q0 -26 -19 -45t-45 -19h-128v-1024h128q26 0 45 -19t19 -45t-19 -45l-256 -256q-19 -19 -45 -19t-45 19l-256 256q-19 19 -19 45t19 45t45 19h128v1024h-128q-26 0 -45 19t-19 45t19 45l256 256q19 19 45 19t45 -19l256 -256q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -26 -19 -45l-256 -256q-19 -19 -45 -19t-45 19t-19 45v128h-1024v-128q0 -26 -19 -45t-45 -19t-45 19l-256 256q-19 19 -19 45t19 45l256 256q19 19 45 19t45 -19t19 -45v-128h1024v128q0 26 19 45t45 19t45 -19l256 -256q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="2048" d="M640 640v-512h-256v512h256zM1024 1152v-1024h-256v1024h256zM2048 0v-128h-2048v1536h128v-1408h1920zM1408 896v-768h-256v768h256zM1792 1280v-1152h-256v1152h256z" /> +<glyph unicode="" d="M1280 926q-56 -25 -121 -34q68 40 93 117q-65 -38 -134 -51q-61 66 -153 66q-87 0 -148.5 -61.5t-61.5 -148.5q0 -29 5 -48q-129 7 -242 65t-192 155q-29 -50 -29 -106q0 -114 91 -175q-47 1 -100 26v-2q0 -75 50 -133.5t123 -72.5q-29 -8 -51 -8q-13 0 -39 4 q21 -63 74.5 -104t121.5 -42q-116 -90 -261 -90q-26 0 -50 3q148 -94 322 -94q112 0 210 35.5t168 95t120.5 137t75 162t24.5 168.5q0 18 -1 27q63 45 105 109zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5 t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1536 160q0 -119 -84.5 -203.5t-203.5 -84.5h-192v608h203l30 224h-233v143q0 54 28 83t96 29l132 1v207q-96 9 -180 9q-136 0 -218 -80.5t-82 -225.5v-166h-224v-224h224v-608h-544q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960 q119 0 203.5 -84.5t84.5 -203.5v-960z" /> +<glyph unicode="" horiz-adv-x="1792" d="M928 704q0 14 -9 23t-23 9q-66 0 -113 -47t-47 -113q0 -14 9 -23t23 -9t23 9t9 23q0 40 28 68t68 28q14 0 23 9t9 23zM1152 574q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM128 0h1536v128h-1536v-128zM1280 574q0 159 -112.5 271.5 t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5t271.5 -112.5t271.5 112.5t112.5 271.5zM256 1216h384v128h-384v-128zM128 1024h1536v118v138h-828l-64 -128h-644v-128zM1792 1280v-1280q0 -53 -37.5 -90.5t-90.5 -37.5h-1536q-53 0 -90.5 37.5t-37.5 90.5v1280 q0 53 37.5 90.5t90.5 37.5h1536q53 0 90.5 -37.5t37.5 -90.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M832 1024q0 80 -56 136t-136 56t-136 -56t-56 -136q0 -42 19 -83q-41 19 -83 19q-80 0 -136 -56t-56 -136t56 -136t136 -56t136 56t56 136q0 42 -19 83q41 -19 83 -19q80 0 136 56t56 136zM1683 320q0 -17 -49 -66t-66 -49q-9 0 -28.5 16t-36.5 33t-38.5 40t-24.5 26 l-96 -96l220 -220q28 -28 28 -68q0 -42 -39 -81t-81 -39q-40 0 -68 28l-671 671q-176 -131 -365 -131q-163 0 -265.5 102.5t-102.5 265.5q0 160 95 313t248 248t313 95q163 0 265.5 -102.5t102.5 -265.5q0 -189 -131 -365l355 -355l96 96q-3 3 -26 24.5t-40 38.5t-33 36.5 t-16 28.5q0 17 49 66t66 49q13 0 23 -10q6 -6 46 -44.5t82 -79.5t86.5 -86t73 -78t28.5 -41z" /> +<glyph unicode="" horiz-adv-x="1920" d="M896 640q0 106 -75 181t-181 75t-181 -75t-75 -181t75 -181t181 -75t181 75t75 181zM1664 128q0 52 -38 90t-90 38t-90 -38t-38 -90q0 -53 37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 1152q0 52 -38 90t-90 38t-90 -38t-38 -90q0 -53 37.5 -90.5t90.5 -37.5 t90.5 37.5t37.5 90.5zM1280 731v-185q0 -10 -7 -19.5t-16 -10.5l-155 -24q-11 -35 -32 -76q34 -48 90 -115q7 -10 7 -20q0 -12 -7 -19q-23 -30 -82.5 -89.5t-78.5 -59.5q-11 0 -21 7l-115 90q-37 -19 -77 -31q-11 -108 -23 -155q-7 -24 -30 -24h-186q-11 0 -20 7.5t-10 17.5 l-23 153q-34 10 -75 31l-118 -89q-7 -7 -20 -7q-11 0 -21 8q-144 133 -144 160q0 9 7 19q10 14 41 53t47 61q-23 44 -35 82l-152 24q-10 1 -17 9.5t-7 19.5v185q0 10 7 19.5t16 10.5l155 24q11 35 32 76q-34 48 -90 115q-7 11 -7 20q0 12 7 20q22 30 82 89t79 59q11 0 21 -7 l115 -90q34 18 77 32q11 108 23 154q7 24 30 24h186q11 0 20 -7.5t10 -17.5l23 -153q34 -10 75 -31l118 89q8 7 20 7q11 0 21 -8q144 -133 144 -160q0 -9 -7 -19q-12 -16 -42 -54t-45 -60q23 -48 34 -82l152 -23q10 -2 17 -10.5t7 -19.5zM1920 198v-140q0 -16 -149 -31 q-12 -27 -30 -52q51 -113 51 -138q0 -4 -4 -7q-122 -71 -124 -71q-8 0 -46 47t-52 68q-20 -2 -30 -2t-30 2q-14 -21 -52 -68t-46 -47q-2 0 -124 71q-4 3 -4 7q0 25 51 138q-18 25 -30 52q-149 15 -149 31v140q0 16 149 31q13 29 30 52q-51 113 -51 138q0 4 4 7q4 2 35 20 t59 34t30 16q8 0 46 -46.5t52 -67.5q20 2 30 2t30 -2q51 71 92 112l6 2q4 0 124 -70q4 -3 4 -7q0 -25 -51 -138q17 -23 30 -52q149 -15 149 -31zM1920 1222v-140q0 -16 -149 -31q-12 -27 -30 -52q51 -113 51 -138q0 -4 -4 -7q-122 -71 -124 -71q-8 0 -46 47t-52 68 q-20 -2 -30 -2t-30 2q-14 -21 -52 -68t-46 -47q-2 0 -124 71q-4 3 -4 7q0 25 51 138q-18 25 -30 52q-149 15 -149 31v140q0 16 149 31q13 29 30 52q-51 113 -51 138q0 4 4 7q4 2 35 20t59 34t30 16q8 0 46 -46.5t52 -67.5q20 2 30 2t30 -2q51 71 92 112l6 2q4 0 124 -70 q4 -3 4 -7q0 -25 -51 -138q17 -23 30 -52q149 -15 149 -31z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1408 768q0 -139 -94 -257t-256.5 -186.5t-353.5 -68.5q-86 0 -176 16q-124 -88 -278 -128q-36 -9 -86 -16h-3q-11 0 -20.5 8t-11.5 21q-1 3 -1 6.5t0.5 6.5t2 6l2.5 5t3.5 5.5t4 5t4.5 5t4 4.5q5 6 23 25t26 29.5t22.5 29t25 38.5t20.5 44q-124 72 -195 177t-71 224 q0 139 94 257t256.5 186.5t353.5 68.5t353.5 -68.5t256.5 -186.5t94 -257zM1792 512q0 -120 -71 -224.5t-195 -176.5q10 -24 20.5 -44t25 -38.5t22.5 -29t26 -29.5t23 -25q1 -1 4 -4.5t4.5 -5t4 -5t3.5 -5.5l2.5 -5t2 -6t0.5 -6.5t-1 -6.5q-3 -14 -13 -22t-22 -7 q-50 7 -86 16q-154 40 -278 128q-90 -16 -176 -16q-271 0 -472 132q58 -4 88 -4q161 0 309 45t264 129q125 92 192 212t67 254q0 77 -23 152q129 -71 204 -178t75 -230z" /> +<glyph unicode="" d="M256 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 768q0 51 -39 89.5t-89 38.5h-352q0 58 48 159.5t48 160.5q0 98 -32 145t-128 47q-26 -26 -38 -85t-30.5 -125.5t-59.5 -109.5q-22 -23 -77 -91q-4 -5 -23 -30t-31.5 -41t-34.5 -42.5 t-40 -44t-38.5 -35.5t-40 -27t-35.5 -9h-32v-640h32q13 0 31.5 -3t33 -6.5t38 -11t35 -11.5t35.5 -12.5t29 -10.5q211 -73 342 -73h121q192 0 192 167q0 26 -5 56q30 16 47.5 52.5t17.5 73.5t-18 69q53 50 53 119q0 25 -10 55.5t-25 47.5q32 1 53.5 47t21.5 81zM1536 769 q0 -89 -49 -163q9 -33 9 -69q0 -77 -38 -144q3 -21 3 -43q0 -101 -60 -178q1 -139 -85 -219.5t-227 -80.5h-36h-93q-96 0 -189.5 22.5t-216.5 65.5q-116 40 -138 40h-288q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5h274q36 24 137 155q58 75 107 128 q24 25 35.5 85.5t30.5 126.5t62 108q39 37 90 37q84 0 151 -32.5t102 -101.5t35 -186q0 -93 -48 -192h176q104 0 180 -76t76 -179z" /> +<glyph unicode="" d="M256 1088q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 512q0 35 -21.5 81t-53.5 47q15 17 25 47.5t10 55.5q0 69 -53 119q18 32 18 69t-17.5 73.5t-47.5 52.5q5 30 5 56q0 85 -49 126t-136 41h-128q-131 0 -342 -73q-5 -2 -29 -10.5 t-35.5 -12.5t-35 -11.5t-38 -11t-33 -6.5t-31.5 -3h-32v-640h32q16 0 35.5 -9t40 -27t38.5 -35.5t40 -44t34.5 -42.5t31.5 -41t23 -30q55 -68 77 -91q41 -43 59.5 -109.5t30.5 -125.5t38 -85q96 0 128 47t32 145q0 59 -48 160.5t-48 159.5h352q50 0 89 38.5t39 89.5z M1536 511q0 -103 -76 -179t-180 -76h-176q48 -99 48 -192q0 -118 -35 -186q-35 -69 -102 -101.5t-151 -32.5q-51 0 -90 37q-34 33 -54 82t-25.5 90.5t-17.5 84.5t-31 64q-48 50 -107 127q-101 131 -137 155h-274q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5 h288q22 0 138 40q128 44 223 66t200 22h112q140 0 226.5 -79t85.5 -216v-5q60 -77 60 -178q0 -22 -3 -43q38 -67 38 -144q0 -36 -9 -69q49 -74 49 -163z" /> +<glyph unicode="" horiz-adv-x="896" d="M832 1504v-1339l-449 -236q-22 -12 -40 -12q-21 0 -31.5 14.5t-10.5 35.5q0 6 2 20l86 500l-364 354q-25 27 -25 48q0 37 56 46l502 73l225 455q19 41 49 41z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1664 940q0 81 -21.5 143t-55 98.5t-81.5 59.5t-94 31t-98 8t-112 -25.5t-110.5 -64t-86.5 -72t-60 -61.5q-18 -22 -49 -22t-49 22q-24 28 -60 61.5t-86.5 72t-110.5 64t-112 25.5t-98 -8t-94 -31t-81.5 -59.5t-55 -98.5t-21.5 -143q0 -168 187 -355l581 -560l580 559 q188 188 188 356zM1792 940q0 -221 -229 -450l-623 -600q-18 -18 -44 -18t-44 18l-624 602q-10 8 -27.5 26t-55.5 65.5t-68 97.5t-53.5 121t-23.5 138q0 220 127 344t351 124q62 0 126.5 -21.5t120 -58t95.5 -68.5t76 -68q36 36 76 68t95.5 68.5t120 58t126.5 21.5 q224 0 351 -124t127 -344z" /> +<glyph unicode="" horiz-adv-x="1664" d="M640 96q0 -4 1 -20t0.5 -26.5t-3 -23.5t-10 -19.5t-20.5 -6.5h-320q-119 0 -203.5 84.5t-84.5 203.5v704q0 119 84.5 203.5t203.5 84.5h320q13 0 22.5 -9.5t9.5 -22.5q0 -4 1 -20t0.5 -26.5t-3 -23.5t-10 -19.5t-20.5 -6.5h-320q-66 0 -113 -47t-47 -113v-704 q0 -66 47 -113t113 -47h288h11h13t11.5 -1t11.5 -3t8 -5.5t7 -9t2 -13.5zM1568 640q0 -26 -19 -45l-544 -544q-19 -19 -45 -19t-45 19t-19 45v288h-448q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h448v288q0 26 19 45t45 19t45 -19l544 -544q19 -19 19 -45z" /> +<glyph unicode="" d="M237 122h231v694h-231v-694zM483 1030q-1 52 -36 86t-93 34t-94.5 -34t-36.5 -86q0 -51 35.5 -85.5t92.5 -34.5h1q59 0 95 34.5t36 85.5zM1068 122h231v398q0 154 -73 233t-193 79q-136 0 -209 -117h2v101h-231q3 -66 0 -694h231v388q0 38 7 56q15 35 45 59.5t74 24.5 q116 0 116 -157v-371zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1152" d="M480 672v448q0 14 -9 23t-23 9t-23 -9t-9 -23v-448q0 -14 9 -23t23 -9t23 9t9 23zM1152 320q0 -26 -19 -45t-45 -19h-429l-51 -483q-2 -12 -10.5 -20.5t-20.5 -8.5h-1q-27 0 -32 27l-76 485h-404q-26 0 -45 19t-19 45q0 123 78.5 221.5t177.5 98.5v512q-52 0 -90 38 t-38 90t38 90t90 38h640q52 0 90 -38t38 -90t-38 -90t-90 -38v-512q99 0 177.5 -98.5t78.5 -221.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1408 608v-320q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h704q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-704q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v320 q0 14 9 23t23 9h64q14 0 23 -9t9 -23zM1792 1472v-512q0 -26 -19 -45t-45 -19t-45 19l-176 176l-652 -652q-10 -10 -23 -10t-23 10l-114 114q-10 10 -10 23t10 23l652 652l-176 176q-19 19 -19 45t19 45t45 19h512q26 0 45 -19t19 -45z" /> +<glyph unicode="" d="M1184 640q0 -26 -19 -45l-544 -544q-19 -19 -45 -19t-45 19t-19 45v288h-448q-26 0 -45 19t-19 45v384q0 26 19 45t45 19h448v288q0 26 19 45t45 19t45 -19l544 -544q19 -19 19 -45zM1536 992v-704q0 -119 -84.5 -203.5t-203.5 -84.5h-320q-13 0 -22.5 9.5t-9.5 22.5 q0 4 -1 20t-0.5 26.5t3 23.5t10 19.5t20.5 6.5h320q66 0 113 47t47 113v704q0 66 -47 113t-113 47h-288h-11h-13t-11.5 1t-11.5 3t-8 5.5t-7 9t-2 13.5q0 4 -1 20t-0.5 26.5t3 23.5t10 19.5t20.5 6.5h320q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M458 653q-74 162 -74 371h-256v-96q0 -78 94.5 -162t235.5 -113zM1536 928v96h-256q0 -209 -74 -371q141 29 235.5 113t94.5 162zM1664 1056v-128q0 -71 -41.5 -143t-112 -130t-173 -97.5t-215.5 -44.5q-42 -54 -95 -95q-38 -34 -52.5 -72.5t-14.5 -89.5q0 -54 30.5 -91 t97.5 -37q75 0 133.5 -45.5t58.5 -114.5v-64q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v64q0 69 58.5 114.5t133.5 45.5q67 0 97.5 37t30.5 91q0 51 -14.5 89.5t-52.5 72.5q-53 41 -95 95q-113 5 -215.5 44.5t-173 97.5t-112 130t-41.5 143v128q0 40 28 68t68 28h288v96 q0 66 47 113t113 47h576q66 0 113 -47t47 -113v-96h288q40 0 68 -28t28 -68z" /> +<glyph unicode="" d="M394 184q-8 -9 -20 3q-13 11 -4 19q8 9 20 -3q12 -11 4 -19zM352 245q9 -12 0 -19q-8 -6 -17 7t0 18q9 7 17 -6zM291 305q-5 -7 -13 -2q-10 5 -7 12q3 5 13 2q10 -5 7 -12zM322 271q-6 -7 -16 3q-9 11 -2 16q6 6 16 -3q9 -11 2 -16zM451 159q-4 -12 -19 -6q-17 4 -13 15 t19 7q16 -5 13 -16zM514 154q0 -11 -16 -11q-17 -2 -17 11q0 11 16 11q17 2 17 -11zM572 164q2 -10 -14 -14t-18 8t14 15q16 2 18 -9zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-224q-16 0 -24.5 1t-19.5 5t-16 14.5t-5 27.5v239q0 97 -52 142q57 6 102.5 18t94 39 t81 66.5t53 105t20.5 150.5q0 121 -79 206q37 91 -8 204q-28 9 -81 -11t-92 -44l-38 -24q-93 26 -192 26t-192 -26q-16 11 -42.5 27t-83.5 38.5t-86 13.5q-44 -113 -7 -204q-79 -85 -79 -206q0 -85 20.5 -150t52.5 -105t80.5 -67t94 -39t102.5 -18q-40 -36 -49 -103 q-21 -10 -45 -15t-57 -5t-65.5 21.5t-55.5 62.5q-19 32 -48.5 52t-49.5 24l-20 3q-21 0 -29 -4.5t-5 -11.5t9 -14t13 -12l7 -5q22 -10 43.5 -38t31.5 -51l10 -23q13 -38 44 -61.5t67 -30t69.5 -7t55.5 3.5l23 4q0 -38 0.5 -103t0.5 -68q0 -22 -11 -33.5t-22 -13t-33 -1.5 h-224q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1280 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 288v-320q0 -40 -28 -68t-68 -28h-1472q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h427q21 -56 70.5 -92 t110.5 -36h256q61 0 110.5 36t70.5 92h427q40 0 68 -28t28 -68zM1339 936q-17 -40 -59 -40h-256v-448q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v448h-256q-42 0 -59 40q-17 39 14 69l448 448q18 19 45 19t45 -19l448 -448q31 -30 14 -69z" /> +<glyph unicode="" d="M1407 710q0 44 -7 113.5t-18 96.5q-12 30 -17 44t-9 36.5t-4 48.5q0 23 5 68.5t5 67.5q0 37 -10 55q-4 1 -13 1q-19 0 -58 -4.5t-59 -4.5q-60 0 -176 24t-175 24q-43 0 -94.5 -11.5t-85 -23.5t-89.5 -34q-137 -54 -202 -103q-96 -73 -159.5 -189.5t-88 -236t-24.5 -248.5 q0 -40 12.5 -120t12.5 -121q0 -23 -11 -66.5t-11 -65.5t12 -36.5t34 -14.5q24 0 72.5 11t73.5 11q57 0 169.5 -15.5t169.5 -15.5q181 0 284 36q129 45 235.5 152.5t166 245.5t59.5 275zM1535 712q0 -165 -70 -327.5t-196 -288t-281 -180.5q-124 -44 -326 -44 q-57 0 -170 14.5t-169 14.5q-24 0 -72.5 -14.5t-73.5 -14.5q-73 0 -123.5 55.5t-50.5 128.5q0 24 11 68t11 67q0 40 -12.5 120.5t-12.5 121.5q0 111 18 217.5t54.5 209.5t100.5 194t150 156q78 59 232 120q194 78 316 78q60 0 175.5 -24t173.5 -24q19 0 57 5t58 5 q81 0 118 -50.5t37 -134.5q0 -23 -5 -68t-5 -68q0 -10 1 -18.5t3 -17t4 -13.5t6.5 -16t6.5 -17q16 -40 25 -118.5t9 -136.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1408 296q0 -27 -10 -70.5t-21 -68.5q-21 -50 -122 -106q-94 -51 -186 -51q-27 0 -52.5 3.5t-57.5 12.5t-47.5 14.5t-55.5 20.5t-49 18q-98 35 -175 83q-128 79 -264.5 215.5t-215.5 264.5q-48 77 -83 175q-3 9 -18 49t-20.5 55.5t-14.5 47.5t-12.5 57.5t-3.5 52.5 q0 92 51 186q56 101 106 122q25 11 68.5 21t70.5 10q14 0 21 -3q18 -6 53 -76q11 -19 30 -54t35 -63.5t31 -53.5q3 -4 17.5 -25t21.5 -35.5t7 -28.5q0 -20 -28.5 -50t-62 -55t-62 -53t-28.5 -46q0 -9 5 -22.5t8.5 -20.5t14 -24t11.5 -19q76 -137 174 -235t235 -174 q2 -1 19 -11.5t24 -14t20.5 -8.5t22.5 -5q18 0 46 28.5t53 62t55 62t50 28.5q14 0 28.5 -7t35.5 -21.5t25 -17.5q25 -15 53.5 -31t63.5 -35t54 -30q70 -35 76 -53q3 -7 3 -21z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1120 1280h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113v832q0 66 -47 113t-113 47zM1408 1120v-832q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832 q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1152 1280h-1024v-1242l423 406l89 85l89 -85l423 -406v1242zM1164 1408q23 0 44 -9q33 -13 52.5 -41t19.5 -62v-1289q0 -34 -19.5 -62t-52.5 -41q-19 -8 -44 -8q-48 0 -83 32l-441 424l-441 -424q-36 -33 -83 -33q-23 0 -44 9q-33 13 -52.5 41t-19.5 62v1289 q0 34 19.5 62t52.5 41q21 9 44 9h1048z" /> +<glyph unicode="" d="M1280 343q0 11 -2 16q-3 8 -38.5 29.5t-88.5 49.5l-53 29q-5 3 -19 13t-25 15t-21 5q-18 0 -47 -32.5t-57 -65.5t-44 -33q-7 0 -16.5 3.5t-15.5 6.5t-17 9.5t-14 8.5q-99 55 -170.5 126.5t-126.5 170.5q-2 3 -8.5 14t-9.5 17t-6.5 15.5t-3.5 16.5q0 13 20.5 33.5t45 38.5 t45 39.5t20.5 36.5q0 10 -5 21t-15 25t-13 19q-3 6 -15 28.5t-25 45.5t-26.5 47.5t-25 40.5t-16.5 18t-16 2q-48 0 -101 -22q-46 -21 -80 -94.5t-34 -130.5q0 -16 2.5 -34t5 -30.5t9 -33t10 -29.5t12.5 -33t11 -30q60 -164 216.5 -320.5t320.5 -216.5q6 -2 30 -11t33 -12.5 t29.5 -10t33 -9t30.5 -5t34 -2.5q57 0 130.5 34t94.5 80q22 53 22 101zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1620 1128q-67 -98 -162 -167q1 -14 1 -42q0 -130 -38 -259.5t-115.5 -248.5t-184.5 -210.5t-258 -146t-323 -54.5q-271 0 -496 145q35 -4 78 -4q225 0 401 138q-105 2 -188 64.5t-114 159.5q33 -5 61 -5q43 0 85 11q-112 23 -185.5 111.5t-73.5 205.5v4q68 -38 146 -41 q-66 44 -105 115t-39 154q0 88 44 163q121 -149 294.5 -238.5t371.5 -99.5q-8 38 -8 74q0 134 94.5 228.5t228.5 94.5q140 0 236 -102q109 21 205 78q-37 -115 -142 -178q93 10 186 50z" /> +<glyph unicode="" horiz-adv-x="1024" d="M959 1524v-264h-157q-86 0 -116 -36t-30 -108v-189h293l-39 -296h-254v-759h-306v759h-255v296h255v218q0 186 104 288.5t277 102.5q147 0 228 -12z" /> +<glyph unicode="" d="M1536 640q0 -251 -146.5 -451.5t-378.5 -277.5q-27 -5 -39.5 7t-12.5 30v211q0 97 -52 142q57 6 102.5 18t94 39t81 66.5t53 105t20.5 150.5q0 121 -79 206q37 91 -8 204q-28 9 -81 -11t-92 -44l-38 -24q-93 26 -192 26t-192 -26q-16 11 -42.5 27t-83.5 38.5t-86 13.5 q-44 -113 -7 -204q-79 -85 -79 -206q0 -85 20.5 -150t52.5 -105t80.5 -67t94 -39t102.5 -18q-40 -36 -49 -103q-21 -10 -45 -15t-57 -5t-65.5 21.5t-55.5 62.5q-19 32 -48.5 52t-49.5 24l-20 3q-21 0 -29 -4.5t-5 -11.5t9 -14t13 -12l7 -5q22 -10 43.5 -38t31.5 -51l10 -23 q13 -38 44 -61.5t67 -30t69.5 -7t55.5 3.5l23 4q0 -38 0.5 -89t0.5 -54q0 -18 -13 -30t-40 -7q-232 77 -378.5 277.5t-146.5 451.5q0 209 103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1664 960v-256q0 -26 -19 -45t-45 -19h-64q-26 0 -45 19t-19 45v256q0 106 -75 181t-181 75t-181 -75t-75 -181v-192h96q40 0 68 -28t28 -68v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h672v192q0 185 131.5 316.5t316.5 131.5 t316.5 -131.5t131.5 -316.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1760 1408q66 0 113 -47t47 -113v-1216q0 -66 -47 -113t-113 -47h-1600q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1600zM160 1280q-13 0 -22.5 -9.5t-9.5 -22.5v-224h1664v224q0 13 -9.5 22.5t-22.5 9.5h-1600zM1760 0q13 0 22.5 9.5t9.5 22.5v608h-1664v-608 q0 -13 9.5 -22.5t22.5 -9.5h1600zM256 128v128h256v-128h-256zM640 128v128h384v-128h-384z" /> +<glyph unicode="" horiz-adv-x="1408" d="M384 192q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM896 69q2 -28 -17 -48q-18 -21 -47 -21h-135q-25 0 -43 16.5t-20 41.5q-22 229 -184.5 391.5t-391.5 184.5q-25 2 -41.5 20t-16.5 43v135q0 29 21 47q17 17 43 17h5q160 -13 306 -80.5 t259 -181.5q114 -113 181.5 -259t80.5 -306zM1408 67q2 -27 -18 -47q-18 -20 -46 -20h-143q-26 0 -44.5 17.5t-19.5 42.5q-12 215 -101 408.5t-231.5 336t-336 231.5t-408.5 102q-25 1 -42.5 19.5t-17.5 43.5v143q0 28 20 46q18 18 44 18h3q262 -13 501.5 -120t425.5 -294 q187 -186 294 -425.5t120 -501.5z" /> +<glyph unicode="" d="M1040 320q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5zM1296 320q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5zM1408 160v320q0 13 -9.5 22.5t-22.5 9.5 h-1216q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h1216q13 0 22.5 9.5t9.5 22.5zM178 640h1180l-157 482q-4 13 -16 21.5t-26 8.5h-782q-14 0 -26 -8.5t-16 -21.5zM1536 480v-320q0 -66 -47 -113t-113 -47h-1216q-66 0 -113 47t-47 113v320q0 25 16 75 l197 606q17 53 63 86t101 33h782q55 0 101 -33t63 -86l197 -606q16 -50 16 -75z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1664 896q53 0 90.5 -37.5t37.5 -90.5t-37.5 -90.5t-90.5 -37.5v-384q0 -52 -38 -90t-90 -38q-417 347 -812 380q-58 -19 -91 -66t-31 -100.5t40 -92.5q-20 -33 -23 -65.5t6 -58t33.5 -55t48 -50t61.5 -50.5q-29 -58 -111.5 -83t-168.5 -11.5t-132 55.5q-7 23 -29.5 87.5 t-32 94.5t-23 89t-15 101t3.5 98.5t22 110.5h-122q-66 0 -113 47t-47 113v192q0 66 47 113t113 47h480q435 0 896 384q52 0 90 -38t38 -90v-384zM1536 292v954q-394 -302 -768 -343v-270q377 -42 768 -341z" /> +<glyph unicode="" horiz-adv-x="1792" d="M912 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM246 128h1300q-266 300 -266 832q0 51 -24 105t-69 103t-121.5 80.5t-169.5 31.5t-169.5 -31.5t-121.5 -80.5t-69 -103t-24 -105q0 -532 -266 -832z M1728 128q0 -52 -38 -90t-90 -38h-448q0 -106 -75 -181t-181 -75t-181 75t-75 181h-448q-52 0 -90 38t-38 90q50 42 91 88t85 119.5t74.5 158.5t50 206t19.5 260q0 152 117 282.5t307 158.5q-8 19 -8 39q0 40 28 68t68 28t68 -28t28 -68q0 -20 -8 -39q190 -28 307 -158.5 t117 -282.5q0 -139 19.5 -260t50 -206t74.5 -158.5t85 -119.5t91 -88z" /> +<glyph unicode="" d="M1376 640l138 -135q30 -28 20 -70q-12 -41 -52 -51l-188 -48l53 -186q12 -41 -19 -70q-29 -31 -70 -19l-186 53l-48 -188q-10 -40 -51 -52q-12 -2 -19 -2q-31 0 -51 22l-135 138l-135 -138q-28 -30 -70 -20q-41 11 -51 52l-48 188l-186 -53q-41 -12 -70 19q-31 29 -19 70 l53 186l-188 48q-40 10 -52 51q-10 42 20 70l138 135l-138 135q-30 28 -20 70q12 41 52 51l188 48l-53 186q-12 41 19 70q29 31 70 19l186 -53l48 188q10 41 51 51q41 12 70 -19l135 -139l135 139q29 30 70 19q41 -10 51 -51l48 -188l186 53q41 12 70 -19q31 -29 19 -70 l-53 -186l188 -48q40 -10 52 -51q10 -42 -20 -70z" /> +<glyph unicode="" horiz-adv-x="1792" d="M256 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1664 768q0 51 -39 89.5t-89 38.5h-576q0 20 15 48.5t33 55t33 68t15 84.5q0 67 -44.5 97.5t-115.5 30.5q-24 0 -90 -139q-24 -44 -37 -65q-40 -64 -112 -145q-71 -81 -101 -106 q-69 -57 -140 -57h-32v-640h32q72 0 167 -32t193.5 -64t179.5 -32q189 0 189 167q0 26 -5 56q30 16 47.5 52.5t17.5 73.5t-18 69q53 50 53 119q0 25 -10 55.5t-25 47.5h331q52 0 90 38t38 90zM1792 769q0 -105 -75.5 -181t-180.5 -76h-169q-4 -62 -37 -119q3 -21 3 -43 q0 -101 -60 -178q1 -139 -85 -219.5t-227 -80.5q-133 0 -322 69q-164 59 -223 59h-288q-53 0 -90.5 37.5t-37.5 90.5v640q0 53 37.5 90.5t90.5 37.5h288q10 0 21.5 4.5t23.5 14t22.5 18t24 22.5t20.5 21.5t19 21.5t14 17q65 74 100 129q13 21 33 62t37 72t40.5 63t55 49.5 t69.5 17.5q125 0 206.5 -67t81.5 -189q0 -68 -22 -128h374q104 0 180 -76t76 -179z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1376 128h32v640h-32q-35 0 -67.5 12t-62.5 37t-50 46t-49 54q-2 3 -3.5 4.5t-4 4.5t-4.5 5q-72 81 -112 145q-14 22 -38 68q-1 3 -10.5 22.5t-18.5 36t-20 35.5t-21.5 30.5t-18.5 11.5q-71 0 -115.5 -30.5t-44.5 -97.5q0 -43 15 -84.5t33 -68t33 -55t15 -48.5h-576 q-50 0 -89 -38.5t-39 -89.5q0 -52 38 -90t90 -38h331q-15 -17 -25 -47.5t-10 -55.5q0 -69 53 -119q-18 -32 -18 -69t17.5 -73.5t47.5 -52.5q-4 -24 -4 -56q0 -85 48.5 -126t135.5 -41q84 0 183 32t194 64t167 32zM1664 192q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45 t45 -19t45 19t19 45zM1792 768v-640q0 -53 -37.5 -90.5t-90.5 -37.5h-288q-59 0 -223 -59q-190 -69 -317 -69q-142 0 -230 77.5t-87 217.5l1 5q-61 76 -61 178q0 22 3 43q-33 57 -37 119h-169q-105 0 -180.5 76t-75.5 181q0 103 76 179t180 76h374q-22 60 -22 128 q0 122 81.5 189t206.5 67q38 0 69.5 -17.5t55 -49.5t40.5 -63t37 -72t33 -62q35 -55 100 -129q2 -3 14 -17t19 -21.5t20.5 -21.5t24 -22.5t22.5 -18t23.5 -14t21.5 -4.5h288q53 0 90.5 -37.5t37.5 -90.5z" /> +<glyph unicode="" d="M1280 -64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 700q0 189 -167 189q-26 0 -56 -5q-16 30 -52.5 47.5t-73.5 17.5t-69 -18q-50 53 -119 53q-25 0 -55.5 -10t-47.5 -25v331q0 52 -38 90t-90 38q-51 0 -89.5 -39t-38.5 -89v-576 q-20 0 -48.5 15t-55 33t-68 33t-84.5 15q-67 0 -97.5 -44.5t-30.5 -115.5q0 -24 139 -90q44 -24 65 -37q64 -40 145 -112q81 -71 106 -101q57 -69 57 -140v-32h640v32q0 72 32 167t64 193.5t32 179.5zM1536 705q0 -133 -69 -322q-59 -164 -59 -223v-288q0 -53 -37.5 -90.5 t-90.5 -37.5h-640q-53 0 -90.5 37.5t-37.5 90.5v288q0 10 -4.5 21.5t-14 23.5t-18 22.5t-22.5 24t-21.5 20.5t-21.5 19t-17 14q-74 65 -129 100q-21 13 -62 33t-72 37t-63 40.5t-49.5 55t-17.5 69.5q0 125 67 206.5t189 81.5q68 0 128 -22v374q0 104 76 180t179 76 q105 0 181 -75.5t76 -180.5v-169q62 -4 119 -37q21 3 43 3q101 0 178 -60q139 1 219.5 -85t80.5 -227z" /> +<glyph unicode="" d="M1408 576q0 84 -32 183t-64 194t-32 167v32h-640v-32q0 -35 -12 -67.5t-37 -62.5t-46 -50t-54 -49q-9 -8 -14 -12q-81 -72 -145 -112q-22 -14 -68 -38q-3 -1 -22.5 -10.5t-36 -18.5t-35.5 -20t-30.5 -21.5t-11.5 -18.5q0 -71 30.5 -115.5t97.5 -44.5q43 0 84.5 15t68 33 t55 33t48.5 15v-576q0 -50 38.5 -89t89.5 -39q52 0 90 38t38 90v331q46 -35 103 -35q69 0 119 53q32 -18 69 -18t73.5 17.5t52.5 47.5q24 -4 56 -4q85 0 126 48.5t41 135.5zM1280 1344q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1536 580 q0 -142 -77.5 -230t-217.5 -87l-5 1q-76 -61 -178 -61q-22 0 -43 3q-54 -30 -119 -37v-169q0 -105 -76 -180.5t-181 -75.5q-103 0 -179 76t-76 180v374q-54 -22 -128 -22q-121 0 -188.5 81.5t-67.5 206.5q0 38 17.5 69.5t49.5 55t63 40.5t72 37t62 33q55 35 129 100 q3 2 17 14t21.5 19t21.5 20.5t22.5 24t18 22.5t14 23.5t4.5 21.5v288q0 53 37.5 90.5t90.5 37.5h640q53 0 90.5 -37.5t37.5 -90.5v-288q0 -59 59 -223q69 -190 69 -317z" /> +<glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-502l189 189q19 19 19 45t-19 45l-91 91q-18 18 -45 18t-45 -18l-362 -362l-91 -91q-18 -18 -18 -45t18 -45l91 -91l362 -362q18 -18 45 -18t45 18l91 91q18 18 18 45t-18 45l-189 189h502q26 0 45 19t19 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1285 640q0 27 -18 45l-91 91l-362 362q-18 18 -45 18t-45 -18l-91 -91q-18 -18 -18 -45t18 -45l189 -189h-502q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h502l-189 -189q-19 -19 -19 -45t19 -45l91 -91q18 -18 45 -18t45 18l362 362l91 91q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1284 641q0 27 -18 45l-362 362l-91 91q-18 18 -45 18t-45 -18l-91 -91l-362 -362q-18 -18 -18 -45t18 -45l91 -91q18 -18 45 -18t45 18l189 189v-502q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v502l189 -189q19 -19 45 -19t45 19l91 91q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1284 639q0 27 -18 45l-91 91q-18 18 -45 18t-45 -18l-189 -189v502q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-502l-189 189q-19 19 -45 19t-45 -19l-91 -91q-18 -18 -18 -45t18 -45l362 -362l91 -91q18 -18 45 -18t45 18l91 91l362 362q18 18 18 45zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM1042 887q-2 -1 -9.5 -9.5t-13.5 -9.5q2 0 4.5 5t5 11t3.5 7q6 7 22 15q14 6 52 12q34 8 51 -11 q-2 2 9.5 13t14.5 12q3 2 15 4.5t15 7.5l2 22q-12 -1 -17.5 7t-6.5 21q0 -2 -6 -8q0 7 -4.5 8t-11.5 -1t-9 -1q-10 3 -15 7.5t-8 16.5t-4 15q-2 5 -9.5 10.5t-9.5 10.5q-1 2 -2.5 5.5t-3 6.5t-4 5.5t-5.5 2.5t-7 -5t-7.5 -10t-4.5 -5q-3 2 -6 1.5t-4.5 -1t-4.5 -3t-5 -3.5 q-3 -2 -8.5 -3t-8.5 -2q15 5 -1 11q-10 4 -16 3q9 4 7.5 12t-8.5 14h5q-1 4 -8.5 8.5t-17.5 8.5t-13 6q-8 5 -34 9.5t-33 0.5q-5 -6 -4.5 -10.5t4 -14t3.5 -12.5q1 -6 -5.5 -13t-6.5 -12q0 -7 14 -15.5t10 -21.5q-3 -8 -16 -16t-16 -12q-5 -8 -1.5 -18.5t10.5 -16.5 q2 -2 1.5 -4t-3.5 -4.5t-5.5 -4t-6.5 -3.5l-3 -2q-11 -5 -20.5 6t-13.5 26q-7 25 -16 30q-23 8 -29 -1q-5 13 -41 26q-25 9 -58 4q6 1 0 15q-7 15 -19 12q3 6 4 17.5t1 13.5q3 13 12 23q1 1 7 8.5t9.5 13.5t0.5 6q35 -4 50 11q5 5 11.5 17t10.5 17q9 6 14 5.5t14.5 -5.5 t14.5 -5q14 -1 15.5 11t-7.5 20q12 -1 3 17q-5 7 -8 9q-12 4 -27 -5q-8 -4 2 -8q-1 1 -9.5 -10.5t-16.5 -17.5t-16 5q-1 1 -5.5 13.5t-9.5 13.5q-8 0 -16 -15q3 8 -11 15t-24 8q19 12 -8 27q-7 4 -20.5 5t-19.5 -4q-5 -7 -5.5 -11.5t5 -8t10.5 -5.5t11.5 -4t8.5 -3 q14 -10 8 -14q-2 -1 -8.5 -3.5t-11.5 -4.5t-6 -4q-3 -4 0 -14t-2 -14q-5 5 -9 17.5t-7 16.5q7 -9 -25 -6l-10 1q-4 0 -16 -2t-20.5 -1t-13.5 8q-4 8 0 20q1 4 4 2q-4 3 -11 9.5t-10 8.5q-46 -15 -94 -41q6 -1 12 1q5 2 13 6.5t10 5.5q34 14 42 7l5 5q14 -16 20 -25 q-7 4 -30 1q-20 -6 -22 -12q7 -12 5 -18q-4 3 -11.5 10t-14.5 11t-15 5q-16 0 -22 -1q-146 -80 -235 -222q7 -7 12 -8q4 -1 5 -9t2.5 -11t11.5 3q9 -8 3 -19q1 1 44 -27q19 -17 21 -21q3 -11 -10 -18q-1 2 -9 9t-9 4q-3 -5 0.5 -18.5t10.5 -12.5q-7 0 -9.5 -16t-2.5 -35.5 t-1 -23.5l2 -1q-3 -12 5.5 -34.5t21.5 -19.5q-13 -3 20 -43q6 -8 8 -9q3 -2 12 -7.5t15 -10t10 -10.5q4 -5 10 -22.5t14 -23.5q-2 -6 9.5 -20t10.5 -23q-1 0 -2.5 -1t-2.5 -1q3 -7 15.5 -14t15.5 -13q1 -3 2 -10t3 -11t8 -2q2 20 -24 62q-15 25 -17 29q-3 5 -5.5 15.5 t-4.5 14.5q2 0 6 -1.5t8.5 -3.5t7.5 -4t2 -3q-3 -7 2 -17.5t12 -18.5t17 -19t12 -13q6 -6 14 -19.5t0 -13.5q9 0 20 -10t17 -20q5 -8 8 -26t5 -24q2 -7 8.5 -13.5t12.5 -9.5l16 -8t13 -7q5 -2 18.5 -10.5t21.5 -11.5q10 -4 16 -4t14.5 2.5t13.5 3.5q15 2 29 -15t21 -21 q36 -19 55 -11q-2 -1 0.5 -7.5t8 -15.5t9 -14.5t5.5 -8.5q5 -6 18 -15t18 -15q6 4 7 9q-3 -8 7 -20t18 -10q14 3 14 32q-31 -15 -49 18q0 1 -2.5 5.5t-4 8.5t-2.5 8.5t0 7.5t5 3q9 0 10 3.5t-2 12.5t-4 13q-1 8 -11 20t-12 15q-5 -9 -16 -8t-16 9q0 -1 -1.5 -5.5t-1.5 -6.5 q-13 0 -15 1q1 3 2.5 17.5t3.5 22.5q1 4 5.5 12t7.5 14.5t4 12.5t-4.5 9.5t-17.5 2.5q-19 -1 -26 -20q-1 -3 -3 -10.5t-5 -11.5t-9 -7q-7 -3 -24 -2t-24 5q-13 8 -22.5 29t-9.5 37q0 10 2.5 26.5t3 25t-5.5 24.5q3 2 9 9.5t10 10.5q2 1 4.5 1.5t4.5 0t4 1.5t3 6q-1 1 -4 3 q-3 3 -4 3q7 -3 28.5 1.5t27.5 -1.5q15 -11 22 2q0 1 -2.5 9.5t-0.5 13.5q5 -27 29 -9q3 -3 15.5 -5t17.5 -5q3 -2 7 -5.5t5.5 -4.5t5 0.5t8.5 6.5q10 -14 12 -24q11 -40 19 -44q7 -3 11 -2t4.5 9.5t0 14t-1.5 12.5l-1 8v18l-1 8q-15 3 -18.5 12t1.5 18.5t15 18.5q1 1 8 3.5 t15.5 6.5t12.5 8q21 19 15 35q7 0 11 9q-1 0 -5 3t-7.5 5t-4.5 2q9 5 2 16q5 3 7.5 11t7.5 10q9 -12 21 -2q7 8 1 16q5 7 20.5 10.5t18.5 9.5q7 -2 8 2t1 12t3 12q4 5 15 9t13 5l17 11q3 4 0 4q18 -2 31 11q10 11 -6 20q3 6 -3 9.5t-15 5.5q3 1 11.5 0.5t10.5 1.5 q15 10 -7 16q-17 5 -43 -12zM879 10q206 36 351 189q-3 3 -12.5 4.5t-12.5 3.5q-18 7 -24 8q1 7 -2.5 13t-8 9t-12.5 8t-11 7q-2 2 -7 6t-7 5.5t-7.5 4.5t-8.5 2t-10 -1l-3 -1q-3 -1 -5.5 -2.5t-5.5 -3t-4 -3t0 -2.5q-21 17 -36 22q-5 1 -11 5.5t-10.5 7t-10 1.5t-11.5 -7 q-5 -5 -6 -15t-2 -13q-7 5 0 17.5t2 18.5q-3 6 -10.5 4.5t-12 -4.5t-11.5 -8.5t-9 -6.5t-8.5 -5.5t-8.5 -7.5q-3 -4 -6 -12t-5 -11q-2 4 -11.5 6.5t-9.5 5.5q2 -10 4 -35t5 -38q7 -31 -12 -48q-27 -25 -29 -40q-4 -22 12 -26q0 -7 -8 -20.5t-7 -21.5q0 -6 2 -16z" /> +<glyph unicode="" horiz-adv-x="1664" d="M384 64q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1028 484l-682 -682q-37 -37 -90 -37q-52 0 -91 37l-106 108q-38 36 -38 90q0 53 38 91l681 681q39 -98 114.5 -173.5t173.5 -114.5zM1662 919q0 -39 -23 -106q-47 -134 -164.5 -217.5 t-258.5 -83.5q-185 0 -316.5 131.5t-131.5 316.5t131.5 316.5t316.5 131.5q58 0 121.5 -16.5t107.5 -46.5q16 -11 16 -28t-16 -28l-293 -169v-224l193 -107q5 3 79 48.5t135.5 81t70.5 35.5q15 0 23.5 -10t8.5 -25z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1024 128h640v128h-640v-128zM640 640h1024v128h-1024v-128zM1280 1152h384v128h-384v-128zM1792 320v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 832v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19 t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45zM1792 1344v-256q0 -26 -19 -45t-45 -19h-1664q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1664q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1403 1241q17 -41 -14 -70l-493 -493v-742q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-256 256q-19 19 -19 45v486l-493 493q-31 29 -14 70q17 39 59 39h1280q42 0 59 -39z" /> +<glyph unicode="" horiz-adv-x="1792" d="M640 1280h512v128h-512v-128zM1792 640v-480q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v480h672v-160q0 -26 19 -45t45 -19h320q26 0 45 19t19 45v160h672zM1024 640v-128h-256v128h256zM1792 1120v-384h-1792v384q0 66 47 113t113 47h352v160q0 40 28 68 t68 28h576q40 0 68 -28t28 -68v-160h352q66 0 113 -47t47 -113z" /> +<glyph unicode="" d="M1283 995l-355 -355l355 -355l144 144q29 31 70 14q39 -17 39 -59v-448q0 -26 -19 -45t-45 -19h-448q-42 0 -59 40q-17 39 14 69l144 144l-355 355l-355 -355l144 -144q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l144 -144 l355 355l-355 355l-144 -144q-19 -19 -45 -19q-12 0 -24 5q-40 17 -40 59v448q0 26 19 45t45 19h448q42 0 59 -40q17 -39 -14 -69l-144 -144l355 -355l355 355l-144 144q-31 30 -14 69q17 40 59 40h448q26 0 45 -19t19 -45v-448q0 -42 -39 -59q-13 -5 -25 -5q-26 0 -45 19z " /> +<glyph unicode="" horiz-adv-x="1920" d="M593 640q-162 -5 -265 -128h-134q-82 0 -138 40.5t-56 118.5q0 353 124 353q6 0 43.5 -21t97.5 -42.5t119 -21.5q67 0 133 23q-5 -37 -5 -66q0 -139 81 -256zM1664 3q0 -120 -73 -189.5t-194 -69.5h-874q-121 0 -194 69.5t-73 189.5q0 53 3.5 103.5t14 109t26.5 108.5 t43 97.5t62 81t85.5 53.5t111.5 20q10 0 43 -21.5t73 -48t107 -48t135 -21.5t135 21.5t107 48t73 48t43 21.5q61 0 111.5 -20t85.5 -53.5t62 -81t43 -97.5t26.5 -108.5t14 -109t3.5 -103.5zM640 1280q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75 t75 -181zM1344 896q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5t271.5 -112.5t112.5 -271.5zM1920 671q0 -78 -56 -118.5t-138 -40.5h-134q-103 123 -265 128q81 117 81 256q0 29 -5 66q66 -23 133 -23q59 0 119 21.5t97.5 42.5 t43.5 21q124 0 124 -353zM1792 1280q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1456 320q0 40 -28 68l-208 208q-28 28 -68 28q-42 0 -72 -32q3 -3 19 -18.5t21.5 -21.5t15 -19t13 -25.5t3.5 -27.5q0 -40 -28 -68t-68 -28q-15 0 -27.5 3.5t-25.5 13t-19 15t-21.5 21.5t-18.5 19q-33 -31 -33 -73q0 -40 28 -68l206 -207q27 -27 68 -27q40 0 68 26 l147 146q28 28 28 67zM753 1025q0 40 -28 68l-206 207q-28 28 -68 28q-39 0 -68 -27l-147 -146q-28 -28 -28 -67q0 -40 28 -68l208 -208q27 -27 68 -27q42 0 72 31q-3 3 -19 18.5t-21.5 21.5t-15 19t-13 25.5t-3.5 27.5q0 40 28 68t68 28q15 0 27.5 -3.5t25.5 -13t19 -15 t21.5 -21.5t18.5 -19q33 31 33 73zM1648 320q0 -120 -85 -203l-147 -146q-83 -83 -203 -83q-121 0 -204 85l-206 207q-83 83 -83 203q0 123 88 209l-88 88q-86 -88 -208 -88q-120 0 -204 84l-208 208q-84 84 -84 204t85 203l147 146q83 83 203 83q121 0 204 -85l206 -207 q83 -83 83 -203q0 -123 -88 -209l88 -88q86 88 208 88q120 0 204 -84l208 -208q84 -84 84 -204z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088q-185 0 -316.5 131.5t-131.5 316.5q0 132 71 241.5t187 163.5q-2 28 -2 43q0 212 150 362t362 150q158 0 286.5 -88t187.5 -230q70 62 166 62q106 0 181 -75t75 -181q0 -75 -41 -138q129 -30 213 -134.5t84 -239.5z " /> +<glyph unicode="" horiz-adv-x="1664" d="M1527 88q56 -89 21.5 -152.5t-140.5 -63.5h-1152q-106 0 -140.5 63.5t21.5 152.5l503 793v399h-64q-26 0 -45 19t-19 45t19 45t45 19h512q26 0 45 -19t19 -45t-19 -45t-45 -19h-64v-399zM748 813l-272 -429h712l-272 429l-20 31v37v399h-128v-399v-37z" /> +<glyph unicode="" horiz-adv-x="1792" d="M960 640q26 0 45 -19t19 -45t-19 -45t-45 -19t-45 19t-19 45t19 45t45 19zM1260 576l507 -398q28 -20 25 -56q-5 -35 -35 -51l-128 -64q-13 -7 -29 -7q-17 0 -31 8l-690 387l-110 -66q-8 -4 -12 -5q14 -49 10 -97q-7 -77 -56 -147.5t-132 -123.5q-132 -84 -277 -84 q-136 0 -222 78q-90 84 -79 207q7 76 56 147t131 124q132 84 278 84q83 0 151 -31q9 13 22 22l122 73l-122 73q-13 9 -22 22q-68 -31 -151 -31q-146 0 -278 84q-82 53 -131 124t-56 147q-5 59 15.5 113t63.5 93q85 79 222 79q145 0 277 -84q83 -52 132 -123t56 -148 q4 -48 -10 -97q4 -1 12 -5l110 -66l690 387q14 8 31 8q16 0 29 -7l128 -64q30 -16 35 -51q3 -36 -25 -56zM579 836q46 42 21 108t-106 117q-92 59 -192 59q-74 0 -113 -36q-46 -42 -21 -108t106 -117q92 -59 192 -59q74 0 113 36zM494 91q81 51 106 117t-21 108 q-39 36 -113 36q-100 0 -192 -59q-81 -51 -106 -117t21 -108q39 -36 113 -36q100 0 192 59zM672 704l96 -58v11q0 36 33 56l14 8l-79 47l-26 -26q-3 -3 -10 -11t-12 -12q-2 -2 -4 -3.5t-3 -2.5zM896 480l96 -32l736 576l-128 64l-768 -431v-113l-160 -96l9 -8q2 -2 7 -6 q4 -4 11 -12t11 -12l26 -26zM1600 64l128 64l-520 408l-177 -138q-2 -3 -13 -7z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1696 1152q40 0 68 -28t28 -68v-1216q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v288h-544q-40 0 -68 28t-28 68v672q0 40 20 88t48 76l408 408q28 28 76 48t88 20h416q40 0 68 -28t28 -68v-328q68 40 128 40h416zM1152 939l-299 -299h299v299zM512 1323l-299 -299 h299v299zM708 676l316 316v416h-384v-416q0 -40 -28 -68t-68 -28h-416v-640h512v256q0 40 20 88t48 76zM1664 -128v1152h-384v-416q0 -40 -28 -68t-68 -28h-416v-640h896z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1404 151q0 -117 -79 -196t-196 -79q-135 0 -235 100l-777 776q-113 115 -113 271q0 159 110 270t269 111q158 0 273 -113l605 -606q10 -10 10 -22q0 -16 -30.5 -46.5t-46.5 -30.5q-13 0 -23 10l-606 607q-79 77 -181 77q-106 0 -179 -75t-73 -181q0 -105 76 -181 l776 -777q63 -63 145 -63q64 0 106 42t42 106q0 82 -63 145l-581 581q-26 24 -60 24q-29 0 -48 -19t-19 -48q0 -32 25 -59l410 -410q10 -10 10 -22q0 -16 -31 -47t-47 -31q-12 0 -22 10l-410 410q-63 61 -63 149q0 82 57 139t139 57q88 0 149 -63l581 -581q100 -98 100 -235 z" /> +<glyph unicode="" d="M384 0h768v384h-768v-384zM1280 0h128v896q0 14 -10 38.5t-20 34.5l-281 281q-10 10 -34 20t-39 10v-416q0 -40 -28 -68t-68 -28h-576q-40 0 -68 28t-28 68v416h-128v-1280h128v416q0 40 28 68t68 28h832q40 0 68 -28t28 -68v-416zM896 928v320q0 13 -9.5 22.5t-22.5 9.5 h-192q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 22.5zM1536 896v-928q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h928q40 0 88 -20t76 -48l280 -280q28 -28 48 -76t20 -88z" /> +<glyph unicode="" d="M1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1536 192v-128q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1536 704v-128q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1536 1216v-128q0 -26 -19 -45 t-45 -19h-1408q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M384 128q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM384 640q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1792 224v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5 t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5zM384 1152q0 -80 -56 -136t-136 -56t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1792 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z M1792 1248v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M381 -84q0 -80 -54.5 -126t-135.5 -46q-106 0 -172 66l57 88q49 -45 106 -45q29 0 50.5 14.5t21.5 42.5q0 64 -105 56l-26 56q8 10 32.5 43.5t42.5 54t37 38.5v1q-16 0 -48.5 -1t-48.5 -1v-53h-106v152h333v-88l-95 -115q51 -12 81 -49t30 -88zM383 543v-159h-362 q-6 36 -6 54q0 51 23.5 93t56.5 68t66 47.5t56.5 43.5t23.5 45q0 25 -14.5 38.5t-39.5 13.5q-46 0 -81 -58l-85 59q24 51 71.5 79.5t105.5 28.5q73 0 123 -41.5t50 -112.5q0 -50 -34 -91.5t-75 -64.5t-75.5 -50.5t-35.5 -52.5h127v60h105zM1792 224v-192q0 -13 -9.5 -22.5 t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 14 9 23t23 9h1216q13 0 22.5 -9.5t9.5 -22.5zM384 1123v-99h-335v99h107q0 41 0.5 122t0.5 121v12h-2q-8 -17 -50 -54l-71 76l136 127h106v-404h108zM1792 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5 t-9.5 22.5v192q0 14 9 23t23 9h1216q13 0 22.5 -9.5t9.5 -22.5zM1792 1248v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1216q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1216q13 0 22.5 -9.5t9.5 -22.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1760 640q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-1728q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h1728zM483 704q-28 35 -51 80q-48 97 -48 188q0 181 134 309q133 127 393 127q50 0 167 -19q66 -12 177 -48q10 -38 21 -118q14 -123 14 -183q0 -18 -5 -45l-12 -3l-84 6 l-14 2q-50 149 -103 205q-88 91 -210 91q-114 0 -182 -59q-67 -58 -67 -146q0 -73 66 -140t279 -129q69 -20 173 -66q58 -28 95 -52h-743zM990 448h411q7 -39 7 -92q0 -111 -41 -212q-23 -55 -71 -104q-37 -35 -109 -81q-80 -48 -153 -66q-80 -21 -203 -21q-114 0 -195 23 l-140 40q-57 16 -72 28q-8 8 -8 22v13q0 108 -2 156q-1 30 0 68l2 37v44l102 2q15 -34 30 -71t22.5 -56t12.5 -27q35 -57 80 -94q43 -36 105 -57q59 -22 132 -22q64 0 139 27q77 26 122 86q47 61 47 129q0 84 -81 157q-34 29 -137 71z" /> +<glyph unicode="" d="M48 1313q-37 2 -45 4l-3 88q13 1 40 1q60 0 112 -4q132 -7 166 -7q86 0 168 3q116 4 146 5q56 0 86 2l-1 -14l2 -64v-9q-60 -9 -124 -9q-60 0 -79 -25q-13 -14 -13 -132q0 -13 0.5 -32.5t0.5 -25.5l1 -229l14 -280q6 -124 51 -202q35 -59 96 -92q88 -47 177 -47 q104 0 191 28q56 18 99 51q48 36 65 64q36 56 53 114q21 73 21 229q0 79 -3.5 128t-11 122.5t-13.5 159.5l-4 59q-5 67 -24 88q-34 35 -77 34l-100 -2l-14 3l2 86h84l205 -10q76 -3 196 10l18 -2q6 -38 6 -51q0 -7 -4 -31q-45 -12 -84 -13q-73 -11 -79 -17q-15 -15 -15 -41 q0 -7 1.5 -27t1.5 -31q8 -19 22 -396q6 -195 -15 -304q-15 -76 -41 -122q-38 -65 -112 -123q-75 -57 -182 -89q-109 -33 -255 -33q-167 0 -284 46q-119 47 -179 122q-61 76 -83 195q-16 80 -16 237v333q0 188 -17 213q-25 36 -147 39zM1536 -96v64q0 14 -9 23t-23 9h-1472 q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h1472q14 0 23 9t9 23z" /> +<glyph unicode="" horiz-adv-x="1664" d="M512 160v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM512 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 160v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23 v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM512 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 160v192 q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1024 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 544v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192 q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1536 928v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM1664 1248v-1088q0 -66 -47 -113t-113 -47h-1344q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1344q66 0 113 -47t47 -113 z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1190 955l293 293l-107 107l-293 -293zM1637 1248q0 -27 -18 -45l-1286 -1286q-18 -18 -45 -18t-45 18l-198 198q-18 18 -18 45t18 45l1286 1286q18 18 45 18t45 -18l198 -198q18 -18 18 -45zM286 1438l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98zM636 1276 l196 -60l-196 -60l-60 -196l-60 196l-196 60l196 60l60 196zM1566 798l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98zM926 1438l98 -30l-98 -30l-30 -98l-30 98l-98 30l98 30l30 98z" /> +<glyph unicode="" horiz-adv-x="1792" d="M640 128q0 52 -38 90t-90 38t-90 -38t-38 -90t38 -90t90 -38t90 38t38 90zM256 640h384v256h-158q-13 0 -22 -9l-195 -195q-9 -9 -9 -22v-30zM1536 128q0 52 -38 90t-90 38t-90 -38t-38 -90t38 -90t90 -38t90 38t38 90zM1792 1216v-1024q0 -15 -4 -26.5t-13.5 -18.5 t-16.5 -11.5t-23.5 -6t-22.5 -2t-25.5 0t-22.5 0.5q0 -106 -75 -181t-181 -75t-181 75t-75 181h-384q0 -106 -75 -181t-181 -75t-181 75t-75 181h-64q-3 0 -22.5 -0.5t-25.5 0t-22.5 2t-23.5 6t-16.5 11.5t-13.5 18.5t-4 26.5q0 26 19 45t45 19v320q0 8 -0.5 35t0 38 t2.5 34.5t6.5 37t14 30.5t22.5 30l198 198q19 19 50.5 32t58.5 13h160v192q0 26 19 45t45 19h1024q26 0 45 -19t19 -45z" /> +<glyph unicode="" d="M1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103q-111 0 -218 32q59 93 78 164q9 34 54 211q20 -39 73 -67.5t114 -28.5q121 0 216 68.5t147 188.5t52 270q0 114 -59.5 214t-172.5 163t-255 63q-105 0 -196 -29t-154.5 -77t-109 -110.5t-67 -129.5t-21.5 -134 q0 -104 40 -183t117 -111q30 -12 38 20q2 7 8 31t8 30q6 23 -11 43q-51 61 -51 151q0 151 104.5 259.5t273.5 108.5q151 0 235.5 -82t84.5 -213q0 -170 -68.5 -289t-175.5 -119q-61 0 -98 43.5t-23 104.5q8 35 26.5 93.5t30 103t11.5 75.5q0 50 -27 83t-77 33 q-62 0 -105 -57t-43 -142q0 -73 25 -122l-99 -418q-17 -70 -13 -177q-206 91 -333 281t-127 423q0 209 103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-725q85 122 108 210q9 34 53 209q21 -39 73.5 -67t112.5 -28q181 0 295.5 147.5t114.5 373.5q0 84 -35 162.5t-96.5 139t-152.5 97t-197 36.5q-104 0 -194.5 -28.5t-153 -76.5 t-107.5 -109.5t-66.5 -128t-21.5 -132.5q0 -102 39.5 -180t116.5 -110q13 -5 23.5 0t14.5 19q10 44 15 61q6 23 -11 42q-50 62 -50 150q0 150 103.5 256.5t270.5 106.5q149 0 232.5 -81t83.5 -210q0 -168 -67.5 -286t-173.5 -118q-60 0 -97 43.5t-23 103.5q8 34 26.5 92.5 t29.5 102t11 74.5q0 49 -26.5 81.5t-75.5 32.5q-61 0 -103.5 -56.5t-42.5 -139.5q0 -72 24 -121l-98 -414q-24 -100 -7 -254h-183q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960z" /> +<glyph unicode="" d="M829 318q0 -76 -58.5 -112.5t-139.5 -36.5q-41 0 -80.5 9.5t-75.5 28.5t-58 53t-22 78q0 46 25 80t65.5 51.5t82 25t84.5 7.5q20 0 31 -2q2 -1 23 -16.5t26 -19t23 -18t24.5 -22t19 -22.5t17 -26t9 -26.5t4.5 -31.5zM755 863q0 -60 -33 -99.5t-92 -39.5q-53 0 -93 42.5 t-57.5 96.5t-17.5 106q0 61 32 104t92 43q53 0 93.5 -45t58 -101t17.5 -107zM861 1120l88 64h-265q-85 0 -161 -32t-127.5 -98t-51.5 -153q0 -93 64.5 -154.5t158.5 -61.5q22 0 43 3q-13 -29 -13 -54q0 -44 40 -94q-175 -12 -257 -63q-47 -29 -75.5 -73t-28.5 -95 q0 -43 18.5 -77.5t48.5 -56.5t69 -37t77.5 -21t76.5 -6q60 0 120.5 15.5t113.5 46t86 82.5t33 117q0 49 -20 89.5t-49 66.5t-58 47.5t-49 44t-20 44.5t15.5 42.5t37.5 39.5t44 42t37.5 59.5t15.5 82.5q0 60 -22.5 99.5t-72.5 90.5h83zM1152 672h128v64h-128v128h-64v-128 h-128v-64h128v-160h64v160zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M735 740q0 -36 32 -70.5t77.5 -68t90.5 -73.5t77 -104t32 -142q0 -90 -48 -173q-72 -122 -211 -179.5t-298 -57.5q-132 0 -246.5 41.5t-171.5 137.5q-37 60 -37 131q0 81 44.5 150t118.5 115q131 82 404 100q-32 42 -47.5 74t-15.5 73q0 36 21 85q-46 -4 -68 -4 q-148 0 -249.5 96.5t-101.5 244.5q0 82 36 159t99 131q77 66 182.5 98t217.5 32h418l-138 -88h-131q74 -63 112 -133t38 -160q0 -72 -24.5 -129.5t-59 -93t-69.5 -65t-59.5 -61.5t-24.5 -66zM589 836q38 0 78 16.5t66 43.5q53 57 53 159q0 58 -17 125t-48.5 129.5 t-84.5 103.5t-117 41q-42 0 -82.5 -19.5t-65.5 -52.5q-47 -59 -47 -160q0 -46 10 -97.5t31.5 -103t52 -92.5t75 -67t96.5 -26zM591 -37q58 0 111.5 13t99 39t73 73t27.5 109q0 25 -7 49t-14.5 42t-27 41.5t-29.5 35t-38.5 34.5t-36.5 29t-41.5 30t-36.5 26q-16 2 -48 2 q-53 0 -105 -7t-107.5 -25t-97 -46t-68.5 -74.5t-27 -105.5q0 -70 35 -123.5t91.5 -83t119 -44t127.5 -14.5zM1401 839h213v-108h-213v-219h-105v219h-212v108h212v217h105v-217z" /> +<glyph unicode="" horiz-adv-x="1920" d="M768 384h384v96h-128v448h-114l-148 -137l77 -80q42 37 55 57h2v-288h-128v-96zM1280 640q0 -70 -21 -142t-59.5 -134t-101.5 -101t-138 -39t-138 39t-101.5 101t-59.5 134t-21 142t21 142t59.5 134t101.5 101t138 39t138 -39t101.5 -101t59.5 -134t21 -142zM1792 384 v512q-106 0 -181 75t-75 181h-1152q0 -106 -75 -181t-181 -75v-512q106 0 181 -75t75 -181h1152q0 106 75 181t181 75zM1920 1216v-1152q0 -26 -19 -45t-45 -19h-1792q-26 0 -45 19t-19 45v1152q0 26 19 45t45 19h1792q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1024 832q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1024 320q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="640" d="M640 1088v-896q0 -26 -19 -45t-45 -19t-45 19l-448 448q-19 19 -19 45t19 45l448 448q19 19 45 19t45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="640" d="M576 640q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19t-19 45v896q0 26 19 45t45 19t45 -19l448 -448q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M160 0h608v1152h-640v-1120q0 -13 9.5 -22.5t22.5 -9.5zM1536 32v1120h-640v-1152h608q13 0 22.5 9.5t9.5 22.5zM1664 1248v-1216q0 -66 -47 -113t-113 -47h-1344q-66 0 -113 47t-47 113v1216q0 66 47 113t113 47h1344q66 0 113 -47t47 -113z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1024 448q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45zM1024 832q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1024 448q0 -26 -19 -45l-448 -448q-19 -19 -45 -19t-45 19l-448 448q-19 19 -19 45t19 45t45 19h896q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1024 832q0 -26 -19 -45t-45 -19h-896q-26 0 -45 19t-19 45t19 45l448 448q19 19 45 19t45 -19l448 -448q19 -19 19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 826v-794q0 -66 -47 -113t-113 -47h-1472q-66 0 -113 47t-47 113v794q44 -49 101 -87q362 -246 497 -345q57 -42 92.5 -65.5t94.5 -48t110 -24.5h1h1q51 0 110 24.5t94.5 48t92.5 65.5q170 123 498 345q57 39 100 87zM1792 1120q0 -79 -49 -151t-122 -123 q-376 -261 -468 -325q-10 -7 -42.5 -30.5t-54 -38t-52 -32.5t-57.5 -27t-50 -9h-1h-1q-23 0 -50 9t-57.5 27t-52 32.5t-54 38t-42.5 30.5q-91 64 -262 182.5t-205 142.5q-62 42 -117 115.5t-55 136.5q0 78 41.5 130t118.5 52h1472q65 0 112.5 -47t47.5 -113z" /> +<glyph unicode="" d="M349 911v-991h-330v991h330zM370 1217q1 -73 -50.5 -122t-135.5 -49h-2q-82 0 -132 49t-50 122q0 74 51.5 122.5t134.5 48.5t133 -48.5t51 -122.5zM1536 488v-568h-329v530q0 105 -40.5 164.5t-126.5 59.5q-63 0 -105.5 -34.5t-63.5 -85.5q-11 -30 -11 -81v-553h-329 q2 399 2 647t-1 296l-1 48h329v-144h-2q20 32 41 56t56.5 52t87 43.5t114.5 15.5q171 0 275 -113.5t104 -332.5z" /> +<glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61q-172 0 -327 72.5t-264 204.5q-7 10 -6.5 22.5t8.5 20.5l137 138q10 9 25 9q16 -2 23 -12q73 -95 179 -147t225 -52q104 0 198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5t-40.5 198.5t-109.5 163.5 t-163.5 109.5t-198.5 40.5q-98 0 -188 -35.5t-160 -101.5l137 -138q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l130 -129q107 101 244.5 156.5t284.5 55.5q156 0 298 -61t245 -164t164 -245t61 -298z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1771 0q0 -53 -37 -90l-107 -108q-39 -37 -91 -37q-53 0 -90 37l-363 364q-38 36 -38 90q0 53 43 96l-256 256l-126 -126q-14 -14 -34 -14t-34 14q2 -2 12.5 -12t12.5 -13t10 -11.5t10 -13.5t6 -13.5t5.5 -16.5t1.5 -18q0 -38 -28 -68q-3 -3 -16.5 -18t-19 -20.5 t-18.5 -16.5t-22 -15.5t-22 -9t-26 -4.5q-40 0 -68 28l-408 408q-28 28 -28 68q0 13 4.5 26t9 22t15.5 22t16.5 18.5t20.5 19t18 16.5q30 28 68 28q10 0 18 -1.5t16.5 -5.5t13.5 -6t13.5 -10t11.5 -10t13 -12.5t12 -12.5q-14 14 -14 34t14 34l348 348q14 14 34 14t34 -14 q-2 2 -12.5 12t-12.5 13t-10 11.5t-10 13.5t-6 13.5t-5.5 16.5t-1.5 18q0 38 28 68q3 3 16.5 18t19 20.5t18.5 16.5t22 15.5t22 9t26 4.5q40 0 68 -28l408 -408q28 -28 28 -68q0 -13 -4.5 -26t-9 -22t-15.5 -22t-16.5 -18.5t-20.5 -19t-18 -16.5q-30 -28 -68 -28 q-10 0 -18 1.5t-16.5 5.5t-13.5 6t-13.5 10t-11.5 10t-13 12.5t-12 12.5q14 -14 14 -34t-14 -34l-126 -126l256 -256q43 43 96 43q52 0 91 -37l363 -363q37 -39 37 -91z" /> +<glyph unicode="" horiz-adv-x="1792" d="M384 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM576 832q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1004 351l101 382q6 26 -7.5 48.5t-38.5 29.5 t-48 -6.5t-30 -39.5l-101 -382q-60 -5 -107 -43.5t-63 -98.5q-20 -77 20 -146t117 -89t146 20t89 117q16 60 -6 117t-72 91zM1664 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1024 1024q0 53 -37.5 90.5 t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1472 832q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1792 384q0 -261 -141 -483q-19 -29 -54 -29h-1402q-35 0 -54 29 q-141 221 -141 483q0 182 71 348t191 286t286 191t348 71t348 -71t286 -191t191 -286t71 -348z" /> +<glyph unicode="" horiz-adv-x="1792" d="M896 1152q-204 0 -381.5 -69.5t-282 -187.5t-104.5 -255q0 -112 71.5 -213.5t201.5 -175.5l87 -50l-27 -96q-24 -91 -70 -172q152 63 275 171l43 38l57 -6q69 -8 130 -8q204 0 381.5 69.5t282 187.5t104.5 255t-104.5 255t-282 187.5t-381.5 69.5zM1792 640 q0 -174 -120 -321.5t-326 -233t-450 -85.5q-70 0 -145 8q-198 -175 -460 -242q-49 -14 -114 -22h-5q-15 0 -27 10.5t-16 27.5v1q-3 4 -0.5 12t2 10t4.5 9.5l6 9t7 8.5t8 9q7 8 31 34.5t34.5 38t31 39.5t32.5 51t27 59t26 76q-157 89 -247.5 220t-90.5 281q0 174 120 321.5 t326 233t450 85.5t450 -85.5t326 -233t120 -321.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M704 1152q-153 0 -286 -52t-211.5 -141t-78.5 -191q0 -82 53 -158t149 -132l97 -56l-35 -84q34 20 62 39l44 31l53 -10q78 -14 153 -14q153 0 286 52t211.5 141t78.5 191t-78.5 191t-211.5 141t-286 52zM704 1280q191 0 353.5 -68.5t256.5 -186.5t94 -257t-94 -257 t-256.5 -186.5t-353.5 -68.5q-86 0 -176 16q-124 -88 -278 -128q-36 -9 -86 -16h-3q-11 0 -20.5 8t-11.5 21q-1 3 -1 6.5t0.5 6.5t2 6l2.5 5t3.5 5.5t4 5t4.5 5t4 4.5q5 6 23 25t26 29.5t22.5 29t25 38.5t20.5 44q-124 72 -195 177t-71 224q0 139 94 257t256.5 186.5 t353.5 68.5zM1526 111q10 -24 20.5 -44t25 -38.5t22.5 -29t26 -29.5t23 -25q1 -1 4 -4.5t4.5 -5t4 -5t3.5 -5.5l2.5 -5t2 -6t0.5 -6.5t-1 -6.5q-3 -14 -13 -22t-22 -7q-50 7 -86 16q-154 40 -278 128q-90 -16 -176 -16q-271 0 -472 132q58 -4 88 -4q161 0 309 45t264 129 q125 92 192 212t67 254q0 77 -23 152q129 -71 204 -178t75 -230q0 -120 -71 -224.5t-195 -176.5z" /> +<glyph unicode="" horiz-adv-x="896" d="M885 970q18 -20 7 -44l-540 -1157q-13 -25 -42 -25q-4 0 -14 2q-17 5 -25.5 19t-4.5 30l197 808l-406 -101q-4 -1 -12 -1q-18 0 -31 11q-18 15 -13 39l201 825q4 14 16 23t28 9h328q19 0 32 -12.5t13 -29.5q0 -8 -5 -18l-171 -463l396 98q8 2 12 2q19 0 34 -15z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 288v-320q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192h-512v-192h96q40 0 68 -28t28 -68v-320q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192h-512v-192h96q40 0 68 -28t28 -68v-320 q0 -40 -28 -68t-68 -28h-320q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h96v192q0 52 38 90t90 38h512v192h-96q-40 0 -68 28t-28 68v320q0 40 28 68t68 28h320q40 0 68 -28t28 -68v-320q0 -40 -28 -68t-68 -28h-96v-192h512q52 0 90 -38t38 -90v-192h96q40 0 68 -28t28 -68 z" /> +<glyph unicode="" horiz-adv-x="1664" d="M896 708v-580q0 -104 -76 -180t-180 -76t-180 76t-76 180q0 26 19 45t45 19t45 -19t19 -45q0 -50 39 -89t89 -39t89 39t39 89v580q33 11 64 11t64 -11zM1664 681q0 -13 -9.5 -22.5t-22.5 -9.5q-11 0 -23 10q-49 46 -93 69t-102 23q-68 0 -128 -37t-103 -97 q-7 -10 -17.5 -28t-14.5 -24q-11 -17 -28 -17q-18 0 -29 17q-4 6 -14.5 24t-17.5 28q-43 60 -102.5 97t-127.5 37t-127.5 -37t-102.5 -97q-7 -10 -17.5 -28t-14.5 -24q-11 -17 -29 -17q-17 0 -28 17q-4 6 -14.5 24t-17.5 28q-43 60 -103 97t-128 37q-58 0 -102 -23t-93 -69 q-12 -10 -23 -10q-13 0 -22.5 9.5t-9.5 22.5q0 5 1 7q45 183 172.5 319.5t298 204.5t360.5 68q140 0 274.5 -40t246.5 -113.5t194.5 -187t115.5 -251.5q1 -2 1 -7zM896 1408v-98q-42 2 -64 2t-64 -2v98q0 26 19 45t45 19t45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M768 -128h896v640h-416q-40 0 -68 28t-28 68v416h-384v-1152zM1024 1312v64q0 13 -9.5 22.5t-22.5 9.5h-704q-13 0 -22.5 -9.5t-9.5 -22.5v-64q0 -13 9.5 -22.5t22.5 -9.5h704q13 0 22.5 9.5t9.5 22.5zM1280 640h299l-299 299v-299zM1792 512v-672q0 -40 -28 -68t-68 -28 h-960q-40 0 -68 28t-28 68v160h-544q-40 0 -68 28t-28 68v1344q0 40 28 68t68 28h1088q40 0 68 -28t28 -68v-328q21 -13 36 -28l408 -408q28 -28 48 -76t20 -88z" /> +<glyph unicode="" horiz-adv-x="1024" d="M736 960q0 -13 -9.5 -22.5t-22.5 -9.5t-22.5 9.5t-9.5 22.5q0 46 -54 71t-106 25q-13 0 -22.5 9.5t-9.5 22.5t9.5 22.5t22.5 9.5q50 0 99.5 -16t87 -54t37.5 -90zM896 960q0 72 -34.5 134t-90 101.5t-123 62t-136.5 22.5t-136.5 -22.5t-123 -62t-90 -101.5t-34.5 -134 q0 -101 68 -180q10 -11 30.5 -33t30.5 -33q128 -153 141 -298h228q13 145 141 298q10 11 30.5 33t30.5 33q68 79 68 180zM1024 960q0 -155 -103 -268q-45 -49 -74.5 -87t-59.5 -95.5t-34 -107.5q47 -28 47 -82q0 -37 -25 -64q25 -27 25 -64q0 -52 -45 -81q13 -23 13 -47 q0 -46 -31.5 -71t-77.5 -25q-20 -44 -60 -70t-87 -26t-87 26t-60 70q-46 0 -77.5 25t-31.5 71q0 24 13 47q-45 29 -45 81q0 37 25 64q-25 27 -25 64q0 54 47 82q-4 50 -34 107.5t-59.5 95.5t-74.5 87q-103 113 -103 268q0 99 44.5 184.5t117 142t164 89t186.5 32.5 t186.5 -32.5t164 -89t117 -142t44.5 -184.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 352v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-1376v-192q0 -13 -9.5 -22.5t-22.5 -9.5q-12 0 -24 10l-319 320q-9 9 -9 22q0 14 9 23l320 320q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5v-192h1376q13 0 22.5 -9.5t9.5 -22.5zM1792 896q0 -14 -9 -23l-320 -320q-9 -9 -23 -9 q-13 0 -22.5 9.5t-9.5 22.5v192h-1376q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h1376v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1280 608q0 14 -9 23t-23 9h-224v352q0 13 -9.5 22.5t-22.5 9.5h-192q-13 0 -22.5 -9.5t-9.5 -22.5v-352h-224q-13 0 -22.5 -9.5t-9.5 -22.5q0 -14 9 -23l352 -352q9 -9 23 -9t23 9l351 351q10 12 10 24zM1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088 q-185 0 -316.5 131.5t-131.5 316.5q0 130 70 240t188 165q-2 30 -2 43q0 212 150 362t362 150q156 0 285.5 -87t188.5 -231q71 62 166 62q106 0 181 -75t75 -181q0 -76 -41 -138q130 -31 213.5 -135.5t83.5 -238.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1280 672q0 14 -9 23l-352 352q-9 9 -23 9t-23 -9l-351 -351q-10 -12 -10 -24q0 -14 9 -23t23 -9h224v-352q0 -13 9.5 -22.5t22.5 -9.5h192q13 0 22.5 9.5t9.5 22.5v352h224q13 0 22.5 9.5t9.5 22.5zM1920 384q0 -159 -112.5 -271.5t-271.5 -112.5h-1088 q-185 0 -316.5 131.5t-131.5 316.5q0 130 70 240t188 165q-2 30 -2 43q0 212 150 362t362 150q156 0 285.5 -87t188.5 -231q71 62 166 62q106 0 181 -75t75 -181q0 -76 -41 -138q130 -31 213.5 -135.5t83.5 -238.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M384 192q0 -26 -19 -45t-45 -19t-45 19t-19 45t19 45t45 19t45 -19t19 -45zM1408 131q0 -121 -73 -190t-194 -69h-874q-121 0 -194 69t-73 190q0 68 5.5 131t24 138t47.5 132.5t81 103t120 60.5q-22 -52 -22 -120v-203q-58 -20 -93 -70t-35 -111q0 -80 56 -136t136 -56 t136 56t56 136q0 61 -35.5 111t-92.5 70v203q0 62 25 93q132 -104 295 -104t295 104q25 -31 25 -93v-64q-106 0 -181 -75t-75 -181v-89q-32 -29 -32 -71q0 -40 28 -68t68 -28t68 28t28 68q0 42 -32 71v89q0 52 38 90t90 38t90 -38t38 -90v-89q-32 -29 -32 -71q0 -40 28 -68 t68 -28t68 28t28 68q0 42 -32 71v89q0 68 -34.5 127.5t-93.5 93.5q0 10 0.5 42.5t0 48t-2.5 41.5t-7 47t-13 40q68 -15 120 -60.5t81 -103t47.5 -132.5t24 -138t5.5 -131zM1088 1024q0 -159 -112.5 -271.5t-271.5 -112.5t-271.5 112.5t-112.5 271.5t112.5 271.5t271.5 112.5 t271.5 -112.5t112.5 -271.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1280 832q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 832q0 -62 -35.5 -111t-92.5 -70v-395q0 -159 -131.5 -271.5t-316.5 -112.5t-316.5 112.5t-131.5 271.5v132q-164 20 -274 128t-110 252v512q0 26 19 45t45 19q6 0 16 -2q17 30 47 48 t65 18q53 0 90.5 -37.5t37.5 -90.5t-37.5 -90.5t-90.5 -37.5q-33 0 -64 18v-402q0 -106 94 -181t226 -75t226 75t94 181v402q-31 -18 -64 -18q-53 0 -90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5q35 0 65 -18t47 -48q10 2 16 2q26 0 45 -19t19 -45v-512q0 -144 -110 -252 t-274 -128v-132q0 -106 94 -181t226 -75t226 75t94 181v395q-57 21 -92.5 70t-35.5 111q0 80 56 136t136 56t136 -56t56 -136z" /> +<glyph unicode="" horiz-adv-x="1792" d="M640 1152h512v128h-512v-128zM288 1152v-1280h-64q-92 0 -158 66t-66 158v832q0 92 66 158t158 66h64zM1408 1152v-1280h-1024v1280h128v160q0 40 28 68t68 28h576q40 0 68 -28t28 -68v-160h128zM1792 928v-832q0 -92 -66 -158t-158 -66h-64v1280h64q92 0 158 -66 t66 -158z" /> +<glyph unicode="" horiz-adv-x="1792" d="M912 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM1728 128q0 -52 -38 -90t-90 -38h-448q0 -106 -75 -181t-181 -75t-181 75t-75 181h-448q-52 0 -90 38t-38 90q50 42 91 88t85 119.5t74.5 158.5 t50 206t19.5 260q0 152 117 282.5t307 158.5q-8 19 -8 39q0 40 28 68t68 28t68 -28t28 -68q0 -20 -8 -39q190 -28 307 -158.5t117 -282.5q0 -139 19.5 -260t50 -206t74.5 -158.5t85 -119.5t91 -88z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1664 896q0 80 -56 136t-136 56h-64v-384h64q80 0 136 56t56 136zM0 128h1792q0 -106 -75 -181t-181 -75h-1280q-106 0 -181 75t-75 181zM1856 896q0 -159 -112.5 -271.5t-271.5 -112.5h-64v-32q0 -92 -66 -158t-158 -66h-704q-92 0 -158 66t-66 158v736q0 26 19 45 t45 19h1152q159 0 271.5 -112.5t112.5 -271.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M640 1472v-640q0 -61 -35.5 -111t-92.5 -70v-779q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v779q-57 20 -92.5 70t-35.5 111v640q0 26 19 45t45 19t45 -19t19 -45v-416q0 -26 19 -45t45 -19t45 19t19 45v416q0 26 19 45t45 19t45 -19t19 -45v-416q0 -26 19 -45 t45 -19t45 19t19 45v416q0 26 19 45t45 19t45 -19t19 -45zM1408 1472v-1600q0 -52 -38 -90t-90 -38h-128q-52 0 -90 38t-38 90v512h-224q-13 0 -22.5 9.5t-9.5 22.5v800q0 132 94 226t226 94h256q26 0 45 -19t19 -45z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M384 736q0 14 9 23t23 9h704q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-704q-14 0 -23 9t-9 23v64zM1120 512q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-704q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h704zM1120 256q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-704 q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h704z" /> +<glyph unicode="" horiz-adv-x="1408" d="M384 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 992v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 1248v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 -128h384v1536h-1152v-1536h384v224q0 13 9.5 22.5t22.5 9.5h320q13 0 22.5 -9.5t9.5 -22.5v-224zM1408 1472v-1664q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1664q0 26 19 45t45 19h1280q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1408" d="M384 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM384 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M1152 224v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM896 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M640 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 480v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5zM1152 736v-64q0 -13 -9.5 -22.5t-22.5 -9.5h-64q-13 0 -22.5 9.5t-9.5 22.5v64q0 13 9.5 22.5t22.5 9.5h64q13 0 22.5 -9.5t9.5 -22.5z M896 -128h384v1152h-256v-32q0 -40 -28 -68t-68 -28h-448q-40 0 -68 28t-28 68v32h-256v-1152h384v224q0 13 9.5 22.5t22.5 9.5h320q13 0 22.5 -9.5t9.5 -22.5v-224zM896 1056v320q0 13 -9.5 22.5t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-96h-128v96q0 13 -9.5 22.5 t-22.5 9.5h-64q-13 0 -22.5 -9.5t-9.5 -22.5v-320q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5v96h128v-96q0 -13 9.5 -22.5t22.5 -9.5h64q13 0 22.5 9.5t9.5 22.5zM1408 1088v-1280q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1280q0 26 19 45t45 19h320 v288q0 40 28 68t68 28h448q40 0 68 -28t28 -68v-288h320q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1920" d="M640 128q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM256 640h384v256h-158q-14 -2 -22 -9l-195 -195q-7 -12 -9 -22v-30zM1536 128q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5 t90.5 37.5t37.5 90.5zM1664 800v192q0 14 -9 23t-23 9h-224v224q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-224h-224q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h224v-224q0 -14 9 -23t23 -9h192q14 0 23 9t9 23v224h224q14 0 23 9t9 23zM1920 1344v-1152 q0 -26 -19 -45t-45 -19h-192q0 -106 -75 -181t-181 -75t-181 75t-75 181h-384q0 -106 -75 -181t-181 -75t-181 75t-75 181h-128q-26 0 -45 19t-19 45t19 45t45 19v416q0 26 13 58t32 51l198 198q19 19 51 32t58 13h160v320q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1280 416v192q0 14 -9 23t-23 9h-224v224q0 14 -9 23t-23 9h-192q-14 0 -23 -9t-9 -23v-224h-224q-14 0 -23 -9t-9 -23v-192q0 -14 9 -23t23 -9h224v-224q0 -14 9 -23t23 -9h192q14 0 23 9t9 23v224h224q14 0 23 9t9 23zM640 1152h512v128h-512v-128zM256 1152v-1280h-32 q-92 0 -158 66t-66 158v832q0 92 66 158t158 66h32zM1440 1152v-1280h-1088v1280h160v160q0 40 28 68t68 28h576q40 0 68 -28t28 -68v-160h160zM1792 928v-832q0 -92 -66 -158t-158 -66h-32v1280h32q92 0 158 -66t66 -158z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1920 576q-1 -32 -288 -96l-352 -32l-224 -64h-64l-293 -352h69q26 0 45 -4.5t19 -11.5t-19 -11.5t-45 -4.5h-96h-160h-64v32h64v416h-160l-192 -224h-96l-32 32v192h32v32h128v8l-192 24v128l192 24v8h-128v32h-32v192l32 32h96l192 -224h160v416h-64v32h64h160h96 q26 0 45 -4.5t19 -11.5t-19 -11.5t-45 -4.5h-69l293 -352h64l224 -64l352 -32q261 -58 287 -93z" /> +<glyph unicode="" horiz-adv-x="1664" d="M640 640v384h-256v-256q0 -53 37.5 -90.5t90.5 -37.5h128zM1664 192v-192h-1152v192l128 192h-128q-159 0 -271.5 112.5t-112.5 271.5v320l-64 64l32 128h480l32 128h960l32 -192l-64 -32v-800z" /> +<glyph unicode="" d="M1280 192v896q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-320h-512v320q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-896q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v320h512v-320q0 -26 19 -45t45 -19h128q26 0 45 19t19 45zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-320v320q0 26 -19 45t-45 19h-128q-26 0 -45 -19t-19 -45v-320h-320q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h320v-320q0 -26 19 -45t45 -19h128q26 0 45 19t19 45v320h320q26 0 45 19t19 45zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M627 160q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23zM1011 160q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23 t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1024" d="M595 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23zM979 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23 l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1075 224q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23zM1075 608q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393 q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1075 672q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23zM1075 1056q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23 t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="640" d="M627 992q0 -13 -10 -23l-393 -393l393 -393q10 -10 10 -23t-10 -23l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="640" d="M595 576q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1075 352q0 -13 -10 -23l-50 -50q-10 -10 -23 -10t-23 10l-393 393l-393 -393q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l466 -466q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1075 800q0 -13 -10 -23l-466 -466q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l393 -393l393 393q10 10 23 10t23 -10l50 -50q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1792 544v832q0 13 -9.5 22.5t-22.5 9.5h-1600q-13 0 -22.5 -9.5t-9.5 -22.5v-832q0 -13 9.5 -22.5t22.5 -9.5h1600q13 0 22.5 9.5t9.5 22.5zM1920 1376v-1088q0 -66 -47 -113t-113 -47h-544q0 -37 16 -77.5t32 -71t16 -43.5q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19 t-19 45q0 14 16 44t32 70t16 78h-544q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h1600q66 0 113 -47t47 -113z" /> +<glyph unicode="" horiz-adv-x="1920" d="M416 256q-66 0 -113 47t-47 113v704q0 66 47 113t113 47h1088q66 0 113 -47t47 -113v-704q0 -66 -47 -113t-113 -47h-1088zM384 1120v-704q0 -13 9.5 -22.5t22.5 -9.5h1088q13 0 22.5 9.5t9.5 22.5v704q0 13 -9.5 22.5t-22.5 9.5h-1088q-13 0 -22.5 -9.5t-9.5 -22.5z M1760 192h160v-96q0 -40 -47 -68t-113 -28h-1600q-66 0 -113 28t-47 68v96h160h1600zM1040 96q16 0 16 16t-16 16h-160q-16 0 -16 -16t16 -16h160z" /> +<glyph unicode="" horiz-adv-x="1152" d="M640 128q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1024 288v960q0 13 -9.5 22.5t-22.5 9.5h-832q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h832q13 0 22.5 9.5t9.5 22.5zM1152 1248v-1088q0 -66 -47 -113t-113 -47h-832 q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h832q66 0 113 -47t47 -113z" /> +<glyph unicode="" horiz-adv-x="768" d="M464 128q0 33 -23.5 56.5t-56.5 23.5t-56.5 -23.5t-23.5 -56.5t23.5 -56.5t56.5 -23.5t56.5 23.5t23.5 56.5zM672 288v704q0 13 -9.5 22.5t-22.5 9.5h-512q-13 0 -22.5 -9.5t-9.5 -22.5v-704q0 -13 9.5 -22.5t22.5 -9.5h512q13 0 22.5 9.5t9.5 22.5zM480 1136 q0 16 -16 16h-160q-16 0 -16 -16t16 -16h160q16 0 16 16zM768 1152v-1024q0 -52 -38 -90t-90 -38h-512q-52 0 -90 38t-38 90v1024q0 52 38 90t90 38h512q52 0 90 -38t38 -90z" /> +<glyph unicode="" d="M768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103 t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M768 576v-384q0 -80 -56 -136t-136 -56h-384q-80 0 -136 56t-56 136v704q0 104 40.5 198.5t109.5 163.5t163.5 109.5t198.5 40.5h64q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-64q-106 0 -181 -75t-75 -181v-32q0 -40 28 -68t68 -28h224q80 0 136 -56t56 -136z M1664 576v-384q0 -80 -56 -136t-136 -56h-384q-80 0 -136 56t-56 136v704q0 104 40.5 198.5t109.5 163.5t163.5 109.5t198.5 40.5h64q26 0 45 -19t19 -45v-128q0 -26 -19 -45t-45 -19h-64q-106 0 -181 -75t-75 -181v-32q0 -40 28 -68t68 -28h224q80 0 136 -56t56 -136z" /> +<glyph unicode="" horiz-adv-x="1664" d="M768 1216v-704q0 -104 -40.5 -198.5t-109.5 -163.5t-163.5 -109.5t-198.5 -40.5h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64q106 0 181 75t75 181v32q0 40 -28 68t-68 28h-224q-80 0 -136 56t-56 136v384q0 80 56 136t136 56h384q80 0 136 -56t56 -136zM1664 1216 v-704q0 -104 -40.5 -198.5t-109.5 -163.5t-163.5 -109.5t-198.5 -40.5h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64q106 0 181 75t75 181v32q0 40 -28 68t-68 28h-224q-80 0 -136 56t-56 136v384q0 80 56 136t136 56h384q80 0 136 -56t56 -136z" /> +<glyph unicode="" horiz-adv-x="1568" d="M496 192q0 -60 -42.5 -102t-101.5 -42q-60 0 -102 42t-42 102t42 102t102 42q59 0 101.5 -42t42.5 -102zM928 0q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM320 640q0 -66 -47 -113t-113 -47t-113 47t-47 113 t47 113t113 47t113 -47t47 -113zM1360 192q0 -46 -33 -79t-79 -33t-79 33t-33 79t33 79t79 33t79 -33t33 -79zM528 1088q0 -73 -51.5 -124.5t-124.5 -51.5t-124.5 51.5t-51.5 124.5t51.5 124.5t124.5 51.5t124.5 -51.5t51.5 -124.5zM992 1280q0 -80 -56 -136t-136 -56 t-136 56t-56 136t56 136t136 56t136 -56t56 -136zM1536 640q0 -40 -28 -68t-68 -28t-68 28t-28 68t28 68t68 28t68 -28t28 -68zM1328 1088q0 -33 -23.5 -56.5t-56.5 -23.5t-56.5 23.5t-23.5 56.5t23.5 56.5t56.5 23.5t56.5 -23.5t23.5 -56.5z" /> +<glyph unicode="" d="M1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 416q0 -166 -127 -451q-3 -7 -10.5 -24t-13.5 -30t-13 -22q-12 -17 -28 -17q-15 0 -23.5 10t-8.5 25q0 9 2.5 26.5t2.5 23.5q5 68 5 123q0 101 -17.5 181t-48.5 138.5t-80 101t-105.5 69.5t-133 42.5t-154 21.5t-175.5 6h-224v-256q0 -26 -19 -45t-45 -19t-45 19 l-512 512q-19 19 -19 45t19 45l512 512q19 19 45 19t45 -19t19 -45v-256h224q713 0 875 -403q53 -134 53 -333z" /> +<glyph unicode="" horiz-adv-x="1664" d="M640 320q0 -40 -12.5 -82t-43 -76t-72.5 -34t-72.5 34t-43 76t-12.5 82t12.5 82t43 76t72.5 34t72.5 -34t43 -76t12.5 -82zM1280 320q0 -40 -12.5 -82t-43 -76t-72.5 -34t-72.5 34t-43 76t-12.5 82t12.5 82t43 76t72.5 34t72.5 -34t43 -76t12.5 -82zM1440 320 q0 120 -69 204t-187 84q-41 0 -195 -21q-71 -11 -157 -11t-157 11q-152 21 -195 21q-118 0 -187 -84t-69 -204q0 -88 32 -153.5t81 -103t122 -60t140 -29.5t149 -7h168q82 0 149 7t140 29.5t122 60t81 103t32 153.5zM1664 496q0 -207 -61 -331q-38 -77 -105.5 -133t-141 -86 t-170 -47.5t-171.5 -22t-167 -4.5q-78 0 -142 3t-147.5 12.5t-152.5 30t-137 51.5t-121 81t-86 115q-62 123 -62 331q0 237 136 396q-27 82 -27 170q0 116 51 218q108 0 190 -39.5t189 -123.5q147 35 309 35q148 0 280 -32q105 82 187 121t189 39q51 -102 51 -218 q0 -87 -27 -168q136 -160 136 -398z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1536 224v704q0 40 -28 68t-68 28h-704q-40 0 -68 28t-28 68v64q0 40 -28 68t-68 28h-320q-40 0 -68 -28t-28 -68v-960q0 -40 28 -68t68 -28h1216q40 0 68 28t28 68zM1664 928v-704q0 -92 -66 -158t-158 -66h-1216q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320 q92 0 158 -66t66 -158v-32h672q92 0 158 -66t66 -158z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1781 605q0 35 -53 35h-1088q-40 0 -85.5 -21.5t-71.5 -52.5l-294 -363q-18 -24 -18 -40q0 -35 53 -35h1088q40 0 86 22t71 53l294 363q18 22 18 39zM640 768h768v160q0 40 -28 68t-68 28h-576q-40 0 -68 28t-28 68v64q0 40 -28 68t-68 28h-320q-40 0 -68 -28t-28 -68 v-853l256 315q44 53 116 87.5t140 34.5zM1909 605q0 -62 -46 -120l-295 -363q-43 -53 -116 -87.5t-140 -34.5h-1088q-92 0 -158 66t-66 158v960q0 92 66 158t158 66h320q92 0 158 -66t66 -158v-32h544q92 0 158 -66t66 -158v-160h192q54 0 99 -24.5t67 -70.5q15 -32 15 -68z " /> +<glyph unicode="" horiz-adv-x="1792" /> +<glyph unicode="" horiz-adv-x="1792" /> +<glyph unicode="" d="M1134 461q-37 -121 -138 -195t-228 -74t-228 74t-138 195q-8 25 4 48.5t38 31.5q25 8 48.5 -4t31.5 -38q25 -80 92.5 -129.5t151.5 -49.5t151.5 49.5t92.5 129.5q8 26 32 38t49 4t37 -31.5t4 -48.5zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5 t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5 t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1134 307q8 -25 -4 -48.5t-37 -31.5t-49 4t-32 38q-25 80 -92.5 129.5t-151.5 49.5t-151.5 -49.5t-92.5 -129.5q-8 -26 -31.5 -38t-48.5 -4q-26 8 -38 31.5t-4 48.5q37 121 138 195t228 74t228 -74t138 -195zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204 t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1152 448q0 -26 -19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h640q26 0 45 -19t19 -45zM640 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1152 896q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5t37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M832 448v128q0 14 -9 23t-23 9h-192v192q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-192h-192q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h192v-192q0 -14 9 -23t23 -9h128q14 0 23 9t9 23v192h192q14 0 23 9t9 23zM1408 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5 t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 640q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1920 512q0 -212 -150 -362t-362 -150q-192 0 -338 128h-220q-146 -128 -338 -128q-212 0 -362 150 t-150 362t150 362t362 150h896q212 0 362 -150t150 -362z" /> +<glyph unicode="" horiz-adv-x="1920" d="M384 368v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM512 624v-96q0 -16 -16 -16h-224q-16 0 -16 16v96q0 16 16 16h224q16 0 16 -16zM384 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1408 368v-96q0 -16 -16 -16 h-864q-16 0 -16 16v96q0 16 16 16h864q16 0 16 -16zM768 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM640 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1024 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16 h96q16 0 16 -16zM896 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1280 624v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1664 368v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1152 880v-96 q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1408 880v-96q0 -16 -16 -16h-96q-16 0 -16 16v96q0 16 16 16h96q16 0 16 -16zM1664 880v-352q0 -16 -16 -16h-224q-16 0 -16 16v96q0 16 16 16h112v240q0 16 16 16h96q16 0 16 -16zM1792 128v896h-1664v-896 h1664zM1920 1024v-896q0 -53 -37.5 -90.5t-90.5 -37.5h-1664q-53 0 -90.5 37.5t-37.5 90.5v896q0 53 37.5 90.5t90.5 37.5h1664q53 0 90.5 -37.5t37.5 -90.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1664 491v616q-169 -91 -306 -91q-82 0 -145 32q-100 49 -184 76.5t-178 27.5q-173 0 -403 -127v-599q245 113 433 113q55 0 103.5 -7.5t98 -26t77 -31t82.5 -39.5l28 -14q44 -22 101 -22q120 0 293 92zM320 1280q0 -35 -17.5 -64t-46.5 -46v-1266q0 -14 -9 -23t-23 -9 h-64q-14 0 -23 9t-9 23v1266q-29 17 -46.5 46t-17.5 64q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -39 -35 -57q-10 -5 -17 -9q-218 -116 -369 -116q-88 0 -158 35l-28 14q-64 33 -99 48t-91 29t-114 14q-102 0 -235.5 -44t-228.5 -102 q-15 -9 -33 -9q-16 0 -32 8q-32 19 -32 56v742q0 35 31 55q35 21 78.5 42.5t114 52t152.5 49.5t155 19q112 0 209 -31t209 -86q38 -19 89 -19q122 0 310 112q22 12 31 17q31 16 62 -2q31 -20 31 -55z" /> +<glyph unicode="" horiz-adv-x="1792" d="M832 536v192q-181 -16 -384 -117v-185q205 96 384 110zM832 954v197q-172 -8 -384 -126v-189q215 111 384 118zM1664 491v184q-235 -116 -384 -71v224q-20 6 -39 15q-5 3 -33 17t-34.5 17t-31.5 15t-34.5 15.5t-32.5 13t-36 12.5t-35 8.5t-39.5 7.5t-39.5 4t-44 2 q-23 0 -49 -3v-222h19q102 0 192.5 -29t197.5 -82q19 -9 39 -15v-188q42 -17 91 -17q120 0 293 92zM1664 918v189q-169 -91 -306 -91q-45 0 -78 8v-196q148 -42 384 90zM320 1280q0 -35 -17.5 -64t-46.5 -46v-1266q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v1266 q-29 17 -46.5 46t-17.5 64q0 53 37.5 90.5t90.5 37.5t90.5 -37.5t37.5 -90.5zM1792 1216v-763q0 -39 -35 -57q-10 -5 -17 -9q-218 -116 -369 -116q-88 0 -158 35l-28 14q-64 33 -99 48t-91 29t-114 14q-102 0 -235.5 -44t-228.5 -102q-15 -9 -33 -9q-16 0 -32 8 q-32 19 -32 56v742q0 35 31 55q35 21 78.5 42.5t114 52t152.5 49.5t155 19q112 0 209 -31t209 -86q38 -19 89 -19q122 0 310 112q22 12 31 17q31 16 62 -2q31 -20 31 -55z" /> +<glyph unicode="" horiz-adv-x="1664" d="M585 553l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23t-10 -23zM1664 96v-64q0 -14 -9 -23t-23 -9h-960q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h960q14 0 23 -9 t9 -23z" /> +<glyph unicode="" horiz-adv-x="1920" d="M617 137l-50 -50q-10 -10 -23 -10t-23 10l-466 466q-10 10 -10 23t10 23l466 466q10 10 23 10t23 -10l50 -50q10 -10 10 -23t-10 -23l-393 -393l393 -393q10 -10 10 -23t-10 -23zM1208 1204l-373 -1291q-4 -13 -15.5 -19.5t-23.5 -2.5l-62 17q-13 4 -19.5 15.5t-2.5 24.5 l373 1291q4 13 15.5 19.5t23.5 2.5l62 -17q13 -4 19.5 -15.5t2.5 -24.5zM1865 553l-466 -466q-10 -10 -23 -10t-23 10l-50 50q-10 10 -10 23t10 23l393 393l-393 393q-10 10 -10 23t10 23l50 50q10 10 23 10t23 -10l466 -466q10 -10 10 -23t-10 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M640 454v-70q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-512 512q-19 19 -19 45t19 45l512 512q29 31 70 14q39 -17 39 -59v-69l-397 -398q-19 -19 -19 -45t19 -45zM1792 416q0 -58 -17 -133.5t-38.5 -138t-48 -125t-40.5 -90.5l-20 -40q-8 -17 -28 -17q-6 0 -9 1 q-25 8 -23 34q43 400 -106 565q-64 71 -170.5 110.5t-267.5 52.5v-251q0 -42 -39 -59q-13 -5 -25 -5q-27 0 -45 19l-512 512q-19 19 -19 45t19 45l512 512q29 31 70 14q39 -17 39 -59v-262q411 -28 599 -221q169 -173 169 -509z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1186 579l257 250l-356 52l-66 10l-30 60l-159 322v-963l59 -31l318 -168l-60 355l-12 66zM1638 841l-363 -354l86 -500q5 -33 -6 -51.5t-34 -18.5q-17 0 -40 12l-449 236l-449 -236q-23 -12 -40 -12q-23 0 -34 18.5t-6 51.5l86 500l-364 354q-32 32 -23 59.5t54 34.5 l502 73l225 455q20 41 49 41q28 0 49 -41l225 -455l502 -73q45 -7 54 -34.5t-24 -59.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1401 1187l-640 -1280q-17 -35 -57 -35q-5 0 -15 2q-22 5 -35.5 22.5t-13.5 39.5v576h-576q-22 0 -39.5 13.5t-22.5 35.5t4 42t29 30l1280 640q13 7 29 7q27 0 45 -19q15 -14 18.5 -34.5t-6.5 -39.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M557 256h595v595zM512 301l595 595h-595v-595zM1664 224v-192q0 -14 -9 -23t-23 -9h-224v-224q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v224h-864q-14 0 -23 9t-9 23v864h-224q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h224v224q0 14 9 23t23 9h192q14 0 23 -9t9 -23 v-224h851l246 247q10 9 23 9t23 -9q9 -10 9 -23t-9 -23l-247 -246v-851h224q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1024" d="M288 64q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM288 1216q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM928 1088q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM1024 1088q0 -52 -26 -96.5t-70 -69.5 q-2 -287 -226 -414q-68 -38 -203 -81q-128 -40 -169.5 -71t-41.5 -100v-26q44 -25 70 -69.5t26 -96.5q0 -80 -56 -136t-136 -56t-136 56t-56 136q0 52 26 96.5t70 69.5v820q-44 25 -70 69.5t-26 96.5q0 80 56 136t136 56t136 -56t56 -136q0 -52 -26 -96.5t-70 -69.5v-497 q54 26 154 57q55 17 87.5 29.5t70.5 31t59 39.5t40.5 51t28 69.5t8.5 91.5q-44 25 -70 69.5t-26 96.5q0 80 56 136t136 56t136 -56t56 -136z" /> +<glyph unicode="" horiz-adv-x="1664" d="M439 265l-256 -256q-10 -9 -23 -9q-12 0 -23 9q-9 10 -9 23t9 23l256 256q10 9 23 9t23 -9q9 -10 9 -23t-9 -23zM608 224v-320q0 -14 -9 -23t-23 -9t-23 9t-9 23v320q0 14 9 23t23 9t23 -9t9 -23zM384 448q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9t-9 23t9 23t23 9h320 q14 0 23 -9t9 -23zM1648 320q0 -120 -85 -203l-147 -146q-83 -83 -203 -83q-121 0 -204 85l-334 335q-21 21 -42 56l239 18l273 -274q27 -27 68 -27.5t68 26.5l147 146q28 28 28 67q0 40 -28 68l-274 275l18 239q35 -21 56 -42l336 -336q84 -86 84 -204zM1031 1044l-239 -18 l-273 274q-28 28 -68 28q-39 0 -68 -27l-147 -146q-28 -28 -28 -67q0 -40 28 -68l274 -274l-18 -240q-35 21 -56 42l-336 336q-84 86 -84 204q0 120 85 203l147 146q83 83 203 83q121 0 204 -85l334 -335q21 -21 42 -56zM1664 960q0 -14 -9 -23t-23 -9h-320q-14 0 -23 9 t-9 23t9 23t23 9h320q14 0 23 -9t9 -23zM1120 1504v-320q0 -14 -9 -23t-23 -9t-23 9t-9 23v320q0 14 9 23t23 9t23 -9t9 -23zM1527 1353l-256 -256q-11 -9 -23 -9t-23 9q-9 10 -9 23t9 23l256 256q10 9 23 9t23 -9q9 -10 9 -23t-9 -23z" /> +<glyph unicode="" horiz-adv-x="1024" d="M704 280v-240q0 -16 -12 -28t-28 -12h-240q-16 0 -28 12t-12 28v240q0 16 12 28t28 12h240q16 0 28 -12t12 -28zM1020 880q0 -54 -15.5 -101t-35 -76.5t-55 -59.5t-57.5 -43.5t-61 -35.5q-41 -23 -68.5 -65t-27.5 -67q0 -17 -12 -32.5t-28 -15.5h-240q-15 0 -25.5 18.5 t-10.5 37.5v45q0 83 65 156.5t143 108.5q59 27 84 56t25 76q0 42 -46.5 74t-107.5 32q-65 0 -108 -29q-35 -25 -107 -115q-13 -16 -31 -16q-12 0 -25 8l-164 125q-13 10 -15.5 25t5.5 28q160 266 464 266q80 0 161 -31t146 -83t106 -127.5t41 -158.5z" /> +<glyph unicode="" horiz-adv-x="640" d="M640 192v-128q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h64v384h-64q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h384q26 0 45 -19t19 -45v-576h64q26 0 45 -19t19 -45zM512 1344v-192q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v192 q0 26 19 45t45 19h256q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="640" d="M512 288v-224q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v224q0 26 19 45t45 19h256q26 0 45 -19t19 -45zM542 1344l-28 -768q-1 -26 -20.5 -45t-45.5 -19h-256q-26 0 -45.5 19t-20.5 45l-28 768q-1 26 17.5 45t44.5 19h320q26 0 44.5 -19t17.5 -45z" /> +<glyph unicode="" d="M897 167v-167h-248l-159 252l-24 42q-8 9 -11 21h-3l-9 -21q-10 -20 -25 -44l-155 -250h-258v167h128l197 291l-185 272h-137v168h276l139 -228q2 -4 23 -42q8 -9 11 -21h3q3 9 11 21l25 42l140 228h257v-168h-125l-184 -267l204 -296h109zM1534 846v-206h-514l-3 27 q-4 28 -4 46q0 64 26 117t65 86.5t84 65t84 54.5t65 54t26 64q0 38 -29.5 62.5t-70.5 24.5q-51 0 -97 -39q-14 -11 -36 -38l-105 92q26 37 63 66q83 65 188 65q110 0 178 -59.5t68 -158.5q0 -56 -24.5 -103t-62 -76.5t-81.5 -58.5t-82 -50.5t-65.5 -51.5t-30.5 -63h232v80 h126z" /> +<glyph unicode="" d="M897 167v-167h-248l-159 252l-24 42q-8 9 -11 21h-3l-9 -21q-10 -20 -25 -44l-155 -250h-258v167h128l197 291l-185 272h-137v168h276l139 -228q2 -4 23 -42q8 -9 11 -21h3q3 9 11 21l25 42l140 228h257v-168h-125l-184 -267l204 -296h109zM1536 -50v-206h-514l-4 27 q-3 45 -3 46q0 64 26 117t65 86.5t84 65t84 54.5t65 54t26 64q0 38 -29.5 62.5t-70.5 24.5q-51 0 -97 -39q-14 -11 -36 -38l-105 92q26 37 63 66q80 65 188 65q110 0 178 -59.5t68 -158.5q0 -66 -34.5 -118.5t-84 -86t-99.5 -62.5t-87 -63t-41 -73h232v80h126z" /> +<glyph unicode="" horiz-adv-x="1920" d="M896 128l336 384h-768l-336 -384h768zM1909 1205q15 -34 9.5 -71.5t-30.5 -65.5l-896 -1024q-38 -44 -96 -44h-768q-38 0 -69.5 20.5t-47.5 54.5q-15 34 -9.5 71.5t30.5 65.5l896 1024q38 44 96 44h768q38 0 69.5 -20.5t47.5 -54.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1664 438q0 -81 -44.5 -135t-123.5 -54q-41 0 -77.5 17.5t-59 38t-56.5 38t-71 17.5q-110 0 -110 -124q0 -39 16 -115t15 -115v-5q-22 0 -33 -1q-34 -3 -97.5 -11.5t-115.5 -13.5t-98 -5q-61 0 -103 26.5t-42 83.5q0 37 17.5 71t38 56.5t38 59t17.5 77.5q0 79 -54 123.5 t-135 44.5q-84 0 -143 -45.5t-59 -127.5q0 -43 15 -83t33.5 -64.5t33.5 -53t15 -50.5q0 -45 -46 -89q-37 -35 -117 -35q-95 0 -245 24q-9 2 -27.5 4t-27.5 4l-13 2q-1 0 -3 1q-2 0 -2 1v1024q2 -1 17.5 -3.5t34 -5t21.5 -3.5q150 -24 245 -24q80 0 117 35q46 44 46 89 q0 22 -15 50.5t-33.5 53t-33.5 64.5t-15 83q0 82 59 127.5t144 45.5q80 0 134 -44.5t54 -123.5q0 -41 -17.5 -77.5t-38 -59t-38 -56.5t-17.5 -71q0 -57 42 -83.5t103 -26.5q64 0 180 15t163 17v-2q-1 -2 -3.5 -17.5t-5 -34t-3.5 -21.5q-24 -150 -24 -245q0 -80 35 -117 q44 -46 89 -46q22 0 50.5 15t53 33.5t64.5 33.5t83 15q82 0 127.5 -59t45.5 -143z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1152 832v-128q0 -221 -147.5 -384.5t-364.5 -187.5v-132h256q26 0 45 -19t19 -45t-19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h256v132q-217 24 -364.5 187.5t-147.5 384.5v128q0 26 19 45t45 19t45 -19t19 -45v-128q0 -185 131.5 -316.5t316.5 -131.5 t316.5 131.5t131.5 316.5v128q0 26 19 45t45 19t45 -19t19 -45zM896 1216v-512q0 -132 -94 -226t-226 -94t-226 94t-94 226v512q0 132 94 226t226 94t226 -94t94 -226z" /> +<glyph unicode="" horiz-adv-x="1408" d="M271 591l-101 -101q-42 103 -42 214v128q0 26 19 45t45 19t45 -19t19 -45v-128q0 -53 15 -113zM1385 1193l-361 -361v-128q0 -132 -94 -226t-226 -94q-55 0 -109 19l-96 -96q97 -51 205 -51q185 0 316.5 131.5t131.5 316.5v128q0 26 19 45t45 19t45 -19t19 -45v-128 q0 -221 -147.5 -384.5t-364.5 -187.5v-132h256q26 0 45 -19t19 -45t-19 -45t-45 -19h-640q-26 0 -45 19t-19 45t19 45t45 19h256v132q-125 13 -235 81l-254 -254q-10 -10 -23 -10t-23 10l-82 82q-10 10 -10 23t10 23l1234 1234q10 10 23 10t23 -10l82 -82q10 -10 10 -23 t-10 -23zM1005 1325l-621 -621v512q0 132 94 226t226 94q102 0 184.5 -59t116.5 -152z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1088 576v640h-448v-1137q119 63 213 137q235 184 235 360zM1280 1344v-768q0 -86 -33.5 -170.5t-83 -150t-118 -127.5t-126.5 -103t-121 -77.5t-89.5 -49.5t-42.5 -20q-12 -6 -26 -6t-26 6q-16 7 -42.5 20t-89.5 49.5t-121 77.5t-126.5 103t-118 127.5t-83 150 t-33.5 170.5v768q0 26 19 45t45 19h1152q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M128 -128h1408v1024h-1408v-1024zM512 1088v288q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-288q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1280 1088v288q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-288q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1664 1152v-1280 q0 -52 -38 -90t-90 -38h-1408q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h128v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h384v96q0 66 47 113t113 47h64q66 0 113 -47t47 -113v-96h128q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="1408" d="M512 1344q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1408 1376v-320q0 -16 -12 -25q-8 -7 -20 -7q-4 0 -7 1l-448 96q-11 2 -18 11t-7 20h-256v-102q111 -23 183.5 -111t72.5 -203v-800q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v800 q0 106 62.5 190.5t161.5 114.5v111h-32q-59 0 -115 -23.5t-91.5 -53t-66 -66.5t-40.5 -53.5t-14 -24.5q-17 -35 -57 -35q-16 0 -29 7q-23 12 -31.5 37t3.5 49q5 10 14.5 26t37.5 53.5t60.5 70t85 67t108.5 52.5q-25 42 -25 86q0 66 47 113t113 47t113 -47t47 -113 q0 -33 -14 -64h302q0 11 7 20t18 11l448 96q3 1 7 1q12 0 20 -7q12 -9 12 -25z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1440 1088q0 40 -28 68t-68 28t-68 -28t-28 -68t28 -68t68 -28t68 28t28 68zM1664 1376q0 -249 -75.5 -430.5t-253.5 -360.5q-81 -80 -195 -176l-20 -379q-2 -16 -16 -26l-384 -224q-7 -4 -16 -4q-12 0 -23 9l-64 64q-13 14 -8 32l85 276l-281 281l-276 -85q-3 -1 -9 -1 q-14 0 -23 9l-64 64q-17 19 -5 39l224 384q10 14 26 16l379 20q96 114 176 195q188 187 358 258t431 71q14 0 24 -9.5t10 -22.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1745 763l-164 -763h-334l178 832q13 56 -15 88q-27 33 -83 33h-169l-204 -953h-334l204 953h-286l-204 -953h-334l204 953l-153 327h1276q101 0 189.5 -40.5t147.5 -113.5q60 -73 81 -168.5t0 -194.5z" /> +<glyph unicode="" d="M909 141l102 102q19 19 19 45t-19 45l-307 307l307 307q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-454 -454q-19 -19 -19 -45t19 -45l454 -454q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M717 141l454 454q19 19 19 45t-19 45l-454 454q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l307 -307l-307 -307q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1165 397l102 102q19 19 19 45t-19 45l-454 454q-19 19 -45 19t-45 -19l-454 -454q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19l307 307l307 -307q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M813 237l454 454q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-307 -307l-307 307q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l454 -454q19 -19 45 -19t45 19zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5 t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1130 939l16 175h-884l47 -534h612l-22 -228l-197 -53l-196 53l-13 140h-175l22 -278l362 -100h4v1l359 99l50 544h-644l-15 181h674zM0 1408h1408l-128 -1438l-578 -162l-574 162z" /> +<glyph unicode="" horiz-adv-x="1792" d="M275 1408h1505l-266 -1333l-804 -267l-698 267l71 356h297l-29 -147l422 -161l486 161l68 339h-1208l58 297h1209l38 191h-1208z" /> +<glyph unicode="" horiz-adv-x="1792" d="M960 1280q0 26 -19 45t-45 19t-45 -19t-19 -45t19 -45t45 -19t45 19t19 45zM1792 352v-352q0 -22 -20 -30q-8 -2 -12 -2q-13 0 -23 9l-93 93q-119 -143 -318.5 -226.5t-429.5 -83.5t-429.5 83.5t-318.5 226.5l-93 -93q-9 -9 -23 -9q-4 0 -12 2q-20 8 -20 30v352 q0 14 9 23t23 9h352q22 0 30 -20q8 -19 -7 -35l-100 -100q67 -91 189.5 -153.5t271.5 -82.5v647h-192q-26 0 -45 19t-19 45v128q0 26 19 45t45 19h192v163q-58 34 -93 92.5t-35 128.5q0 106 75 181t181 75t181 -75t75 -181q0 -70 -35 -128.5t-93 -92.5v-163h192q26 0 45 -19 t19 -45v-128q0 -26 -19 -45t-45 -19h-192v-647q149 20 271.5 82.5t189.5 153.5l-100 100q-15 16 -7 35q8 20 30 20h352q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1056 768q40 0 68 -28t28 -68v-576q0 -40 -28 -68t-68 -28h-960q-40 0 -68 28t-28 68v576q0 40 28 68t68 28h32v320q0 185 131.5 316.5t316.5 131.5t316.5 -131.5t131.5 -316.5q0 -26 -19 -45t-45 -19h-64q-26 0 -45 19t-19 45q0 106 -75 181t-181 75t-181 -75t-75 -181 v-320h736z" /> +<glyph unicode="" d="M1024 640q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM1152 640q0 159 -112.5 271.5t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5t271.5 -112.5t271.5 112.5t112.5 271.5zM1280 640q0 -212 -150 -362t-362 -150t-362 150 t-150 362t150 362t362 150t362 -150t150 -362zM1408 640q0 130 -51 248.5t-136.5 204t-204 136.5t-248.5 51t-248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5zM1536 640 q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M384 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM896 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM1408 800v-192q0 -40 -28 -68t-68 -28h-192 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68z" /> +<glyph unicode="" horiz-adv-x="384" d="M384 288v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM384 800v-192q0 -40 -28 -68t-68 -28h-192q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68zM384 1312v-192q0 -40 -28 -68t-68 -28h-192 q-40 0 -68 28t-28 68v192q0 40 28 68t68 28h192q40 0 68 -28t28 -68z" /> +<glyph unicode="" d="M512 256q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM863 162q-13 232 -177 396t-396 177q-14 1 -24 -9t-10 -23v-128q0 -13 8.5 -22t21.5 -10q154 -11 264 -121t121 -264q1 -13 10 -21.5t22 -8.5h128q13 0 23 10 t9 24zM1247 161q-5 154 -56 297.5t-139.5 260t-205 205t-260 139.5t-297.5 56q-14 1 -23 -9q-10 -10 -10 -23v-128q0 -13 9 -22t22 -10q204 -7 378 -111.5t278.5 -278.5t111.5 -378q1 -13 10 -22t22 -9h128q13 0 23 10q11 9 9 23zM1536 1120v-960q0 -119 -84.5 -203.5 t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M768 1408q209 0 385.5 -103t279.5 -279.5t103 -385.5t-103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103zM1152 585q32 18 32 55t-32 55l-544 320q-31 19 -64 1q-32 -19 -32 -56v-640q0 -37 32 -56 q16 -8 32 -8q17 0 32 9z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1024 1084l316 -316l-572 -572l-316 316zM813 105l618 618q19 19 19 45t-19 45l-362 362q-18 18 -45 18t-45 -18l-618 -618q-19 -19 -19 -45t19 -45l362 -362q18 -18 45 -18t45 18zM1702 742l-907 -908q-37 -37 -90.5 -37t-90.5 37l-126 126q56 56 56 136t-56 136 t-136 56t-136 -56l-125 126q-37 37 -37 90.5t37 90.5l907 906q37 37 90.5 37t90.5 -37l125 -125q-56 -56 -56 -136t56 -136t136 -56t136 56l126 -125q37 -37 37 -90.5t-37 -90.5z" /> +<glyph unicode="" d="M1280 576v128q0 26 -19 45t-45 19h-896q-26 0 -45 -19t-19 -45v-128q0 -26 19 -45t45 -19h896q26 0 45 19t19 45zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5 t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1152 736v-64q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h832q14 0 23 -9t9 -23zM1280 288v832q0 66 -47 113t-113 47h-832q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113zM1408 1120v-832q0 -119 -84.5 -203.5 t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1018 933q-18 -37 -58 -37h-192v-864q0 -14 -9 -23t-23 -9h-704q-21 0 -29 18q-8 20 4 35l160 192q9 11 25 11h320v640h-192q-40 0 -58 37q-17 37 9 68l320 384q18 22 49 22t49 -22l320 -384q27 -32 9 -68z" /> +<glyph unicode="" horiz-adv-x="1024" d="M32 1280h704q13 0 22.5 -9.5t9.5 -23.5v-863h192q40 0 58 -37t-9 -69l-320 -384q-18 -22 -49 -22t-49 22l-320 384q-26 31 -9 69q18 37 58 37h192v640h-320q-14 0 -25 11l-160 192q-13 14 -4 34q9 19 29 19z" /> +<glyph unicode="" d="M685 237l614 614q19 19 19 45t-19 45l-102 102q-19 19 -45 19t-45 -19l-467 -467l-211 211q-19 19 -45 19t-45 -19l-102 -102q-19 -19 -19 -45t19 -45l358 -358q19 -19 45 -19t45 19zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5 t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M404 428l152 -152l-52 -52h-56v96h-96v56zM818 818q14 -13 -3 -30l-291 -291q-17 -17 -30 -3q-14 13 3 30l291 291q17 17 30 3zM544 128l544 544l-288 288l-544 -544v-288h288zM1152 736l92 92q28 28 28 68t-28 68l-152 152q-28 28 -68 28t-68 -28l-92 -92zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1280 608v480q0 26 -19 45t-45 19h-480q-42 0 -59 -39q-17 -41 14 -70l144 -144l-534 -534q-19 -19 -19 -45t19 -45l102 -102q19 -19 45 -19t45 19l534 534l144 -144q18 -19 45 -19q12 0 25 5q39 17 39 59zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960 q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1005 435l352 352q19 19 19 45t-19 45l-352 352q-30 31 -69 14q-40 -17 -40 -59v-160q-119 0 -216 -19.5t-162.5 -51t-114 -79t-76.5 -95.5t-44.5 -109t-21.5 -111.5t-5 -110.5q0 -181 167 -404q10 -12 25 -12q7 0 13 3q22 9 19 33q-44 354 62 473q46 52 130 75.5 t224 23.5v-160q0 -42 40 -59q12 -5 24 -5q26 0 45 19zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M640 448l256 128l-256 128v-256zM1024 1039v-542l-512 -256v542zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1145 861q18 -35 -5 -66l-320 -448q-19 -27 -52 -27t-52 27l-320 448q-23 31 -5 66q17 35 57 35h640q40 0 57 -35zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1145 419q-17 -35 -57 -35h-640q-40 0 -57 35q-18 35 5 66l320 448q19 27 52 27t52 -27l320 -448q23 -31 5 -66zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1088 640q0 -33 -27 -52l-448 -320q-31 -23 -66 -5q-35 17 -35 57v640q0 40 35 57q35 18 66 -5l448 -320q27 -19 27 -52zM1280 160v960q0 14 -9 23t-23 9h-960q-14 0 -23 -9t-9 -23v-960q0 -14 9 -23t23 -9h960q14 0 23 9t9 23zM1536 1120v-960q0 -119 -84.5 -203.5 t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M976 229l35 -159q3 -12 -3 -22.5t-17 -14.5l-5 -1q-4 -2 -10.5 -3.5t-16 -4.5t-21.5 -5.5t-25.5 -5t-30 -5t-33.5 -4.5t-36.5 -3t-38.5 -1q-234 0 -409 130.5t-238 351.5h-95q-13 0 -22.5 9.5t-9.5 22.5v113q0 13 9.5 22.5t22.5 9.5h66q-2 57 1 105h-67q-14 0 -23 9 t-9 23v114q0 14 9 23t23 9h98q67 210 243.5 338t400.5 128q102 0 194 -23q11 -3 20 -15q6 -11 3 -24l-43 -159q-3 -13 -14 -19.5t-24 -2.5l-4 1q-4 1 -11.5 2.5l-17.5 3.5t-22.5 3.5t-26 3t-29 2.5t-29.5 1q-126 0 -226 -64t-150 -176h468q16 0 25 -12q10 -12 7 -26 l-24 -114q-5 -26 -32 -26h-488q-3 -37 0 -105h459q15 0 25 -12q9 -12 6 -27l-24 -112q-2 -11 -11 -18.5t-20 -7.5h-387q48 -117 149.5 -185.5t228.5 -68.5q18 0 36 1.5t33.5 3.5t29.5 4.5t24.5 5t18.5 4.5l12 3l5 2q13 5 26 -2q12 -7 15 -21z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1020 399v-367q0 -14 -9 -23t-23 -9h-956q-14 0 -23 9t-9 23v150q0 13 9.5 22.5t22.5 9.5h97v383h-95q-14 0 -23 9.5t-9 22.5v131q0 14 9 23t23 9h95v223q0 171 123.5 282t314.5 111q185 0 335 -125q9 -8 10 -20.5t-7 -22.5l-103 -127q-9 -11 -22 -12q-13 -2 -23 7 q-5 5 -26 19t-69 32t-93 18q-85 0 -137 -47t-52 -123v-215h305q13 0 22.5 -9t9.5 -23v-131q0 -13 -9.5 -22.5t-22.5 -9.5h-305v-379h414v181q0 13 9 22.5t23 9.5h162q14 0 23 -9.5t9 -22.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M978 351q0 -153 -99.5 -263.5t-258.5 -136.5v-175q0 -14 -9 -23t-23 -9h-135q-13 0 -22.5 9.5t-9.5 22.5v175q-66 9 -127.5 31t-101.5 44.5t-74 48t-46.5 37.5t-17.5 18q-17 21 -2 41l103 135q7 10 23 12q15 2 24 -9l2 -2q113 -99 243 -125q37 -8 74 -8q81 0 142.5 43 t61.5 122q0 28 -15 53t-33.5 42t-58.5 37.5t-66 32t-80 32.5q-39 16 -61.5 25t-61.5 26.5t-62.5 31t-56.5 35.5t-53.5 42.5t-43.5 49t-35.5 58t-21 66.5t-8.5 78q0 138 98 242t255 134v180q0 13 9.5 22.5t22.5 9.5h135q14 0 23 -9t9 -23v-176q57 -6 110.5 -23t87 -33.5 t63.5 -37.5t39 -29t15 -14q17 -18 5 -38l-81 -146q-8 -15 -23 -16q-14 -3 -27 7q-3 3 -14.5 12t-39 26.5t-58.5 32t-74.5 26t-85.5 11.5q-95 0 -155 -43t-60 -111q0 -26 8.5 -48t29.5 -41.5t39.5 -33t56 -31t60.5 -27t70 -27.5q53 -20 81 -31.5t76 -35t75.5 -42.5t62 -50 t53 -63.5t31.5 -76.5t13 -94z" /> +<glyph unicode="" horiz-adv-x="898" d="M898 1066v-102q0 -14 -9 -23t-23 -9h-168q-23 -144 -129 -234t-276 -110q167 -178 459 -536q14 -16 4 -34q-8 -18 -29 -18h-195q-16 0 -25 12q-306 367 -498 571q-9 9 -9 22v127q0 13 9.5 22.5t22.5 9.5h112q132 0 212.5 43t102.5 125h-427q-14 0 -23 9t-9 23v102 q0 14 9 23t23 9h413q-57 113 -268 113h-145q-13 0 -22.5 9.5t-9.5 22.5v133q0 14 9 23t23 9h832q14 0 23 -9t9 -23v-102q0 -14 -9 -23t-23 -9h-233q47 -61 64 -144h171q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1027" d="M603 0h-172q-13 0 -22.5 9t-9.5 23v330h-288q-13 0 -22.5 9t-9.5 23v103q0 13 9.5 22.5t22.5 9.5h288v85h-288q-13 0 -22.5 9t-9.5 23v104q0 13 9.5 22.5t22.5 9.5h214l-321 578q-8 16 0 32q10 16 28 16h194q19 0 29 -18l215 -425q19 -38 56 -125q10 24 30.5 68t27.5 61 l191 420q8 19 29 19h191q17 0 27 -16q9 -14 1 -31l-313 -579h215q13 0 22.5 -9.5t9.5 -22.5v-104q0 -14 -9.5 -23t-22.5 -9h-290v-85h290q13 0 22.5 -9.5t9.5 -22.5v-103q0 -14 -9.5 -23t-22.5 -9h-290v-330q0 -13 -9.5 -22.5t-22.5 -9.5z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1043 971q0 100 -65 162t-171 62h-320v-448h320q106 0 171 62t65 162zM1280 971q0 -193 -126.5 -315t-326.5 -122h-340v-118h505q14 0 23 -9t9 -23v-128q0 -14 -9 -23t-23 -9h-505v-192q0 -14 -9.5 -23t-22.5 -9h-167q-14 0 -23 9t-9 23v192h-224q-14 0 -23 9t-9 23v128 q0 14 9 23t23 9h224v118h-224q-14 0 -23 9t-9 23v149q0 13 9 22.5t23 9.5h224v629q0 14 9 23t23 9h539q200 0 326.5 -122t126.5 -315z" /> +<glyph unicode="" horiz-adv-x="1792" d="M514 341l81 299h-159l75 -300q1 -1 1 -3t1 -3q0 1 0.5 3.5t0.5 3.5zM630 768l35 128h-292l32 -128h225zM822 768h139l-35 128h-70zM1271 340l78 300h-162l81 -299q0 -1 0.5 -3.5t1.5 -3.5q0 1 0.5 3t0.5 3zM1382 768l33 128h-297l34 -128h230zM1792 736v-64q0 -14 -9 -23 t-23 -9h-213l-164 -616q-7 -24 -31 -24h-159q-24 0 -31 24l-166 616h-209l-167 -616q-7 -24 -31 -24h-159q-11 0 -19.5 7t-10.5 17l-160 616h-208q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h175l-33 128h-142q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h109l-89 344q-5 15 5 28 q10 12 26 12h137q26 0 31 -24l90 -360h359l97 360q7 24 31 24h126q24 0 31 -24l98 -360h365l93 360q5 24 31 24h137q16 0 26 -12q10 -13 5 -28l-91 -344h111q14 0 23 -9t9 -23v-64q0 -14 -9 -23t-23 -9h-145l-34 -128h179q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1167 896q18 -182 -131 -258q117 -28 175 -103t45 -214q-7 -71 -32.5 -125t-64.5 -89t-97 -58.5t-121.5 -34.5t-145.5 -15v-255h-154v251q-80 0 -122 1v-252h-154v255q-18 0 -54 0.5t-55 0.5h-200l31 183h111q50 0 58 51v402h16q-6 1 -16 1v287q-13 68 -89 68h-111v164 l212 -1q64 0 97 1v252h154v-247q82 2 122 2v245h154v-252q79 -7 140 -22.5t113 -45t82.5 -78t36.5 -114.5zM952 351q0 36 -15 64t-37 46t-57.5 30.5t-65.5 18.5t-74 9t-69 3t-64.5 -1t-47.5 -1v-338q8 0 37 -0.5t48 -0.5t53 1.5t58.5 4t57 8.5t55.5 14t47.5 21t39.5 30 t24.5 40t9.5 51zM881 827q0 33 -12.5 58.5t-30.5 42t-48 28t-55 16.5t-61.5 8t-58 2.5t-54 -1t-39.5 -0.5v-307q5 0 34.5 -0.5t46.5 0t50 2t55 5.5t51.5 11t48.5 18.5t37 27t27 38.5t9 51z" /> +<glyph unicode="" d="M1024 1024v472q22 -14 36 -28l408 -408q14 -14 28 -36h-472zM896 992q0 -40 28 -68t68 -28h544v-1056q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h800v-544z" /> +<glyph unicode="" d="M1468 1060q14 -14 28 -36h-472v472q22 -14 36 -28zM992 896h544v-1056q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h800v-544q0 -40 28 -68t68 -28zM1152 160v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704 q14 0 23 9t9 23zM1152 416v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704q14 0 23 9t9 23zM1152 672v64q0 14 -9 23t-23 9h-704q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h704q14 0 23 9t9 23z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1191 1128h177l-72 218l-12 47q-2 16 -2 20h-4l-3 -20q0 -1 -3.5 -18t-7.5 -29zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1572 -23 v-233h-584v90l369 529q12 18 21 27l11 9v3q-2 0 -6.5 -0.5t-7.5 -0.5q-12 -3 -30 -3h-232v-115h-120v229h567v-89l-369 -530q-6 -8 -21 -26l-11 -11v-2l14 2q9 2 30 2h248v119h121zM1661 874v-106h-288v106h75l-47 144h-243l-47 -144h75v-106h-287v106h70l230 662h162 l230 -662h70z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1191 104h177l-72 218l-12 47q-2 16 -2 20h-4l-3 -20q0 -1 -3.5 -18t-7.5 -29zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1661 -150 v-106h-288v106h75l-47 144h-243l-47 -144h75v-106h-287v106h70l230 662h162l230 -662h70zM1572 1001v-233h-584v90l369 529q12 18 21 27l11 9v3q-2 0 -6.5 -0.5t-7.5 -0.5q-12 -3 -30 -3h-232v-115h-120v229h567v-89l-369 -530q-6 -8 -21 -26l-11 -10v-3l14 3q9 1 30 1h248 v119h121z" /> +<glyph unicode="" horiz-adv-x="1792" d="M736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23zM1792 -32v-192q0 -14 -9 -23t-23 -9h-832q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h832 q14 0 23 -9t9 -23zM1600 480v-192q0 -14 -9 -23t-23 -9h-640q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h640q14 0 23 -9t9 -23zM1408 992v-192q0 -14 -9 -23t-23 -9h-448q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h448q14 0 23 -9t9 -23zM1216 1504v-192q0 -14 -9 -23t-23 -9h-256 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h256q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1216 -32v-192q0 -14 -9 -23t-23 -9h-256q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h256q14 0 23 -9t9 -23zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192 q14 0 23 -9t9 -23zM1408 480v-192q0 -14 -9 -23t-23 -9h-448q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h448q14 0 23 -9t9 -23zM1600 992v-192q0 -14 -9 -23t-23 -9h-640q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h640q14 0 23 -9t9 -23zM1792 1504v-192q0 -14 -9 -23t-23 -9h-832 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h832q14 0 23 -9t9 -23z" /> +<glyph unicode="" d="M1346 223q0 63 -44 116t-103 53q-52 0 -83 -37t-31 -94t36.5 -95t104.5 -38q50 0 85 27t35 68zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9t9 -23 zM1486 165q0 -62 -13 -121.5t-41 -114t-68 -95.5t-98.5 -65.5t-127.5 -24.5q-62 0 -108 16q-24 8 -42 15l39 113q15 -7 31 -11q37 -13 75 -13q84 0 134.5 58.5t66.5 145.5h-2q-21 -23 -61.5 -37t-84.5 -14q-106 0 -173 71.5t-67 172.5q0 105 72 178t181 73q123 0 205 -94.5 t82 -252.5zM1456 882v-114h-469v114h167v432q0 7 0.5 19t0.5 17v16h-2l-7 -12q-8 -13 -26 -31l-62 -58l-82 86l192 185h123v-654h165z" /> +<glyph unicode="" d="M1346 1247q0 63 -44 116t-103 53q-52 0 -83 -37t-31 -94t36.5 -95t104.5 -38q50 0 85 27t35 68zM736 96q0 -12 -10 -24l-319 -319q-10 -9 -23 -9q-12 0 -23 9l-320 320q-15 16 -7 35q8 20 30 20h192v1376q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1376h192q14 0 23 -9 t9 -23zM1456 -142v-114h-469v114h167v432q0 7 0.5 19t0.5 17v16h-2l-7 -12q-8 -13 -26 -31l-62 -58l-82 86l192 185h123v-654h165zM1486 1189q0 -62 -13 -121.5t-41 -114t-68 -95.5t-98.5 -65.5t-127.5 -24.5q-62 0 -108 16q-24 8 -42 15l39 113q15 -7 31 -11q37 -13 75 -13 q84 0 134.5 58.5t66.5 145.5h-2q-21 -23 -61.5 -37t-84.5 -14q-106 0 -173 71.5t-67 172.5q0 105 72 178t181 73q123 0 205 -94.5t82 -252.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M256 192q0 26 -19 45t-45 19q-27 0 -45.5 -19t-18.5 -45q0 -27 18.5 -45.5t45.5 -18.5q26 0 45 18.5t19 45.5zM416 704v-640q0 -26 -19 -45t-45 -19h-288q-26 0 -45 19t-19 45v640q0 26 19 45t45 19h288q26 0 45 -19t19 -45zM1600 704q0 -86 -55 -149q15 -44 15 -76 q3 -76 -43 -137q17 -56 0 -117q-15 -57 -54 -94q9 -112 -49 -181q-64 -76 -197 -78h-36h-76h-17q-66 0 -144 15.5t-121.5 29t-120.5 39.5q-123 43 -158 44q-26 1 -45 19.5t-19 44.5v641q0 25 18 43.5t43 20.5q24 2 76 59t101 121q68 87 101 120q18 18 31 48t17.5 48.5 t13.5 60.5q7 39 12.5 61t19.5 52t34 50q19 19 45 19q46 0 82.5 -10.5t60 -26t40 -40.5t24 -45t12 -50t5 -45t0.5 -39q0 -38 -9.5 -76t-19 -60t-27.5 -56q-3 -6 -10 -18t-11 -22t-8 -24h277q78 0 135 -57t57 -135z" /> +<glyph unicode="" horiz-adv-x="1664" d="M256 960q0 -26 -19 -45t-45 -19q-27 0 -45.5 19t-18.5 45q0 27 18.5 45.5t45.5 18.5q26 0 45 -18.5t19 -45.5zM416 448v640q0 26 -19 45t-45 19h-288q-26 0 -45 -19t-19 -45v-640q0 -26 19 -45t45 -19h288q26 0 45 19t19 45zM1545 597q55 -61 55 -149q-1 -78 -57.5 -135 t-134.5 -57h-277q4 -14 8 -24t11 -22t10 -18q18 -37 27 -57t19 -58.5t10 -76.5q0 -24 -0.5 -39t-5 -45t-12 -50t-24 -45t-40 -40.5t-60 -26t-82.5 -10.5q-26 0 -45 19q-20 20 -34 50t-19.5 52t-12.5 61q-9 42 -13.5 60.5t-17.5 48.5t-31 48q-33 33 -101 120q-49 64 -101 121 t-76 59q-25 2 -43 20.5t-18 43.5v641q0 26 19 44.5t45 19.5q35 1 158 44q77 26 120.5 39.5t121.5 29t144 15.5h17h76h36q133 -2 197 -78q58 -69 49 -181q39 -37 54 -94q17 -61 0 -117q46 -61 43 -137q0 -32 -15 -76z" /> +<glyph unicode="" d="M919 233v157q0 50 -29 50q-17 0 -33 -16v-224q16 -16 33 -16q29 0 29 49zM1103 355h66v34q0 51 -33 51t-33 -51v-34zM532 621v-70h-80v-423h-74v423h-78v70h232zM733 495v-367h-67v40q-39 -45 -76 -45q-33 0 -42 28q-6 16 -6 54v290h66v-270q0 -24 1 -26q1 -15 15 -15 q20 0 42 31v280h67zM985 384v-146q0 -52 -7 -73q-12 -42 -53 -42q-35 0 -68 41v-36h-67v493h67v-161q32 40 68 40q41 0 53 -42q7 -21 7 -74zM1236 255v-9q0 -29 -2 -43q-3 -22 -15 -40q-27 -40 -80 -40q-52 0 -81 38q-21 27 -21 86v129q0 59 20 86q29 38 80 38t78 -38 q21 -28 21 -86v-76h-133v-65q0 -51 34 -51q24 0 30 26q0 1 0.5 7t0.5 16.5v21.5h68zM785 1079v-156q0 -51 -32 -51t-32 51v156q0 52 32 52t32 -52zM1318 366q0 177 -19 260q-10 44 -43 73.5t-76 34.5q-136 15 -412 15q-275 0 -411 -15q-44 -5 -76.5 -34.5t-42.5 -73.5 q-20 -87 -20 -260q0 -176 20 -260q10 -43 42.5 -73t75.5 -35q137 -15 412 -15t412 15q43 5 75.5 35t42.5 73q20 84 20 260zM563 1017l90 296h-75l-51 -195l-53 195h-78l24 -69t23 -69q35 -103 46 -158v-201h74v201zM852 936v130q0 58 -21 87q-29 38 -78 38q-51 0 -78 -38 q-21 -29 -21 -87v-130q0 -58 21 -87q27 -38 78 -38q49 0 78 38q21 27 21 87zM1033 816h67v370h-67v-283q-22 -31 -42 -31q-15 0 -16 16q-1 2 -1 26v272h-67v-293q0 -37 6 -55q11 -27 43 -27q36 0 77 45v-40zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960 q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M971 292v-211q0 -67 -39 -67q-23 0 -45 22v301q22 22 45 22q39 0 39 -67zM1309 291v-46h-90v46q0 68 45 68t45 -68zM343 509h107v94h-312v-94h105v-569h100v569zM631 -60h89v494h-89v-378q-30 -42 -57 -42q-18 0 -21 21q-1 3 -1 35v364h-89v-391q0 -49 8 -73 q12 -37 58 -37q48 0 102 61v-54zM1060 88v197q0 73 -9 99q-17 56 -71 56q-50 0 -93 -54v217h-89v-663h89v48q45 -55 93 -55q54 0 71 55q9 27 9 100zM1398 98v13h-91q0 -51 -2 -61q-7 -36 -40 -36q-46 0 -46 69v87h179v103q0 79 -27 116q-39 51 -106 51q-68 0 -107 -51 q-28 -37 -28 -116v-173q0 -79 29 -116q39 -51 108 -51q72 0 108 53q18 27 21 54q2 9 2 58zM790 1011v210q0 69 -43 69t-43 -69v-210q0 -70 43 -70t43 70zM1509 260q0 -234 -26 -350q-14 -59 -58 -99t-102 -46q-184 -21 -555 -21t-555 21q-58 6 -102.5 46t-57.5 99 q-26 112 -26 350q0 234 26 350q14 59 58 99t103 47q183 20 554 20t555 -20q58 -7 102.5 -47t57.5 -99q26 -112 26 -350zM511 1536h102l-121 -399v-271h-100v271q-14 74 -61 212q-37 103 -65 187h106l71 -263zM881 1203v-175q0 -81 -28 -118q-37 -51 -106 -51q-67 0 -105 51 q-28 38 -28 118v175q0 80 28 117q38 51 105 51q69 0 106 -51q28 -37 28 -117zM1216 1365v-499h-91v55q-53 -62 -103 -62q-46 0 -59 37q-8 24 -8 75v394h91v-367q0 -33 1 -35q3 -22 21 -22q27 0 57 43v381h91z" /> +<glyph unicode="" horiz-adv-x="1408" d="M597 869q-10 -18 -257 -456q-27 -46 -65 -46h-239q-21 0 -31 17t0 36l253 448q1 0 0 1l-161 279q-12 22 -1 37q9 15 32 15h239q40 0 66 -45zM1403 1511q11 -16 0 -37l-528 -934v-1l336 -615q11 -20 1 -37q-10 -15 -32 -15h-239q-42 0 -66 45l-339 622q18 32 531 942 q25 45 64 45h241q22 0 31 -15z" /> +<glyph unicode="" d="M685 771q0 1 -126 222q-21 34 -52 34h-184q-18 0 -26 -11q-7 -12 1 -29l125 -216v-1l-196 -346q-9 -14 0 -28q8 -13 24 -13h185q31 0 50 36zM1309 1268q-7 12 -24 12h-187q-30 0 -49 -35l-411 -729q1 -2 262 -481q20 -35 52 -35h184q18 0 25 12q8 13 -1 28l-260 476v1 l409 723q8 16 0 28zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1280 640q0 37 -30 54l-512 320q-31 20 -65 2q-33 -18 -33 -56v-640q0 -38 33 -56q16 -8 31 -8q20 0 34 10l512 320q30 17 30 54zM1792 640q0 -96 -1 -150t-8.5 -136.5t-22.5 -147.5q-16 -73 -69 -123t-124 -58q-222 -25 -671 -25t-671 25q-71 8 -124.5 58t-69.5 123 q-14 65 -21.5 147.5t-8.5 136.5t-1 150t1 150t8.5 136.5t22.5 147.5q16 73 69 123t124 58q222 25 671 25t671 -25q71 -8 124.5 -58t69.5 -123q14 -65 21.5 -147.5t8.5 -136.5t1 -150z" /> +<glyph unicode="" horiz-adv-x="1792" d="M402 829l494 -305l-342 -285l-490 319zM1388 274v-108l-490 -293v-1l-1 1l-1 -1v1l-489 293v108l147 -96l342 284v2l1 -1l1 1v-2l343 -284zM554 1418l342 -285l-494 -304l-338 270zM1390 829l338 -271l-489 -319l-343 285zM1239 1418l489 -319l-338 -270l-494 304z" /> +<glyph unicode="" horiz-adv-x="1408" d="M928 135v-151l-707 -1v151zM1169 481v-701l-1 -35v-1h-1132l-35 1h-1v736h121v-618h928v618h120zM241 393l704 -65l-13 -150l-705 65zM309 709l683 -183l-39 -146l-683 183zM472 1058l609 -360l-77 -130l-609 360zM832 1389l398 -585l-124 -85l-399 584zM1285 1536 l121 -697l-149 -26l-121 697z" /> +<glyph unicode="" d="M1362 110v648h-135q20 -63 20 -131q0 -126 -64 -232.5t-174 -168.5t-240 -62q-197 0 -337 135.5t-140 327.5q0 68 20 131h-141v-648q0 -26 17.5 -43.5t43.5 -17.5h1069q25 0 43 17.5t18 43.5zM1078 643q0 124 -90.5 211.5t-218.5 87.5q-127 0 -217.5 -87.5t-90.5 -211.5 t90.5 -211.5t217.5 -87.5q128 0 218.5 87.5t90.5 211.5zM1362 1003v165q0 28 -20 48.5t-49 20.5h-174q-29 0 -49 -20.5t-20 -48.5v-165q0 -29 20 -49t49 -20h174q29 0 49 20t20 49zM1536 1211v-1142q0 -81 -58 -139t-139 -58h-1142q-81 0 -139 58t-58 139v1142q0 81 58 139 t139 58h1142q81 0 139 -58t58 -139z" /> +<glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960zM698 640q0 88 -62 150t-150 62t-150 -62t-62 -150t62 -150t150 -62t150 62t62 150zM1262 640q0 88 -62 150 t-150 62t-150 -62t-62 -150t62 -150t150 -62t150 62t62 150z" /> +<glyph unicode="" d="M768 914l201 -306h-402zM1133 384h94l-459 691l-459 -691h94l104 160h522zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M815 677q8 -63 -50.5 -101t-111.5 -6q-39 17 -53.5 58t-0.5 82t52 58q36 18 72.5 12t64 -35.5t27.5 -67.5zM926 698q-14 107 -113 164t-197 13q-63 -28 -100.5 -88.5t-34.5 -129.5q4 -91 77.5 -155t165.5 -56q91 8 152 84t50 168zM1165 1240q-20 27 -56 44.5t-58 22 t-71 12.5q-291 47 -566 -2q-43 -7 -66 -12t-55 -22t-50 -43q30 -28 76 -45.5t73.5 -22t87.5 -11.5q228 -29 448 -1q63 8 89.5 12t72.5 21.5t75 46.5zM1222 205q-8 -26 -15.5 -76.5t-14 -84t-28.5 -70t-58 -56.5q-86 -48 -189.5 -71.5t-202 -22t-201.5 18.5q-46 8 -81.5 18 t-76.5 27t-73 43.5t-52 61.5q-25 96 -57 292l6 16l18 9q223 -148 506.5 -148t507.5 148q21 -6 24 -23t-5 -45t-8 -37zM1403 1166q-26 -167 -111 -655q-5 -30 -27 -56t-43.5 -40t-54.5 -31q-252 -126 -610 -88q-248 27 -394 139q-15 12 -25.5 26.5t-17 35t-9 34t-6 39.5 t-5.5 35q-9 50 -26.5 150t-28 161.5t-23.5 147.5t-22 158q3 26 17.5 48.5t31.5 37.5t45 30t46 22.5t48 18.5q125 46 313 64q379 37 676 -50q155 -46 215 -122q16 -20 16.5 -51t-5.5 -54z" /> +<glyph unicode="" d="M848 666q0 43 -41 66t-77 1q-43 -20 -42.5 -72.5t43.5 -70.5q39 -23 81 4t36 72zM928 682q8 -66 -36 -121t-110 -61t-119 40t-56 113q-2 49 25.5 93t72.5 64q70 31 141.5 -10t81.5 -118zM1100 1073q-20 -21 -53.5 -34t-53 -16t-63.5 -8q-155 -20 -324 0q-44 6 -63 9.5 t-52.5 16t-54.5 32.5q13 19 36 31t40 15.5t47 8.5q198 35 408 1q33 -5 51 -8.5t43 -16t39 -31.5zM1142 327q0 7 5.5 26.5t3 32t-17.5 16.5q-161 -106 -365 -106t-366 106l-12 -6l-5 -12q26 -154 41 -210q47 -81 204 -108q249 -46 428 53q34 19 49 51.5t22.5 85.5t12.5 71z M1272 1020q9 53 -8 75q-43 55 -155 88q-216 63 -487 36q-132 -12 -226 -46q-38 -15 -59.5 -25t-47 -34t-29.5 -54q8 -68 19 -138t29 -171t24 -137q1 -5 5 -31t7 -36t12 -27t22 -28q105 -80 284 -100q259 -28 440 63q24 13 39.5 23t31 29t19.5 40q48 267 80 473zM1536 1120 v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M944 207l80 -237q-23 -35 -111 -66t-177 -32q-104 -2 -190.5 26t-142.5 74t-95 106t-55.5 120t-16.5 118v544h-168v215q72 26 129 69.5t91 90t58 102t34 99t15 88.5q1 5 4.5 8.5t7.5 3.5h244v-424h333v-252h-334v-518q0 -30 6.5 -56t22.5 -52.5t49.5 -41.5t81.5 -14 q78 2 134 29z" /> +<glyph unicode="" d="M1136 75l-62 183q-44 -22 -103 -22q-36 -1 -62 10.5t-38.5 31.5t-17.5 40.5t-5 43.5v398h257v194h-256v326h-188q-8 0 -9 -10q-5 -44 -17.5 -87t-39 -95t-77 -95t-118.5 -68v-165h130v-418q0 -57 21.5 -115t65 -111t121 -85.5t176.5 -30.5q69 1 136.5 25t85.5 50z M1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="768" d="M765 237q8 -19 -5 -35l-350 -384q-10 -10 -23 -10q-14 0 -24 10l-355 384q-13 16 -5 35q9 19 29 19h224v1248q0 14 9 23t23 9h192q14 0 23 -9t9 -23v-1248h224q21 0 29 -19z" /> +<glyph unicode="" horiz-adv-x="768" d="M765 1043q-9 -19 -29 -19h-224v-1248q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v1248h-224q-21 0 -29 19t5 35l350 384q10 10 23 10q14 0 24 -10l355 -384q13 -16 5 -35z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 736v-192q0 -14 -9 -23t-23 -9h-1248v-224q0 -21 -19 -29t-35 5l-384 350q-10 10 -10 23q0 14 10 24l384 354q16 14 35 6q19 -9 19 -29v-224h1248q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1728 643q0 -14 -10 -24l-384 -354q-16 -14 -35 -6q-19 9 -19 29v224h-1248q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h1248v224q0 21 19 29t35 -5l384 -350q10 -10 10 -23z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1393 321q-39 -125 -123 -250q-129 -196 -257 -196q-49 0 -140 32q-86 32 -151 32q-61 0 -142 -33q-81 -34 -132 -34q-152 0 -301 259q-147 261 -147 503q0 228 113 374q112 144 284 144q72 0 177 -30q104 -30 138 -30q45 0 143 34q102 34 173 34q119 0 213 -65 q52 -36 104 -100q-79 -67 -114 -118q-65 -94 -65 -207q0 -124 69 -223t158 -126zM1017 1494q0 -61 -29 -136q-30 -75 -93 -138q-54 -54 -108 -72q-37 -11 -104 -17q3 149 78 257q74 107 250 148q1 -3 2.5 -11t2.5 -11q0 -4 0.5 -10t0.5 -10z" /> +<glyph unicode="" horiz-adv-x="1664" d="M682 530v-651l-682 94v557h682zM682 1273v-659h-682v565zM1664 530v-786l-907 125v661h907zM1664 1408v-794h-907v669z" /> +<glyph unicode="" horiz-adv-x="1408" d="M493 1053q16 0 27.5 11.5t11.5 27.5t-11.5 27.5t-27.5 11.5t-27 -11.5t-11 -27.5t11 -27.5t27 -11.5zM915 1053q16 0 27 11.5t11 27.5t-11 27.5t-27 11.5t-27.5 -11.5t-11.5 -27.5t11.5 -27.5t27.5 -11.5zM103 869q42 0 72 -30t30 -72v-430q0 -43 -29.5 -73t-72.5 -30 t-73 30t-30 73v430q0 42 30 72t73 30zM1163 850v-666q0 -46 -32 -78t-77 -32h-75v-227q0 -43 -30 -73t-73 -30t-73 30t-30 73v227h-138v-227q0 -43 -30 -73t-73 -30q-42 0 -72 30t-30 73l-1 227h-74q-46 0 -78 32t-32 78v666h918zM931 1255q107 -55 171 -153.5t64 -215.5 h-925q0 117 64 215.5t172 153.5l-71 131q-7 13 5 20q13 6 20 -6l72 -132q95 42 201 42t201 -42l72 132q7 12 20 6q12 -7 5 -20zM1408 767v-430q0 -43 -30 -73t-73 -30q-42 0 -72 30t-30 73v430q0 43 30 72.5t72 29.5q43 0 73 -29.5t30 -72.5z" /> +<glyph unicode="" d="M663 1125q-11 -1 -15.5 -10.5t-8.5 -9.5q-5 -1 -5 5q0 12 19 15h10zM750 1111q-4 -1 -11.5 6.5t-17.5 4.5q24 11 32 -2q3 -6 -3 -9zM399 684q-4 1 -6 -3t-4.5 -12.5t-5.5 -13.5t-10 -13q-7 -10 -1 -12q4 -1 12.5 7t12.5 18q1 3 2 7t2 6t1.5 4.5t0.5 4v3t-1 2.5t-3 2z M1254 325q0 18 -55 42q4 15 7.5 27.5t5 26t3 21.5t0.5 22.5t-1 19.5t-3.5 22t-4 20.5t-5 25t-5.5 26.5q-10 48 -47 103t-72 75q24 -20 57 -83q87 -162 54 -278q-11 -40 -50 -42q-31 -4 -38.5 18.5t-8 83.5t-11.5 107q-9 39 -19.5 69t-19.5 45.5t-15.5 24.5t-13 15t-7.5 7 q-14 62 -31 103t-29.5 56t-23.5 33t-15 40q-4 21 6 53.5t4.5 49.5t-44.5 25q-15 3 -44.5 18t-35.5 16q-8 1 -11 26t8 51t36 27q37 3 51 -30t4 -58q-11 -19 -2 -26.5t30 -0.5q13 4 13 36v37q-5 30 -13.5 50t-21 30.5t-23.5 15t-27 7.5q-107 -8 -89 -134q0 -15 -1 -15 q-9 9 -29.5 10.5t-33 -0.5t-15.5 5q1 57 -16 90t-45 34q-27 1 -41.5 -27.5t-16.5 -59.5q-1 -15 3.5 -37t13 -37.5t15.5 -13.5q10 3 16 14q4 9 -7 8q-7 0 -15.5 14.5t-9.5 33.5q-1 22 9 37t34 14q17 0 27 -21t9.5 -39t-1.5 -22q-22 -15 -31 -29q-8 -12 -27.5 -23.5 t-20.5 -12.5q-13 -14 -15.5 -27t7.5 -18q14 -8 25 -19.5t16 -19t18.5 -13t35.5 -6.5q47 -2 102 15q2 1 23 7t34.5 10.5t29.5 13t21 17.5q9 14 20 8q5 -3 6.5 -8.5t-3 -12t-16.5 -9.5q-20 -6 -56.5 -21.5t-45.5 -19.5q-44 -19 -70 -23q-25 -5 -79 2q-10 2 -9 -2t17 -19 q25 -23 67 -22q17 1 36 7t36 14t33.5 17.5t30 17t24.5 12t17.5 2.5t8.5 -11q0 -2 -1 -4.5t-4 -5t-6 -4.5t-8.5 -5t-9 -4.5t-10 -5t-9.5 -4.5q-28 -14 -67.5 -44t-66.5 -43t-49 -1q-21 11 -63 73q-22 31 -25 22q-1 -3 -1 -10q0 -25 -15 -56.5t-29.5 -55.5t-21 -58t11.5 -63 q-23 -6 -62.5 -90t-47.5 -141q-2 -18 -1.5 -69t-5.5 -59q-8 -24 -29 -3q-32 31 -36 94q-2 28 4 56q4 19 -1 18l-4 -5q-36 -65 10 -166q5 -12 25 -28t24 -20q20 -23 104 -90.5t93 -76.5q16 -15 17.5 -38t-14 -43t-45.5 -23q8 -15 29 -44.5t28 -54t7 -70.5q46 24 7 92 q-4 8 -10.5 16t-9.5 12t-2 6q3 5 13 9.5t20 -2.5q46 -52 166 -36q133 15 177 87q23 38 34 30q12 -6 10 -52q-1 -25 -23 -92q-9 -23 -6 -37.5t24 -15.5q3 19 14.5 77t13.5 90q2 21 -6.5 73.5t-7.5 97t23 70.5q15 18 51 18q1 37 34.5 53t72.5 10.5t60 -22.5zM626 1152 q3 17 -2.5 30t-11.5 15q-9 2 -9 -7q2 -5 5 -6q10 0 7 -15q-3 -20 8 -20q3 0 3 3zM1045 955q-2 8 -6.5 11.5t-13 5t-14.5 5.5q-5 3 -9.5 8t-7 8t-5.5 6.5t-4 4t-4 -1.5q-14 -16 7 -43.5t39 -31.5q9 -1 14.5 8t3.5 20zM867 1168q0 11 -5 19.5t-11 12.5t-9 3q-14 -1 -7 -7l4 -2 q14 -4 18 -31q0 -3 8 2zM921 1401q0 2 -2.5 5t-9 7t-9.5 6q-15 15 -24 15q-9 -1 -11.5 -7.5t-1 -13t-0.5 -12.5q-1 -4 -6 -10.5t-6 -9t3 -8.5q4 -3 8 0t11 9t15 9q1 1 9 1t15 2t9 7zM1486 60q20 -12 31 -24.5t12 -24t-2.5 -22.5t-15.5 -22t-23.5 -19.5t-30 -18.5 t-31.5 -16.5t-32 -15.5t-27 -13q-38 -19 -85.5 -56t-75.5 -64q-17 -16 -68 -19.5t-89 14.5q-18 9 -29.5 23.5t-16.5 25.5t-22 19.5t-47 9.5q-44 1 -130 1q-19 0 -57 -1.5t-58 -2.5q-44 -1 -79.5 -15t-53.5 -30t-43.5 -28.5t-53.5 -11.5q-29 1 -111 31t-146 43q-19 4 -51 9.5 t-50 9t-39.5 9.5t-33.5 14.5t-17 19.5q-10 23 7 66.5t18 54.5q1 16 -4 40t-10 42.5t-4.5 36.5t10.5 27q14 12 57 14t60 12q30 18 42 35t12 51q21 -73 -32 -106q-32 -20 -83 -15q-34 3 -43 -10q-13 -15 5 -57q2 -6 8 -18t8.5 -18t4.5 -17t1 -22q0 -15 -17 -49t-14 -48 q3 -17 37 -26q20 -6 84.5 -18.5t99.5 -20.5q24 -6 74 -22t82.5 -23t55.5 -4q43 6 64.5 28t23 48t-7.5 58.5t-19 52t-20 36.5q-121 190 -169 242q-68 74 -113 40q-11 -9 -15 15q-3 16 -2 38q1 29 10 52t24 47t22 42q8 21 26.5 72t29.5 78t30 61t39 54q110 143 124 195 q-12 112 -16 310q-2 90 24 151.5t106 104.5q39 21 104 21q53 1 106 -13.5t89 -41.5q57 -42 91.5 -121.5t29.5 -147.5q-5 -95 30 -214q34 -113 133 -218q55 -59 99.5 -163t59.5 -191q8 -49 5 -84.5t-12 -55.5t-20 -22q-10 -2 -23.5 -19t-27 -35.5t-40.5 -33.5t-61 -14 q-18 1 -31.5 5t-22.5 13.5t-13.5 15.5t-11.5 20.5t-9 19.5q-22 37 -41 30t-28 -49t7 -97q20 -70 1 -195q-10 -65 18 -100.5t73 -33t85 35.5q59 49 89.5 66.5t103.5 42.5q53 18 77 36.5t18.5 34.5t-25 28.5t-51.5 23.5q-33 11 -49.5 48t-15 72.5t15.5 47.5q1 -31 8 -56.5 t14.5 -40.5t20.5 -28.5t21 -19t21.5 -13t16.5 -9.5z" /> +<glyph unicode="" d="M1024 36q-42 241 -140 498h-2l-2 -1q-16 -6 -43 -16.5t-101 -49t-137 -82t-131 -114.5t-103 -148l-15 11q184 -150 418 -150q132 0 256 52zM839 643q-21 49 -53 111q-311 -93 -673 -93q-1 -7 -1 -21q0 -124 44 -236.5t124 -201.5q50 89 123.5 166.5t142.5 124.5t130.5 81 t99.5 48l37 13q4 1 13 3.5t13 4.5zM732 855q-120 213 -244 378q-138 -65 -234 -186t-128 -272q302 0 606 80zM1416 536q-210 60 -409 29q87 -239 128 -469q111 75 185 189.5t96 250.5zM611 1277q-1 0 -2 -1q1 1 2 1zM1201 1132q-185 164 -433 164q-76 0 -155 -19 q131 -170 246 -382q69 26 130 60.5t96.5 61.5t65.5 57t37.5 40.5zM1424 647q-3 232 -149 410l-1 -1q-9 -12 -19 -24.5t-43.5 -44.5t-71 -60.5t-100 -65t-131.5 -64.5q25 -53 44 -95q2 -6 6.5 -17.5t7.5 -16.5q36 5 74.5 7t73.5 2t69 -1.5t64 -4t56.5 -5.5t48 -6.5t36.5 -6 t25 -4.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1173 473q0 50 -19.5 91.5t-48.5 68.5t-73 49t-82.5 34t-87.5 23l-104 24q-30 7 -44 10.5t-35 11.5t-30 16t-16.5 21t-7.5 30q0 77 144 77q43 0 77 -12t54 -28.5t38 -33.5t40 -29t48 -12q47 0 75.5 32t28.5 77q0 55 -56 99.5t-142 67.5t-182 23q-68 0 -132 -15.5 t-119.5 -47t-89 -87t-33.5 -128.5q0 -61 19 -106.5t56 -75.5t80 -48.5t103 -32.5l146 -36q90 -22 112 -36q32 -20 32 -60q0 -39 -40 -64.5t-105 -25.5q-51 0 -91.5 16t-65 38.5t-45.5 45t-46 38.5t-54 16q-50 0 -75.5 -30t-25.5 -75q0 -92 122 -157.5t291 -65.5 q73 0 140 18.5t122.5 53.5t88.5 93.5t33 131.5zM1536 256q0 -159 -112.5 -271.5t-271.5 -112.5q-130 0 -234 80q-77 -16 -150 -16q-143 0 -273.5 55.5t-225 150t-150 225t-55.5 273.5q0 73 16 150q-80 104 -80 234q0 159 112.5 271.5t271.5 112.5q130 0 234 -80 q77 16 150 16q143 0 273.5 -55.5t225 -150t150 -225t55.5 -273.5q0 -73 -16 -150q80 -104 80 -234z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1000 1102l37 194q5 23 -9 40t-35 17h-712q-23 0 -38.5 -17t-15.5 -37v-1101q0 -7 6 -1l291 352q23 26 38 33.5t48 7.5h239q22 0 37 14.5t18 29.5q24 130 37 191q4 21 -11.5 40t-36.5 19h-294q-29 0 -48 19t-19 48v42q0 29 19 47.5t48 18.5h346q18 0 35 13.5t20 29.5z M1227 1324q-15 -73 -53.5 -266.5t-69.5 -350t-35 -173.5q-6 -22 -9 -32.5t-14 -32.5t-24.5 -33t-38.5 -21t-58 -10h-271q-13 0 -22 -10q-8 -9 -426 -494q-22 -25 -58.5 -28.5t-48.5 5.5q-55 22 -55 98v1410q0 55 38 102.5t120 47.5h888q95 0 127 -53t10 -159zM1227 1324 l-158 -790q4 17 35 173.5t69.5 350t53.5 266.5z" /> +<glyph unicode="" d="M704 192v1024q0 14 -9 23t-23 9h-480q-14 0 -23 -9t-9 -23v-1024q0 -14 9 -23t23 -9h480q14 0 23 9t9 23zM1376 576v640q0 14 -9 23t-23 9h-480q-14 0 -23 -9t-9 -23v-640q0 -14 9 -23t23 -9h480q14 0 23 9t9 23zM1536 1344v-1408q0 -26 -19 -45t-45 -19h-1408 q-26 0 -45 19t-19 45v1408q0 26 19 45t45 19h1408q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1280 480q0 -40 -28 -68t-68 -28q-51 0 -80 43l-227 341h-45v-132l247 -411q9 -15 9 -33q0 -26 -19 -45t-45 -19h-192v-272q0 -46 -33 -79t-79 -33h-160q-46 0 -79 33t-33 79v272h-192q-26 0 -45 19t-19 45q0 18 9 33l247 411v132h-45l-227 -341q-29 -43 -80 -43 q-40 0 -68 28t-28 68q0 29 16 53l256 384q73 107 176 107h384q103 0 176 -107l256 -384q16 -24 16 -53zM864 1280q0 -93 -65.5 -158.5t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" /> +<glyph unicode="" horiz-adv-x="1024" d="M1024 832v-416q0 -40 -28 -68t-68 -28t-68 28t-28 68v352h-64v-912q0 -46 -33 -79t-79 -33t-79 33t-33 79v464h-64v-464q0 -46 -33 -79t-79 -33t-79 33t-33 79v912h-64v-352q0 -40 -28 -68t-68 -28t-68 28t-28 68v416q0 80 56 136t136 56h640q80 0 136 -56t56 -136z M736 1280q0 -93 -65.5 -158.5t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" /> +<glyph unicode="" d="M773 234l350 473q16 22 24.5 59t-6 85t-61.5 79q-40 26 -83 25.5t-73.5 -17.5t-54.5 -45q-36 -40 -96 -40q-59 0 -95 40q-24 28 -54.5 45t-73.5 17.5t-84 -25.5q-46 -31 -60.5 -79t-6 -85t24.5 -59zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1472 640q0 117 -45.5 223.5t-123 184t-184 123t-223.5 45.5t-223.5 -45.5t-184 -123t-123 -184t-45.5 -223.5t45.5 -223.5t123 -184t184 -123t223.5 -45.5t223.5 45.5t184 123t123 184t45.5 223.5zM1748 363q-4 -15 -20 -20l-292 -96v-306q0 -16 -13 -26q-15 -10 -29 -4 l-292 94l-180 -248q-10 -13 -26 -13t-26 13l-180 248l-292 -94q-14 -6 -29 4q-13 10 -13 26v306l-292 96q-16 5 -20 20q-5 17 4 29l180 248l-180 248q-9 13 -4 29q4 15 20 20l292 96v306q0 16 13 26q15 10 29 4l292 -94l180 248q9 12 26 12t26 -12l180 -248l292 94 q14 6 29 -4q13 -10 13 -26v-306l292 -96q16 -5 20 -20q5 -16 -4 -29l-180 -248l180 -248q9 -12 4 -29z" /> +<glyph unicode="" d="M1262 233q-54 -9 -110 -9q-182 0 -337 90t-245 245t-90 337q0 192 104 357q-201 -60 -328.5 -229t-127.5 -384q0 -130 51 -248.5t136.5 -204t204 -136.5t248.5 -51q144 0 273.5 61.5t220.5 171.5zM1465 318q-94 -203 -283.5 -324.5t-413.5 -121.5q-156 0 -298 61 t-245 164t-164 245t-61 298q0 153 57.5 292.5t156 241.5t235.5 164.5t290 68.5q44 2 61 -39q18 -41 -15 -72q-86 -78 -131.5 -181.5t-45.5 -218.5q0 -148 73 -273t198 -198t273 -73q118 0 228 51q41 18 72 -13q14 -14 17.5 -34t-4.5 -38z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1088 704q0 26 -19 45t-45 19h-256q-26 0 -45 -19t-19 -45t19 -45t45 -19h256q26 0 45 19t19 45zM1664 896v-960q0 -26 -19 -45t-45 -19h-1408q-26 0 -45 19t-19 45v960q0 26 19 45t45 19h1408q26 0 45 -19t19 -45zM1728 1344v-256q0 -26 -19 -45t-45 -19h-1536 q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h1536q26 0 45 -19t19 -45z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1632 576q0 -26 -19 -45t-45 -19h-224q0 -171 -67 -290l208 -209q19 -19 19 -45t-19 -45q-18 -19 -45 -19t-45 19l-198 197q-5 -5 -15 -13t-42 -28.5t-65 -36.5t-82 -29t-97 -13v896h-128v-896q-51 0 -101.5 13.5t-87 33t-66 39t-43.5 32.5l-15 14l-183 -207 q-20 -21 -48 -21q-24 0 -43 16q-19 18 -20.5 44.5t15.5 46.5l202 227q-58 114 -58 274h-224q-26 0 -45 19t-19 45t19 45t45 19h224v294l-173 173q-19 19 -19 45t19 45t45 19t45 -19l173 -173h844l173 173q19 19 45 19t45 -19t19 -45t-19 -45l-173 -173v-294h224q26 0 45 -19 t19 -45zM1152 1152h-640q0 133 93.5 226.5t226.5 93.5t226.5 -93.5t93.5 -226.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1917 1016q23 -64 -150 -294q-24 -32 -65 -85q-78 -100 -90 -131q-17 -41 14 -81q17 -21 81 -82h1l1 -1l1 -1l2 -2q141 -131 191 -221q3 -5 6.5 -12.5t7 -26.5t-0.5 -34t-25 -27.5t-59 -12.5l-256 -4q-24 -5 -56 5t-52 22l-20 12q-30 21 -70 64t-68.5 77.5t-61 58 t-56.5 15.5q-3 -1 -8 -3.5t-17 -14.5t-21.5 -29.5t-17 -52t-6.5 -77.5q0 -15 -3.5 -27.5t-7.5 -18.5l-4 -5q-18 -19 -53 -22h-115q-71 -4 -146 16.5t-131.5 53t-103 66t-70.5 57.5l-25 24q-10 10 -27.5 30t-71.5 91t-106 151t-122.5 211t-130.5 272q-6 16 -6 27t3 16l4 6 q15 19 57 19l274 2q12 -2 23 -6.5t16 -8.5l5 -3q16 -11 24 -32q20 -50 46 -103.5t41 -81.5l16 -29q29 -60 56 -104t48.5 -68.5t41.5 -38.5t34 -14t27 5q2 1 5 5t12 22t13.5 47t9.5 81t0 125q-2 40 -9 73t-14 46l-6 12q-25 34 -85 43q-13 2 5 24q17 19 38 30q53 26 239 24 q82 -1 135 -13q20 -5 33.5 -13.5t20.5 -24t10.5 -32t3.5 -45.5t-1 -55t-2.5 -70.5t-1.5 -82.5q0 -11 -1 -42t-0.5 -48t3.5 -40.5t11.5 -39t22.5 -24.5q8 -2 17 -4t26 11t38 34.5t52 67t68 107.5q60 104 107 225q4 10 10 17.5t11 10.5l4 3l5 2.5t13 3t20 0.5l288 2 q39 5 64 -2.5t31 -16.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M675 252q21 34 11 69t-45 50q-34 14 -73 1t-60 -46q-22 -34 -13 -68.5t43 -50.5t74.5 -2.5t62.5 47.5zM769 373q8 13 3.5 26.5t-17.5 18.5q-14 5 -28.5 -0.5t-21.5 -18.5q-17 -31 13 -45q14 -5 29 0.5t22 18.5zM943 266q-45 -102 -158 -150t-224 -12 q-107 34 -147.5 126.5t6.5 187.5q47 93 151.5 139t210.5 19q111 -29 158.5 -119.5t2.5 -190.5zM1255 426q-9 96 -89 170t-208.5 109t-274.5 21q-223 -23 -369.5 -141.5t-132.5 -264.5q9 -96 89 -170t208.5 -109t274.5 -21q223 23 369.5 141.5t132.5 264.5zM1563 422 q0 -68 -37 -139.5t-109 -137t-168.5 -117.5t-226 -83t-270.5 -31t-275 33.5t-240.5 93t-171.5 151t-65 199.5q0 115 69.5 245t197.5 258q169 169 341.5 236t246.5 -7q65 -64 20 -209q-4 -14 -1 -20t10 -7t14.5 0.5t13.5 3.5l6 2q139 59 246 59t153 -61q45 -63 0 -178 q-2 -13 -4.5 -20t4.5 -12.5t12 -7.5t17 -6q57 -18 103 -47t80 -81.5t34 -116.5zM1489 1046q42 -47 54.5 -108.5t-6.5 -117.5q-8 -23 -29.5 -34t-44.5 -4q-23 8 -34 29.5t-4 44.5q20 63 -24 111t-107 35q-24 -5 -45 8t-25 37q-5 24 8 44.5t37 25.5q60 13 119 -5.5t101 -65.5z M1670 1209q87 -96 112.5 -222.5t-13.5 -241.5q-9 -27 -34 -40t-52 -4t-40 34t-5 52q28 82 10 172t-80 158q-62 69 -148 95.5t-173 8.5q-28 -6 -52 9.5t-30 43.5t9.5 51.5t43.5 29.5q123 26 244 -11.5t208 -134.5z" /> +<glyph unicode="" d="M1133 -34q-171 -94 -368 -94q-196 0 -367 94q138 87 235.5 211t131.5 268q35 -144 132.5 -268t235.5 -211zM638 1394v-485q0 -252 -126.5 -459.5t-330.5 -306.5q-181 215 -181 495q0 187 83.5 349.5t229.5 269.5t325 137zM1536 638q0 -280 -181 -495 q-204 99 -330.5 306.5t-126.5 459.5v485q179 -30 325 -137t229.5 -269.5t83.5 -349.5z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1402 433q-32 -80 -76 -138t-91 -88.5t-99 -46.5t-101.5 -14.5t-96.5 8.5t-86.5 22t-69.5 27.5t-46 22.5l-17 10q-113 -228 -289.5 -359.5t-384.5 -132.5q-19 0 -32 13t-13 32t13 31.5t32 12.5q173 1 322.5 107.5t251.5 294.5q-36 -14 -72 -23t-83 -13t-91 2.5t-93 28.5 t-92 59t-84.5 100t-74.5 146q114 47 214 57t167.5 -7.5t124.5 -56.5t88.5 -77t56.5 -82q53 131 79 291q-7 -1 -18 -2.5t-46.5 -2.5t-69.5 0.5t-81.5 10t-88.5 23t-84 42.5t-75 65t-54.5 94.5t-28.5 127.5q70 28 133.5 36.5t112.5 -1t92 -30t73.5 -50t56 -61t42 -63t27.5 -56 t16 -39.5l4 -16q12 122 12 195q-8 6 -21.5 16t-49 44.5t-63.5 71.5t-54 93t-33 112.5t12 127t70 138.5q73 -25 127.5 -61.5t84.5 -76.5t48 -85t20.5 -89t-0.5 -85.5t-13 -76.5t-19 -62t-17 -42l-7 -15q1 -5 1 -50.5t-1 -71.5q3 7 10 18.5t30.5 43t50.5 58t71 55.5t91.5 44.5 t112 14.5t132.5 -24q-2 -78 -21.5 -141.5t-50 -104.5t-69.5 -71.5t-81.5 -45.5t-84.5 -24t-80 -9.5t-67.5 1t-46.5 4.5l-17 3q-23 -147 -73 -283q6 7 18 18.5t49.5 41t77.5 52.5t99.5 42t117.5 20t129 -23.5t137 -77.5z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1259 283v-66q0 -85 -57.5 -144.5t-138.5 -59.5h-57l-260 -269v269h-529q-81 0 -138.5 59.5t-57.5 144.5v66h1238zM1259 609v-255h-1238v255h1238zM1259 937v-255h-1238v255h1238zM1259 1077v-67h-1238v67q0 84 57.5 143.5t138.5 59.5h846q81 0 138.5 -59.5t57.5 -143.5z " /> +<glyph unicode="" d="M1152 640q0 -14 -9 -23l-320 -320q-9 -9 -23 -9q-13 0 -22.5 9.5t-9.5 22.5v192h-352q-13 0 -22.5 9.5t-9.5 22.5v192q0 13 9.5 22.5t22.5 9.5h352v192q0 14 9 23t23 9q12 0 24 -10l319 -319q9 -9 9 -23zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198 t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1152 736v-192q0 -13 -9.5 -22.5t-22.5 -9.5h-352v-192q0 -14 -9 -23t-23 -9q-12 0 -24 10l-319 319q-9 9 -9 23t9 23l320 320q9 9 23 9q13 0 22.5 -9.5t9.5 -22.5v-192h352q13 0 22.5 -9.5t9.5 -22.5zM1312 640q0 148 -73 273t-198 198t-273 73t-273 -73t-198 -198 t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1024 960v-640q0 -26 -19 -45t-45 -19q-20 0 -37 12l-448 320q-27 19 -27 52t27 52l448 320q17 12 37 12q26 0 45 -19t19 -45zM1280 160v960q0 13 -9.5 22.5t-22.5 9.5h-960q-13 0 -22.5 -9.5t-9.5 -22.5v-960q0 -13 9.5 -22.5t22.5 -9.5h960q13 0 22.5 9.5t9.5 22.5z M1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" d="M1024 640q0 -106 -75 -181t-181 -75t-181 75t-75 181t75 181t181 75t181 -75t75 -181zM768 1184q-148 0 -273 -73t-198 -198t-73 -273t73 -273t198 -198t273 -73t273 73t198 198t73 273t-73 273t-198 198t-273 73zM1536 640q0 -209 -103 -385.5t-279.5 -279.5 t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1023 349l102 -204q-58 -179 -210 -290t-339 -111q-156 0 -288.5 77.5t-210 210t-77.5 288.5q0 181 104.5 330t274.5 211l17 -131q-122 -54 -195 -165.5t-73 -244.5q0 -185 131.5 -316.5t316.5 -131.5q126 0 232.5 65t165 175.5t49.5 236.5zM1571 249l58 -114l-256 -128 q-13 -7 -29 -7q-40 0 -57 35l-239 477h-472q-24 0 -42.5 16.5t-21.5 40.5l-96 779q-2 16 6 42q14 51 57 82.5t97 31.5q66 0 113 -47t47 -113q0 -69 -52 -117.5t-120 -41.5l37 -289h423v-128h-407l16 -128h455q40 0 57 -35l228 -455z" /> +<glyph unicode="" d="M1254 899q16 85 -21 132q-52 65 -187 45q-17 -3 -41 -12.5t-57.5 -30.5t-64.5 -48.5t-59.5 -70t-44.5 -91.5q80 7 113.5 -16t26.5 -99q-5 -52 -52 -143q-43 -78 -71 -99q-44 -32 -87 14q-23 24 -37.5 64.5t-19 73t-10 84t-8.5 71.5q-23 129 -34 164q-12 37 -35.5 69 t-50.5 40q-57 16 -127 -25q-54 -32 -136.5 -106t-122.5 -102v-7q16 -8 25.5 -26t21.5 -20q21 -3 54.5 8.5t58 10.5t41.5 -30q11 -18 18.5 -38.5t15 -48t12.5 -40.5q17 -46 53 -187q36 -146 57 -197q42 -99 103 -125q43 -12 85 -1.5t76 31.5q131 77 250 237 q104 139 172.5 292.5t82.5 226.5zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1152" d="M1152 704q0 -191 -94.5 -353t-256.5 -256.5t-353 -94.5h-160q-14 0 -23 9t-9 23v611l-215 -66q-3 -1 -9 -1q-10 0 -19 6q-13 10 -13 26v128q0 23 23 31l233 71v93l-215 -66q-3 -1 -9 -1q-10 0 -19 6q-13 10 -13 26v128q0 23 23 31l233 71v250q0 14 9 23t23 9h160 q14 0 23 -9t9 -23v-181l375 116q15 5 28 -5t13 -26v-128q0 -23 -23 -31l-393 -121v-93l375 116q15 5 28 -5t13 -26v-128q0 -23 -23 -31l-393 -121v-487q188 13 318 151t130 328q0 14 9 23t23 9h160q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1408" d="M1152 736v-64q0 -14 -9 -23t-23 -9h-352v-352q0 -14 -9 -23t-23 -9h-64q-14 0 -23 9t-9 23v352h-352q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h352v352q0 14 9 23t23 9h64q14 0 23 -9t9 -23v-352h352q14 0 23 -9t9 -23zM1280 288v832q0 66 -47 113t-113 47h-832 q-66 0 -113 -47t-47 -113v-832q0 -66 47 -113t113 -47h832q66 0 113 47t47 113zM1408 1120v-832q0 -119 -84.5 -203.5t-203.5 -84.5h-832q-119 0 -203.5 84.5t-84.5 203.5v832q0 119 84.5 203.5t203.5 84.5h832q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="2176" d="M620 416q-110 -64 -268 -64h-128v64h-64q-13 0 -22.5 23.5t-9.5 56.5q0 24 7 49q-58 2 -96.5 10.5t-38.5 20.5t38.5 20.5t96.5 10.5q-7 25 -7 49q0 33 9.5 56.5t22.5 23.5h64v64h128q158 0 268 -64h1113q42 -7 106.5 -18t80.5 -14q89 -15 150 -40.5t83.5 -47.5t22.5 -40 t-22.5 -40t-83.5 -47.5t-150 -40.5q-16 -3 -80.5 -14t-106.5 -18h-1113zM1739 668q53 -36 53 -92t-53 -92l81 -30q68 48 68 122t-68 122zM625 400h1015q-217 -38 -456 -80q-57 0 -113 -24t-83 -48l-28 -24l-288 -288q-26 -26 -70.5 -45t-89.5 -19h-96l-93 464h29 q157 0 273 64zM352 816h-29l93 464h96q46 0 90 -19t70 -45l288 -288q4 -4 11 -10.5t30.5 -23t48.5 -29t61.5 -23t72.5 -10.5l456 -80h-1015q-116 64 -273 64z" /> +<glyph unicode="" horiz-adv-x="1664" d="M1519 760q62 0 103.5 -40.5t41.5 -101.5q0 -97 -93 -130l-172 -59l56 -167q7 -21 7 -47q0 -59 -42 -102t-101 -43q-47 0 -85.5 27t-53.5 72l-55 165l-310 -106l55 -164q8 -24 8 -47q0 -59 -42 -102t-102 -43q-47 0 -85 27t-53 72l-55 163l-153 -53q-29 -9 -50 -9 q-61 0 -101.5 40t-40.5 101q0 47 27.5 85t71.5 53l156 53l-105 313l-156 -54q-26 -8 -48 -8q-60 0 -101 40.5t-41 100.5q0 47 27.5 85t71.5 53l157 53l-53 159q-8 24 -8 47q0 60 42 102.5t102 42.5q47 0 85 -27t53 -72l54 -160l310 105l-54 160q-8 24 -8 47q0 59 42.5 102 t101.5 43q47 0 85.5 -27.5t53.5 -71.5l53 -161l162 55q21 6 43 6q60 0 102.5 -39.5t42.5 -98.5q0 -45 -30 -81.5t-74 -51.5l-157 -54l105 -316l164 56q24 8 46 8zM725 498l310 105l-105 315l-310 -107z" /> +<glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960zM1280 352v436q-31 -35 -64 -55q-34 -22 -132.5 -85t-151.5 -99q-98 -69 -164 -69v0v0q-66 0 -164 69 q-46 32 -141.5 92.5t-142.5 92.5q-12 8 -33 27t-31 27v-436q0 -40 28 -68t68 -28h832q40 0 68 28t28 68zM1280 925q0 41 -27.5 70t-68.5 29h-832q-40 0 -68 -28t-28 -68q0 -37 30.5 -76.5t67.5 -64.5q47 -32 137.5 -89t129.5 -83q3 -2 17 -11.5t21 -14t21 -13t23.5 -13 t21.5 -9.5t22.5 -7.5t20.5 -2.5t20.5 2.5t22.5 7.5t21.5 9.5t23.5 13t21 13t21 14t17 11.5l267 174q35 23 66.5 62.5t31.5 73.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M127 640q0 163 67 313l367 -1005q-196 95 -315 281t-119 411zM1415 679q0 -19 -2.5 -38.5t-10 -49.5t-11.5 -44t-17.5 -59t-17.5 -58l-76 -256l-278 826q46 3 88 8q19 2 26 18.5t-2.5 31t-28.5 13.5l-205 -10q-75 1 -202 10q-12 1 -20.5 -5t-11.5 -15t-1.5 -18.5t9 -16.5 t19.5 -8l80 -8l120 -328l-168 -504l-280 832q46 3 88 8q19 2 26 18.5t-2.5 31t-28.5 13.5l-205 -10q-7 0 -23 0.5t-26 0.5q105 160 274.5 253.5t367.5 93.5q147 0 280.5 -53t238.5 -149h-10q-55 0 -92 -40.5t-37 -95.5q0 -12 2 -24t4 -21.5t8 -23t9 -21t12 -22.5t12.5 -21 t14.5 -24t14 -23q63 -107 63 -212zM909 573l237 -647q1 -6 5 -11q-126 -44 -255 -44q-112 0 -217 32zM1570 1009q95 -174 95 -369q0 -209 -104 -385.5t-279 -278.5l235 678q59 169 59 276q0 42 -6 79zM896 1536q182 0 348 -71t286 -191t191 -286t71 -348t-71 -348t-191 -286 t-286 -191t-348 -71t-348 71t-286 191t-191 286t-71 348t71 348t191 286t286 191t348 71zM896 -215q173 0 331.5 68t273 182.5t182.5 273t68 331.5t-68 331.5t-182.5 273t-273 182.5t-331.5 68t-331.5 -68t-273 -182.5t-182.5 -273t-68 -331.5t68 -331.5t182.5 -273 t273 -182.5t331.5 -68z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1086 1536v-1536l-272 -128q-228 20 -414 102t-293 208.5t-107 272.5q0 140 100.5 263.5t275 205.5t391.5 108v-172q-217 -38 -356.5 -150t-139.5 -255q0 -152 154.5 -267t388.5 -145v1360zM1755 954l37 -390l-525 114l147 83q-119 70 -280 99v172q277 -33 481 -157z" /> +<glyph unicode="" horiz-adv-x="2048" d="M960 1536l960 -384v-128h-128q0 -26 -20.5 -45t-48.5 -19h-1526q-28 0 -48.5 19t-20.5 45h-128v128zM256 896h256v-768h128v768h256v-768h128v768h256v-768h128v768h256v-768h59q28 0 48.5 -19t20.5 -45v-64h-1664v64q0 26 20.5 45t48.5 19h59v768zM1851 -64 q28 0 48.5 -19t20.5 -45v-128h-1920v128q0 26 20.5 45t48.5 19h1782z" /> +<glyph unicode="" horiz-adv-x="2304" d="M1774 700l18 -316q4 -69 -82 -128t-235 -93.5t-323 -34.5t-323 34.5t-235 93.5t-82 128l18 316l574 -181q22 -7 48 -7t48 7zM2304 1024q0 -23 -22 -31l-1120 -352q-4 -1 -10 -1t-10 1l-652 206q-43 -34 -71 -111.5t-34 -178.5q63 -36 63 -109q0 -69 -58 -107l58 -433 q2 -14 -8 -25q-9 -11 -24 -11h-192q-15 0 -24 11q-10 11 -8 25l58 433q-58 38 -58 107q0 73 65 111q11 207 98 330l-333 104q-22 8 -22 31t22 31l1120 352q4 1 10 1t10 -1l1120 -352q22 -8 22 -31z" /> +<glyph unicode="" d="M859 579l13 -707q-62 11 -105 11q-41 0 -105 -11l13 707q-40 69 -168.5 295.5t-216.5 374.5t-181 287q58 -15 108 -15q43 0 111 15q63 -111 133.5 -229.5t167 -276.5t138.5 -227q37 61 109.5 177.5t117.5 190t105 176t107 189.5q54 -14 107 -14q56 0 114 14v0 q-28 -39 -60 -88.5t-49.5 -78.5t-56.5 -96t-49 -84q-146 -248 -353 -610z" /> +<glyph unicode="" horiz-adv-x="1280" d="M981 197q0 25 -7 49t-14.5 42t-27 41.5t-29.5 35t-38.5 34.5t-36.5 29t-41.5 30t-36.5 26q-16 2 -49 2q-53 0 -104.5 -7t-107 -25t-97 -46t-68.5 -74.5t-27 -105.5q0 -56 23.5 -102t61 -75.5t87 -50t100 -29t101.5 -8.5q58 0 111.5 13t99 39t73 73t27.5 109zM864 1055 q0 59 -17 125.5t-48 129t-84 103.5t-117 41q-42 0 -82.5 -19.5t-66.5 -52.5q-46 -59 -46 -160q0 -46 10 -97.5t31.5 -103t52 -92.5t75 -67t96.5 -26q37 0 77.5 16.5t65.5 43.5q53 56 53 159zM752 1536h417l-137 -88h-132q75 -63 113 -133t38 -160q0 -72 -24.5 -129.5 t-59.5 -93t-69.5 -65t-59 -61.5t-24.5 -66q0 -36 32 -70.5t77 -68t90.5 -73.5t77.5 -104t32 -142q0 -91 -49 -173q-71 -122 -209.5 -179.5t-298.5 -57.5q-132 0 -246.5 41.5t-172.5 137.5q-36 59 -36 131q0 81 44.5 150t118.5 115q131 82 404 100q-32 41 -47.5 73.5 t-15.5 73.5q0 40 21 85q-46 -4 -68 -4q-148 0 -249.5 96.5t-101.5 244.5q0 82 36 159t99 131q76 66 182 98t218 32z" /> +<glyph unicode="" horiz-adv-x="1984" d="M831 572q0 -56 -40.5 -96t-96.5 -40q-57 0 -98 40t-41 96q0 57 41.5 98t97.5 41t96.5 -41t40.5 -98zM1292 711q56 0 96.5 -41t40.5 -98q0 -56 -40.5 -96t-96.5 -40q-57 0 -98 40t-41 96q0 57 41.5 98t97.5 41zM1984 722q0 -62 -31 -114t-83 -82q5 -33 5 -61 q0 -121 -68.5 -230.5t-197.5 -193.5q-125 -82 -285.5 -125.5t-335.5 -43.5q-176 0 -336.5 43.5t-284.5 125.5q-129 84 -197.5 193t-68.5 231q0 29 5 66q-48 31 -77 81.5t-29 109.5q0 94 66 160t160 66q83 0 148 -55q248 158 592 164l134 423q4 14 17.5 21.5t28.5 4.5 l347 -82q22 50 68.5 81t102.5 31q77 0 131.5 -54.5t54.5 -131.5t-54.5 -132t-131.5 -55q-76 0 -130.5 54t-55.5 131l-315 74l-116 -366q327 -14 560 -166q64 58 151 58q94 0 160 -66t66 -160zM1664 1459q-45 0 -77 -32t-32 -77t32 -77t77 -32t77 32t32 77t-32 77t-77 32z M77 722q0 -67 51 -111q49 131 180 235q-36 25 -82 25q-62 0 -105.5 -43.5t-43.5 -105.5zM1567 105q112 73 171.5 166t59.5 194t-59.5 193.5t-171.5 165.5q-116 75 -265.5 115.5t-313.5 40.5t-313.5 -40.5t-265.5 -115.5q-112 -73 -171.5 -165.5t-59.5 -193.5t59.5 -194 t171.5 -166q116 -75 265.5 -115.5t313.5 -40.5t313.5 40.5t265.5 115.5zM1850 605q57 46 57 117q0 62 -43.5 105.5t-105.5 43.5q-49 0 -86 -28q131 -105 178 -238zM1258 237q11 11 27 11t27 -11t11 -27.5t-11 -27.5q-99 -99 -319 -99h-2q-220 0 -319 99q-11 11 -11 27.5 t11 27.5t27 11t27 -11q77 -77 265 -77h2q188 0 265 77z" /> +<glyph unicode="" d="M950 393q7 7 17.5 7t17.5 -7t7 -18t-7 -18q-65 -64 -208 -64h-1h-1q-143 0 -207 64q-8 7 -8 18t8 18q7 7 17.5 7t17.5 -7q49 -51 172 -51h1h1q122 0 173 51zM671 613q0 -37 -26 -64t-63 -27t-63 27t-26 64t26 63t63 26t63 -26t26 -63zM1214 1049q-29 0 -50 21t-21 50 q0 30 21 51t50 21q30 0 51 -21t21 -51q0 -29 -21 -50t-51 -21zM1216 1408q132 0 226 -94t94 -227v-894q0 -133 -94 -227t-226 -94h-896q-132 0 -226 94t-94 227v894q0 133 94 227t226 94h896zM1321 596q35 14 57 45.5t22 70.5q0 51 -36 87.5t-87 36.5q-60 0 -98 -48 q-151 107 -375 115l83 265l206 -49q1 -50 36.5 -85t84.5 -35q50 0 86 35.5t36 85.5t-36 86t-86 36q-36 0 -66 -20.5t-45 -53.5l-227 54q-9 2 -17.5 -2.5t-11.5 -14.5l-95 -302q-224 -4 -381 -113q-36 43 -93 43q-51 0 -87 -36.5t-36 -87.5q0 -37 19.5 -67.5t52.5 -45.5 q-7 -25 -7 -54q0 -98 74 -181.5t201.5 -132t278.5 -48.5q150 0 277.5 48.5t201.5 132t74 181.5q0 27 -6 54zM971 702q37 0 63 -26t26 -63t-26 -64t-63 -27t-63 27t-26 64t26 63t63 26z" /> +<glyph unicode="" d="M866 697l90 27v62q0 79 -58 135t-138 56t-138 -55.5t-58 -134.5v-283q0 -20 -14 -33.5t-33 -13.5t-32.5 13.5t-13.5 33.5v120h-151v-122q0 -82 57.5 -139t139.5 -57q81 0 138.5 56.5t57.5 136.5v280q0 19 13.5 33t33.5 14q19 0 32.5 -14t13.5 -33v-54zM1199 502v122h-150 v-126q0 -20 -13.5 -33.5t-33.5 -13.5q-19 0 -32.5 14t-13.5 33v123l-90 -26l-60 28v-123q0 -80 58 -137t139 -57t138.5 57t57.5 139zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103 t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1920" d="M1062 824v118q0 42 -30 72t-72 30t-72 -30t-30 -72v-612q0 -175 -126 -299t-303 -124q-178 0 -303.5 125.5t-125.5 303.5v266h328v-262q0 -43 30 -72.5t72 -29.5t72 29.5t30 72.5v620q0 171 126.5 292t301.5 121q176 0 302 -122t126 -294v-136l-195 -58zM1592 602h328 v-266q0 -178 -125.5 -303.5t-303.5 -125.5q-177 0 -303 124.5t-126 300.5v268l131 -61l195 58v-270q0 -42 30 -71.5t72 -29.5t72 29.5t30 71.5v275z" /> +<glyph unicode="" d="M1472 160v480h-704v704h-480q-93 0 -158.5 -65.5t-65.5 -158.5v-480h704v-704h480q93 0 158.5 65.5t65.5 158.5zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5 t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="2048" d="M328 1254h204v-983h-532v697h328v286zM328 435v369h-123v-369h123zM614 968v-697h205v697h-205zM614 1254v-204h205v204h-205zM901 968h533v-942h-533v163h328v82h-328v697zM1229 435v369h-123v-369h123zM1516 968h532v-942h-532v163h327v82h-327v697zM1843 435v369h-123 v-369h123z" /> +<glyph unicode="" d="M1046 516q0 -64 -38 -109t-91 -45q-43 0 -70 15v277q28 17 70 17q53 0 91 -45.5t38 -109.5zM703 944q0 -64 -38 -109.5t-91 -45.5q-43 0 -70 15v277q28 17 70 17q53 0 91 -45t38 -109zM1265 513q0 134 -88 229t-213 95q-20 0 -39 -3q-23 -78 -78 -136q-87 -95 -211 -101 v-636l211 41v206q51 -19 117 -19q125 0 213 95t88 229zM922 940q0 134 -88.5 229t-213.5 95q-74 0 -141 -36h-186v-840l211 41v206q55 -19 116 -19q125 0 213.5 95t88.5 229zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960 q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="2038" d="M1222 607q75 3 143.5 -20.5t118 -58.5t101 -94.5t84 -108t75.5 -120.5q33 -56 78.5 -109t75.5 -80.5t99 -88.5q-48 -30 -108.5 -57.5t-138.5 -59t-114 -47.5q-44 37 -74 115t-43.5 164.5t-33 180.5t-42.5 168.5t-72.5 123t-122.5 48.5l-10 -2l-6 -4q4 -5 13 -14 q6 -5 28 -23.5t25.5 -22t19 -18t18 -20.5t11.5 -21t10.5 -27.5t4.5 -31t4 -40.5l1 -33q1 -26 -2.5 -57.5t-7.5 -52t-12.5 -58.5t-11.5 -53q-35 1 -101 -9.5t-98 -10.5q-39 0 -72 10q-2 16 -2 47q0 74 3 96q2 13 31.5 41.5t57 59t26.5 51.5q-24 2 -43 -24 q-36 -53 -111.5 -99.5t-136.5 -46.5q-25 0 -75.5 63t-106.5 139.5t-84 96.5q-6 4 -27 30q-482 -112 -513 -112q-16 0 -28 11t-12 27q0 15 8.5 26.5t22.5 14.5l486 106q-8 14 -8 25t5.5 17.5t16 11.5t20 7t23 4.5t18.5 4.5q4 1 15.5 7.5t17.5 6.5q15 0 28 -16t20 -33 q163 37 172 37q17 0 29.5 -11t12.5 -28q0 -15 -8.5 -26t-23.5 -14l-182 -40l-1 -16q-1 -26 81.5 -117.5t104.5 -91.5q47 0 119 80t72 129q0 36 -23.5 53t-51 18.5t-51 11.5t-23.5 34q0 16 10 34l-68 19q43 44 43 117q0 26 -5 58q82 16 144 16q44 0 71.5 -1.5t48.5 -8.5 t31 -13.5t20.5 -24.5t15.5 -33.5t17 -47.5t24 -60l50 25q-3 -40 -23 -60t-42.5 -21t-40 -6.5t-16.5 -20.5zM1282 842q-5 5 -13.5 15.5t-12 14.5t-10.5 11.5t-10 10.5l-8 8t-8.5 7.5t-8 5t-8.5 4.5q-7 3 -14.5 5t-20.5 2.5t-22 0.5h-32.5h-37.5q-126 0 -217 -43 q16 30 36 46.5t54 29.5t65.5 36t46 36.5t50 55t43.5 50.5q12 -9 28 -31.5t32 -36.5t38 -13l12 1v-76l22 -1q247 95 371 190q28 21 50 39t42.5 37.5t33 31t29.5 34t24 31t24.5 37t23 38t27 47.5t29.5 53l7 9q-2 -53 -43 -139q-79 -165 -205 -264t-306 -142q-14 -3 -42 -7.5 t-50 -9.5t-39 -14q3 -19 24.5 -46t21.5 -34q0 -11 -26 -30zM1061 -79q39 26 131.5 47.5t146.5 21.5q9 0 22.5 -15.5t28 -42.5t26 -50t24 -51t14.5 -33q-121 -45 -244 -45q-61 0 -125 11zM822 568l48 12l109 -177l-73 -48zM1323 51q3 -15 3 -16q0 -7 -17.5 -14.5t-46 -13 t-54 -9.5t-53.5 -7.5t-32 -4.5l-7 43q21 2 60.5 8.5t72 10t60.5 3.5h14zM866 679l-96 -20l-6 17q10 1 32.5 7t34.5 6q19 0 35 -10zM1061 45h31l10 -83l-41 -12v95zM1950 1535v1v-1zM1950 1535l-1 -5l-2 -2l1 3zM1950 1535l1 1z" /> +<glyph unicode="" d="M1167 -50q-5 19 -24 5q-30 -22 -87 -39t-131 -17q-129 0 -193 49q-5 4 -13 4q-11 0 -26 -12q-7 -6 -7.5 -16t7.5 -20q34 -32 87.5 -46t102.5 -12.5t99 4.5q41 4 84.5 20.5t65 30t28.5 20.5q12 12 7 29zM1128 65q-19 47 -39 61q-23 15 -76 15q-47 0 -71 -10 q-29 -12 -78 -56q-26 -24 -12 -44q9 -8 17.5 -4.5t31.5 23.5q3 2 10.5 8.5t10.5 8.5t10 7t11.5 7t12.5 5t15 4.5t16.5 2.5t20.5 1q27 0 44.5 -7.5t23 -14.5t13.5 -22q10 -17 12.5 -20t12.5 1q23 12 14 34zM1483 346q0 22 -5 44.5t-16.5 45t-34 36.5t-52.5 14 q-33 0 -97 -41.5t-129 -83.5t-101 -42q-27 -1 -63.5 19t-76 49t-83.5 58t-100 49t-111 19q-115 -1 -197 -78.5t-84 -178.5q-2 -112 74 -164q29 -20 62.5 -28.5t103.5 -8.5q57 0 132 32.5t134 71t120 70.5t93 31q26 -1 65 -31.5t71.5 -67t68 -67.5t55.5 -32q35 -3 58.5 14 t55.5 63q28 41 42.5 101t14.5 106zM1536 506q0 -164 -62 -304.5t-166 -236t-242.5 -149.5t-290.5 -54t-293 57.5t-247.5 157t-170.5 241.5t-64 302q0 89 19.5 172.5t49 145.5t70.5 118.5t78.5 94t78.5 69.5t64.5 46.5t42.5 24.5q14 8 51 26.5t54.5 28.5t48 30t60.5 44 q36 28 58 72.5t30 125.5q129 -155 186 -193q44 -29 130 -68t129 -66q21 -13 39 -25t60.5 -46.5t76 -70.5t75 -95t69 -122t47 -148.5t19.5 -177.5z" /> +<glyph unicode="" d="M1070 463l-160 -160l-151 -152l-30 -30q-65 -64 -151.5 -87t-171.5 -2q-16 -70 -72 -115t-129 -45q-85 0 -145 60.5t-60 145.5q0 72 44.5 128t113.5 72q-22 86 1 173t88 152l12 12l151 -152l-11 -11q-37 -37 -37 -89t37 -90q37 -37 89 -37t89 37l30 30l151 152l161 160z M729 1145l12 -12l-152 -152l-12 12q-37 37 -89 37t-89 -37t-37 -89.5t37 -89.5l29 -29l152 -152l160 -160l-151 -152l-161 160l-151 152l-30 30q-68 67 -90 159.5t5 179.5q-70 15 -115 71t-45 129q0 85 60 145.5t145 60.5q76 0 133.5 -49t69.5 -123q84 20 169.5 -3.5 t149.5 -87.5zM1536 78q0 -85 -60 -145.5t-145 -60.5q-74 0 -131 47t-71 118q-86 -28 -179.5 -6t-161.5 90l-11 12l151 152l12 -12q37 -37 89 -37t89 37t37 89t-37 89l-30 30l-152 152l-160 160l152 152l160 -160l152 -152l29 -30q64 -64 87.5 -150.5t2.5 -171.5 q76 -11 126.5 -68.5t50.5 -134.5zM1534 1202q0 -77 -51 -135t-127 -69q26 -85 3 -176.5t-90 -158.5l-12 -12l-151 152l12 12q37 37 37 89t-37 89t-89 37t-89 -37l-30 -30l-152 -152l-160 -160l-152 152l161 160l152 152l29 30q67 67 159 89.5t178 -3.5q11 75 68.5 126 t135.5 51q85 0 145 -60.5t60 -145.5z" /> +<glyph unicode="" d="M654 458q-1 -3 -12.5 0.5t-31.5 11.5l-20 9q-44 20 -87 49q-7 5 -41 31.5t-38 28.5q-67 -103 -134 -181q-81 -95 -105 -110q-4 -2 -19.5 -4t-18.5 0q6 4 82 92q21 24 85.5 115t78.5 118q17 30 51 98.5t36 77.5q-8 1 -110 -33q-8 -2 -27.5 -7.5t-34.5 -9.5t-17 -5 q-2 -2 -2 -10.5t-1 -9.5q-5 -10 -31 -15q-23 -7 -47 0q-18 4 -28 21q-4 6 -5 23q6 2 24.5 5t29.5 6q58 16 105 32q100 35 102 35q10 2 43 19.5t44 21.5q9 3 21.5 8t14.5 5.5t6 -0.5q2 -12 -1 -33q0 -2 -12.5 -27t-26.5 -53.5t-17 -33.5q-25 -50 -77 -131l64 -28 q12 -6 74.5 -32t67.5 -28q4 -1 10.5 -25.5t4.5 -30.5zM449 944q3 -15 -4 -28q-12 -23 -50 -38q-30 -12 -60 -12q-26 3 -49 26q-14 15 -18 41l1 3q3 -3 19.5 -5t26.5 0t58 16q36 12 55 14q17 0 21 -17zM1147 815l63 -227l-139 42zM39 15l694 232v1032l-694 -233v-1031z M1280 332l102 -31l-181 657l-100 31l-216 -536l102 -31l45 110l211 -65zM777 1294l573 -184v380zM1088 -29l158 -13l-54 -160l-40 66q-130 -83 -276 -108q-58 -12 -91 -12h-84q-79 0 -199.5 39t-183.5 85q-8 7 -8 16q0 8 5 13.5t13 5.5q4 0 18 -7.5t30.5 -16.5t20.5 -11 q73 -37 159.5 -61.5t157.5 -24.5q95 0 167 14.5t157 50.5q15 7 30.5 15.5t34 19t28.5 16.5zM1536 1050v-1079l-774 246q-14 -6 -375 -127.5t-368 -121.5q-13 0 -18 13q0 1 -1 3v1078q3 9 4 10q5 6 20 11q106 35 149 50v384l558 -198q2 0 160.5 55t316 108.5t161.5 53.5 q20 0 20 -21v-418z" /> +<glyph unicode="" horiz-adv-x="1792" d="M288 1152q66 0 113 -47t47 -113v-1088q0 -66 -47 -113t-113 -47h-128q-66 0 -113 47t-47 113v1088q0 66 47 113t113 47h128zM1664 989q58 -34 93 -93t35 -128v-768q0 -106 -75 -181t-181 -75h-864q-66 0 -113 47t-47 113v1536q0 40 28 68t68 28h672q40 0 88 -20t76 -48 l152 -152q28 -28 48 -76t20 -88v-163zM928 0v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM928 256v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM928 512v128q0 14 -9 23 t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM1184 0v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM1184 256v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128 q14 0 23 9t9 23zM1184 512v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM1440 0v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM1440 256v128q0 14 -9 23t-23 9h-128 q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM1440 512v128q0 14 -9 23t-23 9h-128q-14 0 -23 -9t-9 -23v-128q0 -14 9 -23t23 -9h128q14 0 23 9t9 23zM1536 896v256h-160q-40 0 -68 28t-28 68v160h-640v-512h896z" /> +<glyph unicode="" d="M1344 1536q26 0 45 -19t19 -45v-1664q0 -26 -19 -45t-45 -19h-1280q-26 0 -45 19t-19 45v1664q0 26 19 45t45 19h1280zM512 1248v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23zM512 992v-64q0 -14 9 -23t23 -9h64q14 0 23 9 t9 23v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23zM512 736v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23zM512 480v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23zM384 160v64 q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM384 416v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM384 672v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64 q14 0 23 9t9 23zM384 928v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM384 1184v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM896 -96v192q0 14 -9 23t-23 9h-320q-14 0 -23 -9 t-9 -23v-192q0 -14 9 -23t23 -9h320q14 0 23 9t9 23zM896 416v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM896 672v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM896 928v64 q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM896 1184v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1152 160v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64 q14 0 23 9t9 23zM1152 416v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1152 672v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1152 928v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9 t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1152 1184v64q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-64q0 -14 9 -23t23 -9h64q14 0 23 9t9 23z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1188 988l-292 -292v-824q0 -46 -33 -79t-79 -33t-79 33t-33 79v384h-64v-384q0 -46 -33 -79t-79 -33t-79 33t-33 79v824l-292 292q-28 28 -28 68t28 68t68 28t68 -28l228 -228h368l228 228q28 28 68 28t68 -28t28 -68t-28 -68zM864 1152q0 -93 -65.5 -158.5 t-158.5 -65.5t-158.5 65.5t-65.5 158.5t65.5 158.5t158.5 65.5t158.5 -65.5t65.5 -158.5z" /> +<glyph unicode="" horiz-adv-x="1664" d="M780 1064q0 -60 -19 -113.5t-63 -92.5t-105 -39q-76 0 -138 57.5t-92 135.5t-30 151q0 60 19 113.5t63 92.5t105 39q77 0 138.5 -57.5t91.5 -135t30 -151.5zM438 581q0 -80 -42 -139t-119 -59q-76 0 -141.5 55.5t-100.5 133.5t-35 152q0 80 42 139.5t119 59.5 q76 0 141.5 -55.5t100.5 -134t35 -152.5zM832 608q118 0 255 -97.5t229 -237t92 -254.5q0 -46 -17 -76.5t-48.5 -45t-64.5 -20t-76 -5.5q-68 0 -187.5 45t-182.5 45q-66 0 -192.5 -44.5t-200.5 -44.5q-183 0 -183 146q0 86 56 191.5t139.5 192.5t187.5 146t193 59zM1071 819 q-61 0 -105 39t-63 92.5t-19 113.5q0 74 30 151.5t91.5 135t138.5 57.5q61 0 105 -39t63 -92.5t19 -113.5q0 -73 -30 -151t-92 -135.5t-138 -57.5zM1503 923q77 0 119 -59.5t42 -139.5q0 -74 -35 -152t-100.5 -133.5t-141.5 -55.5q-77 0 -119 59t-42 139q0 74 35 152.5 t100.5 134t141.5 55.5z" /> +<glyph unicode="" horiz-adv-x="768" d="M704 1008q0 -145 -57 -243.5t-152 -135.5l45 -821q2 -26 -16 -45t-44 -19h-192q-26 0 -44 19t-16 45l45 821q-95 37 -152 135.5t-57 243.5q0 128 42.5 249.5t117.5 200t160 78.5t160 -78.5t117.5 -200t42.5 -249.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M896 -93l640 349v636l-640 -233v-752zM832 772l698 254l-698 254l-698 -254zM1664 1024v-768q0 -35 -18 -65t-49 -47l-704 -384q-28 -16 -61 -16t-61 16l-704 384q-31 17 -49 47t-18 65v768q0 40 23 73t61 47l704 256q22 8 44 8t44 -8l704 -256q38 -14 61 -47t23 -73z " /> +<glyph unicode="" horiz-adv-x="2304" d="M640 -96l384 192v314l-384 -164v-342zM576 358l404 173l-404 173l-404 -173zM1664 -96l384 192v314l-384 -164v-342zM1600 358l404 173l-404 173l-404 -173zM1152 651l384 165v266l-384 -164v-267zM1088 1030l441 189l-441 189l-441 -189zM2176 512v-416q0 -36 -19 -67 t-52 -47l-448 -224q-25 -14 -57 -14t-57 14l-448 224q-5 2 -7 4q-2 -2 -7 -4l-448 -224q-25 -14 -57 -14t-57 14l-448 224q-33 16 -52 47t-19 67v416q0 38 21.5 70t56.5 48l434 186v400q0 38 21.5 70t56.5 48l448 192q23 10 50 10t50 -10l448 -192q35 -16 56.5 -48t21.5 -70 v-400l434 -186q36 -16 57 -48t21 -70z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1848 1197h-511v-124h511v124zM1596 771q-90 0 -146 -52.5t-62 -142.5h408q-18 195 -200 195zM1612 186q63 0 122 32t76 87h221q-100 -307 -427 -307q-214 0 -340.5 132t-126.5 347q0 208 130.5 345.5t336.5 137.5q138 0 240.5 -68t153 -179t50.5 -248q0 -17 -2 -47h-658 q0 -111 57.5 -171.5t166.5 -60.5zM277 236h296q205 0 205 167q0 180 -199 180h-302v-347zM277 773h281q78 0 123.5 36.5t45.5 113.5q0 144 -190 144h-260v-294zM0 1282h594q87 0 155 -14t126.5 -47.5t90 -96.5t31.5 -154q0 -181 -172 -263q114 -32 172 -115t58 -204 q0 -75 -24.5 -136.5t-66 -103.5t-98.5 -71t-121 -42t-134 -13h-611v1260z" /> +<glyph unicode="" d="M1248 1408q119 0 203.5 -84.5t84.5 -203.5v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960zM499 1041h-371v-787h382q117 0 197 57.5t80 170.5q0 158 -143 200q107 52 107 164q0 57 -19.5 96.5 t-56.5 60.5t-79 29.5t-97 8.5zM477 723h-176v184h163q119 0 119 -90q0 -94 -106 -94zM486 388h-185v217h189q124 0 124 -113q0 -104 -128 -104zM1136 356q-68 0 -104 38t-36 107h411q1 10 1 30q0 132 -74.5 220.5t-203.5 88.5q-128 0 -210 -86t-82 -216q0 -135 79 -217 t213 -82q205 0 267 191h-138q-11 -34 -47.5 -54t-75.5 -20zM1126 722q113 0 124 -122h-254q4 56 39 89t91 33zM964 988h319v-77h-319v77z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1582 954q0 -101 -71.5 -172.5t-172.5 -71.5t-172.5 71.5t-71.5 172.5t71.5 172.5t172.5 71.5t172.5 -71.5t71.5 -172.5zM812 212q0 104 -73 177t-177 73q-27 0 -54 -6l104 -42q77 -31 109.5 -106.5t1.5 -151.5q-31 -77 -107 -109t-152 -1q-21 8 -62 24.5t-61 24.5 q32 -60 91 -96.5t130 -36.5q104 0 177 73t73 177zM1642 953q0 126 -89.5 215.5t-215.5 89.5q-127 0 -216.5 -89.5t-89.5 -215.5q0 -127 89.5 -216t216.5 -89q126 0 215.5 89t89.5 216zM1792 953q0 -189 -133.5 -322t-321.5 -133l-437 -319q-12 -129 -109 -218t-229 -89 q-121 0 -214 76t-118 192l-230 92v429l389 -157q79 48 173 48q13 0 35 -2l284 407q2 187 135.5 319t320.5 132q188 0 321.5 -133.5t133.5 -321.5z" /> +<glyph unicode="" d="M1242 889q0 80 -57 136.5t-137 56.5t-136.5 -57t-56.5 -136q0 -80 56.5 -136.5t136.5 -56.5t137 56.5t57 136.5zM632 301q0 -83 -58 -140.5t-140 -57.5q-56 0 -103 29t-72 77q52 -20 98 -40q60 -24 120 1.5t85 86.5q24 60 -1.5 120t-86.5 84l-82 33q22 5 42 5 q82 0 140 -57.5t58 -140.5zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v153l172 -69q20 -92 93.5 -152t168.5 -60q104 0 181 70t87 173l345 252q150 0 255.5 105.5t105.5 254.5q0 150 -105.5 255.5t-255.5 105.5 q-148 0 -253 -104.5t-107 -252.5l-225 -322q-9 1 -28 1q-75 0 -137 -37l-297 119v468q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5zM1289 887q0 -100 -71 -170.5t-171 -70.5t-170.5 70.5t-70.5 170.5t70.5 171t170.5 71q101 0 171.5 -70.5t70.5 -171.5z " /> +<glyph unicode="" horiz-adv-x="1792" d="M836 367l-15 -368l-2 -22l-420 29q-36 3 -67 31.5t-47 65.5q-11 27 -14.5 55t4 65t12 55t21.5 64t19 53q78 -12 509 -28zM449 953l180 -379l-147 92q-63 -72 -111.5 -144.5t-72.5 -125t-39.5 -94.5t-18.5 -63l-4 -21l-190 357q-17 26 -18 56t6 47l8 18q35 63 114 188 l-140 86zM1680 436l-188 -359q-12 -29 -36.5 -46.5t-43.5 -20.5l-18 -4q-71 -7 -219 -12l8 -164l-230 367l211 362l7 -173q170 -16 283 -5t170 33zM895 1360q-47 -63 -265 -435l-317 187l-19 12l225 356q20 31 60 45t80 10q24 -2 48.5 -12t42 -21t41.5 -33t36 -34.5 t36 -39.5t32 -35zM1550 1053l212 -363q18 -37 12.5 -76t-27.5 -74q-13 -20 -33 -37t-38 -28t-48.5 -22t-47 -16t-51.5 -14t-46 -12q-34 72 -265 436l313 195zM1407 1279l142 83l-220 -373l-419 20l151 86q-34 89 -75 166t-75.5 123.5t-64.5 80t-47 46.5l-17 13l405 -1 q31 3 58 -10.5t39 -28.5l11 -15q39 -61 112 -190z" /> +<glyph unicode="" horiz-adv-x="2048" d="M480 448q0 66 -47 113t-113 47t-113 -47t-47 -113t47 -113t113 -47t113 47t47 113zM516 768h1016l-89 357q-2 8 -14 17.5t-21 9.5h-768q-9 0 -21 -9.5t-14 -17.5zM1888 448q0 66 -47 113t-113 47t-113 -47t-47 -113t47 -113t113 -47t113 47t47 113zM2048 544v-384 q0 -14 -9 -23t-23 -9h-96v-128q0 -80 -56 -136t-136 -56t-136 56t-56 136v128h-1024v-128q0 -80 -56 -136t-136 -56t-136 56t-56 136v128h-96q-14 0 -23 9t-9 23v384q0 93 65.5 158.5t158.5 65.5h28l105 419q23 94 104 157.5t179 63.5h768q98 0 179 -63.5t104 -157.5 l105 -419h28q93 0 158.5 -65.5t65.5 -158.5z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1824 640q93 0 158.5 -65.5t65.5 -158.5v-384q0 -14 -9 -23t-23 -9h-96v-64q0 -80 -56 -136t-136 -56t-136 56t-56 136v64h-1024v-64q0 -80 -56 -136t-136 -56t-136 56t-56 136v64h-96q-14 0 -23 9t-9 23v384q0 93 65.5 158.5t158.5 65.5h28l105 419q23 94 104 157.5 t179 63.5h128v224q0 14 9 23t23 9h448q14 0 23 -9t9 -23v-224h128q98 0 179 -63.5t104 -157.5l105 -419h28zM320 160q66 0 113 47t47 113t-47 113t-113 47t-113 -47t-47 -113t47 -113t113 -47zM516 640h1016l-89 357q-2 8 -14 17.5t-21 9.5h-768q-9 0 -21 -9.5t-14 -17.5z M1728 160q66 0 113 47t47 113t-47 113t-113 47t-113 -47t-47 -113t47 -113t113 -47z" /> +<glyph unicode="" d="M1504 64q0 -26 -19 -45t-45 -19h-462q1 -17 6 -87.5t5 -108.5q0 -25 -18 -42.5t-43 -17.5h-320q-25 0 -43 17.5t-18 42.5q0 38 5 108.5t6 87.5h-462q-26 0 -45 19t-19 45t19 45l402 403h-229q-26 0 -45 19t-19 45t19 45l402 403h-197q-26 0 -45 19t-19 45t19 45l384 384 q19 19 45 19t45 -19l384 -384q19 -19 19 -45t-19 -45t-45 -19h-197l402 -403q19 -19 19 -45t-19 -45t-45 -19h-229l402 -403q19 -19 19 -45z" /> +<glyph unicode="" d="M1127 326q0 32 -30 51q-193 115 -447 115q-133 0 -287 -34q-42 -9 -42 -52q0 -20 13.5 -34.5t35.5 -14.5q5 0 37 8q132 27 243 27q226 0 397 -103q19 -11 33 -11q19 0 33 13.5t14 34.5zM1223 541q0 40 -35 61q-237 141 -548 141q-153 0 -303 -42q-48 -13 -48 -64 q0 -25 17.5 -42.5t42.5 -17.5q7 0 37 8q122 33 251 33q279 0 488 -124q24 -13 38 -13q25 0 42.5 17.5t17.5 42.5zM1331 789q0 47 -40 70q-126 73 -293 110.5t-343 37.5q-204 0 -364 -47q-23 -7 -38.5 -25.5t-15.5 -48.5q0 -31 20.5 -52t51.5 -21q11 0 40 8q133 37 307 37 q159 0 309.5 -34t253.5 -95q21 -12 40 -12q29 0 50.5 20.5t21.5 51.5zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M1397 1408q58 0 98.5 -40.5t40.5 -98.5v-1258q0 -58 -40.5 -98.5t-98.5 -40.5h-1258q-58 0 -98.5 40.5t-40.5 98.5v1258q0 58 40.5 98.5t98.5 40.5h1258zM1465 11v1258q0 28 -20 48t-48 20h-1258q-28 0 -48 -20t-20 -48v-1258q0 -28 20 -48t48 -20h1258q28 0 48 20t20 48 zM694 749l188 -387l533 145v-496q0 -7 -5.5 -12.5t-12.5 -5.5h-1258q-7 0 -12.5 5.5t-5.5 12.5v141l711 195l-212 439q4 1 12 2.5t12 1.5q170 32 303.5 21.5t221 -46t143.5 -94.5q27 -28 -25 -42q-64 -16 -256 -62l-97 198q-111 7 -240 -16zM1397 1287q7 0 12.5 -5.5 t5.5 -12.5v-428q-85 30 -188 52q-294 64 -645 12l-18 -3l-65 134h-233l85 -190q-132 -51 -230 -137v560q0 7 5.5 12.5t12.5 5.5h1258zM286 387q-14 -3 -26 4.5t-14 21.5q-24 203 166 305l129 -270z" /> +<glyph unicode="" horiz-adv-x="2304" d="M784 164l16 241l-16 523q-1 10 -7.5 17t-16.5 7q-9 0 -16 -7t-7 -17l-14 -523l14 -241q1 -10 7.5 -16.5t15.5 -6.5q22 0 24 23zM1080 193l11 211l-12 586q0 16 -13 24q-8 5 -16 5t-16 -5q-13 -8 -13 -24l-1 -6l-10 -579q0 -1 11 -236v-1q0 -10 6 -17q9 -11 23 -11 q11 0 20 9q9 7 9 20zM35 533l20 -128l-20 -126q-2 -9 -9 -9t-9 9l-17 126l17 128q2 9 9 9t9 -9zM121 612l26 -207l-26 -203q-2 -9 -10 -9q-9 0 -9 10l-23 202l23 207q0 9 9 9q8 0 10 -9zM401 159zM213 650l25 -245l-25 -237q0 -11 -11 -11q-10 0 -12 11l-21 237l21 245 q2 12 12 12q11 0 11 -12zM307 657l23 -252l-23 -244q-2 -13 -14 -13q-13 0 -13 13l-21 244l21 252q0 13 13 13q12 0 14 -13zM401 639l21 -234l-21 -246q-2 -16 -16 -16q-6 0 -10.5 4.5t-4.5 11.5l-20 246l20 234q0 6 4.5 10.5t10.5 4.5q14 0 16 -15zM784 164zM495 785 l21 -380l-21 -246q0 -7 -5 -12.5t-12 -5.5q-16 0 -18 18l-18 246l18 380q2 18 18 18q7 0 12 -5.5t5 -12.5zM589 871l19 -468l-19 -244q0 -8 -5.5 -13.5t-13.5 -5.5q-18 0 -20 19l-16 244l16 468q2 19 20 19q8 0 13.5 -5.5t5.5 -13.5zM687 911l18 -506l-18 -242 q-2 -21 -22 -21q-19 0 -21 21l-16 242l16 506q0 9 6.5 15.5t14.5 6.5q9 0 15 -6.5t7 -15.5zM1079 169v0v0zM881 915l15 -510l-15 -239q0 -10 -7.5 -17.5t-17.5 -7.5t-17 7t-8 18l-14 239l14 510q0 11 7.5 18t17.5 7t17.5 -7t7.5 -18zM980 896l14 -492l-14 -236q0 -11 -8 -19 t-19 -8t-19 8t-9 19l-12 236l12 492q1 12 9 20t19 8t18.5 -8t8.5 -20zM1192 404l-14 -231v0q0 -13 -9 -22t-22 -9t-22 9t-10 22l-6 114l-6 117l12 636v3q2 15 12 24q9 7 20 7q8 0 15 -5q14 -8 16 -26zM2304 423q0 -117 -83 -199.5t-200 -82.5h-786q-13 2 -22 11t-9 22v899 q0 23 28 33q85 34 181 34q195 0 338 -131.5t160 -323.5q53 22 110 22q117 0 200 -83t83 -201z" /> +<glyph unicode="" d="M768 768q237 0 443 43t325 127v-170q0 -69 -103 -128t-280 -93.5t-385 -34.5t-385 34.5t-280 93.5t-103 128v170q119 -84 325 -127t443 -43zM768 0q237 0 443 43t325 127v-170q0 -69 -103 -128t-280 -93.5t-385 -34.5t-385 34.5t-280 93.5t-103 128v170q119 -84 325 -127 t443 -43zM768 384q237 0 443 43t325 127v-170q0 -69 -103 -128t-280 -93.5t-385 -34.5t-385 34.5t-280 93.5t-103 128v170q119 -84 325 -127t443 -43zM768 1536q208 0 385 -34.5t280 -93.5t103 -128v-128q0 -69 -103 -128t-280 -93.5t-385 -34.5t-385 34.5t-280 93.5 t-103 128v128q0 69 103 128t280 93.5t385 34.5z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M894 465q33 -26 84 -56q59 7 117 7q147 0 177 -49q16 -22 2 -52q0 -1 -1 -2l-2 -2v-1q-6 -38 -71 -38q-48 0 -115 20t-130 53q-221 -24 -392 -83q-153 -262 -242 -262q-15 0 -28 7l-24 12q-1 1 -6 5q-10 10 -6 36q9 40 56 91.5t132 96.5q14 9 23 -6q2 -2 2 -4q52 85 107 197 q68 136 104 262q-24 82 -30.5 159.5t6.5 127.5q11 40 42 40h21h1q23 0 35 -15q18 -21 9 -68q-2 -6 -4 -8q1 -3 1 -8v-30q-2 -123 -14 -192q55 -164 146 -238zM318 54q52 24 137 158q-51 -40 -87.5 -84t-49.5 -74zM716 974q-15 -42 -2 -132q1 7 7 44q0 3 7 43q1 4 4 8 q-1 1 -1 2t-0.5 1.5t-0.5 1.5q-1 22 -13 36q0 -1 -1 -2v-2zM592 313q135 54 284 81q-2 1 -13 9.5t-16 13.5q-76 67 -127 176q-27 -86 -83 -197q-30 -56 -45 -83zM1238 329q-24 24 -140 24q76 -28 124 -28q14 0 18 1q0 1 -2 3z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M233 768v-107h70l164 -661h159l128 485q7 20 10 46q2 16 2 24h4l3 -24q1 -3 3.5 -20t5.5 -26l128 -485h159l164 661h70v107h-300v-107h90l-99 -438q-5 -20 -7 -46l-2 -21h-4l-3 21q-1 5 -4 21t-5 25l-144 545h-114l-144 -545q-2 -9 -4.5 -24.5t-3.5 -21.5l-4 -21h-4l-2 21 q-2 26 -7 46l-99 438h90v107h-300z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M429 106v-106h281v106h-75l103 161q5 7 10 16.5t7.5 13.5t3.5 4h2q1 -4 5 -10q2 -4 4.5 -7.5t6 -8t6.5 -8.5l107 -161h-76v-106h291v106h-68l-192 273l195 282h67v107h-279v-107h74l-103 -159q-4 -7 -10 -16.5t-9 -13.5l-2 -3h-2q-1 4 -5 10q-6 11 -17 23l-106 159h76v107 h-290v-107h68l189 -272l-194 -283h-68z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M416 106v-106h327v106h-93v167h137q76 0 118 15q67 23 106.5 87t39.5 146q0 81 -37 141t-100 87q-48 19 -130 19h-368v-107h92v-555h-92zM769 386h-119v268h120q52 0 83 -18q56 -33 56 -115q0 -89 -62 -120q-31 -15 -78 -15z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M1280 320v-320h-1024v192l192 192l128 -128l384 384zM448 512q-80 0 -136 56t-56 136t56 136t136 56t136 -56t56 -136t-56 -136t-136 -56z" /> +<glyph unicode="" d="M640 1152v128h-128v-128h128zM768 1024v128h-128v-128h128zM640 896v128h-128v-128h128zM768 768v128h-128v-128h128zM1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400 v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-128v-128h-128v128h-512v-1536h1280zM781 593l107 -349q8 -27 8 -52q0 -83 -72.5 -137.5t-183.5 -54.5t-183.5 54.5t-72.5 137.5q0 25 8 52q21 63 120 396v128h128v-128h79 q22 0 39 -13t23 -34zM640 128q53 0 90.5 19t37.5 45t-37.5 45t-90.5 19t-90.5 -19t-37.5 -45t37.5 -45t90.5 -19z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M620 686q20 -8 20 -30v-544q0 -22 -20 -30q-8 -2 -12 -2q-12 0 -23 9l-166 167h-131q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h131l166 167q16 15 35 7zM1037 -3q31 0 50 24q129 159 129 363t-129 363q-16 21 -43 24t-47 -14q-21 -17 -23.5 -43.5t14.5 -47.5 q100 -123 100 -282t-100 -282q-17 -21 -14.5 -47.5t23.5 -42.5q18 -15 40 -15zM826 145q27 0 47 20q87 93 87 219t-87 219q-18 19 -45 20t-46 -17t-20 -44.5t18 -46.5q52 -57 52 -131t-52 -131q-19 -20 -18 -46.5t20 -44.5q20 -17 44 -17z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M768 768q52 0 90 -38t38 -90v-384q0 -52 -38 -90t-90 -38h-384q-52 0 -90 38t-38 90v384q0 52 38 90t90 38h384zM1260 766q20 -8 20 -30v-576q0 -22 -20 -30q-8 -2 -12 -2q-14 0 -23 9l-265 266v90l265 266q9 9 23 9q4 0 12 -2z" /> +<glyph unicode="" d="M1468 1156q28 -28 48 -76t20 -88v-1152q0 -40 -28 -68t-68 -28h-1344q-40 0 -68 28t-28 68v1600q0 40 28 68t68 28h896q40 0 88 -20t76 -48zM1024 1400v-376h376q-10 29 -22 41l-313 313q-12 12 -41 22zM1408 -128v1024h-416q-40 0 -68 28t-28 68v416h-768v-1536h1280z M480 768q8 11 21 12.5t24 -6.5l51 -38q11 -8 12.5 -21t-6.5 -24l-182 -243l182 -243q8 -11 6.5 -24t-12.5 -21l-51 -38q-11 -8 -24 -6.5t-21 12.5l-226 301q-14 19 0 38zM1282 467q14 -19 0 -38l-226 -301q-8 -11 -21 -12.5t-24 6.5l-51 38q-11 8 -12.5 21t6.5 24l182 243 l-182 243q-8 11 -6.5 24t12.5 21l51 38q11 8 24 6.5t21 -12.5zM662 6q-13 2 -20.5 13t-5.5 24l138 831q2 13 13 20.5t24 5.5l63 -10q13 -2 20.5 -13t5.5 -24l-138 -831q-2 -13 -13 -20.5t-24 -5.5z" /> +<glyph unicode="" d="M1497 709v-198q-101 -23 -198 -23q-65 -136 -165.5 -271t-181.5 -215.5t-128 -106.5q-80 -45 -162 3q-28 17 -60.5 43.5t-85 83.5t-102.5 128.5t-107.5 184t-105.5 244t-91.5 314.5t-70.5 390h283q26 -218 70 -398.5t104.5 -317t121.5 -235.5t140 -195q169 169 287 406 q-142 72 -223 220t-81 333q0 192 104 314.5t284 122.5q178 0 273 -105.5t95 -297.5q0 -159 -58 -286q-7 -1 -19.5 -3t-46 -2t-63 6t-62 25.5t-50.5 51.5q31 103 31 184q0 87 -29 132t-79 45q-53 0 -85 -49.5t-32 -140.5q0 -186 105 -293.5t267 -107.5q62 0 121 14z" /> +<glyph unicode="" horiz-adv-x="1792" d="M216 367l603 -402v359l-334 223zM154 511l193 129l-193 129v-258zM973 -35l603 402l-269 180l-334 -223v-359zM896 458l272 182l-272 182l-272 -182zM485 733l334 223v359l-603 -402zM1445 640l193 -129v258zM1307 733l269 180l-603 402v-359zM1792 913v-546 q0 -41 -34 -64l-819 -546q-21 -13 -43 -13t-43 13l-819 546q-34 23 -34 64v546q0 41 34 64l819 546q21 13 43 13t43 -13l819 -546q34 -23 34 -64z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1800 764q111 -46 179.5 -145.5t68.5 -221.5q0 -164 -118 -280.5t-285 -116.5q-4 0 -11.5 0.5t-10.5 0.5h-1209h-1h-2h-5q-170 10 -288 125.5t-118 280.5q0 110 55 203t147 147q-12 39 -12 82q0 115 82 196t199 81q95 0 172 -58q75 154 222.5 248t326.5 94 q166 0 306 -80.5t221.5 -218.5t81.5 -301q0 -6 -0.5 -18t-0.5 -18zM468 498q0 -122 84 -193t208 -71q137 0 240 99q-16 20 -47.5 56.5t-43.5 50.5q-67 -65 -144 -65q-55 0 -93.5 33.5t-38.5 87.5q0 53 38.5 87t91.5 34q44 0 84.5 -21t73 -55t65 -75t69 -82t77 -75t97 -55 t121.5 -21q121 0 204.5 71.5t83.5 190.5q0 121 -84 192t-207 71q-143 0 -241 -97q14 -16 29.5 -34t34.5 -40t29 -34q66 64 142 64q52 0 92 -33t40 -84q0 -57 -37 -91.5t-94 -34.5q-43 0 -82.5 21t-72 55t-65.5 75t-69.5 82t-77.5 75t-96.5 55t-118.5 21q-122 0 -207 -70.5 t-85 -189.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M896 1536q182 0 348 -71t286 -191t191 -286t71 -348t-71 -348t-191 -286t-286 -191t-348 -71t-348 71t-286 191t-191 286t-71 348t71 348t191 286t286 191t348 71zM896 1408q-190 0 -361 -90l194 -194q82 28 167 28t167 -28l194 194q-171 90 -361 90zM218 279l194 194 q-28 82 -28 167t28 167l-194 194q-90 -171 -90 -361t90 -361zM896 -128q190 0 361 90l-194 194q-82 -28 -167 -28t-167 28l-194 -194q171 -90 361 -90zM896 256q159 0 271.5 112.5t112.5 271.5t-112.5 271.5t-271.5 112.5t-271.5 -112.5t-112.5 -271.5t112.5 -271.5 t271.5 -112.5zM1380 473l194 -194q90 171 90 361t-90 361l-194 -194q28 -82 28 -167t-28 -167z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 640q0 -182 -71 -348t-191 -286t-286 -191t-348 -71t-348 71t-286 191t-191 286t-71 348q0 222 101 414.5t276.5 317t390.5 155.5v-260q-221 -45 -366.5 -221t-145.5 -406q0 -130 51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5 q0 230 -145.5 406t-366.5 221v260q215 -31 390.5 -155.5t276.5 -317t101 -414.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M19 662q8 217 116 406t305 318h5q0 -1 -1 -3q-8 -8 -28 -33.5t-52 -76.5t-60 -110.5t-44.5 -135.5t-14 -150.5t39 -157.5t108.5 -154q50 -50 102 -69.5t90.5 -11.5t69.5 23.5t47 32.5l16 16q39 51 53 116.5t6.5 122.5t-21 107t-26.5 80l-14 29q-10 25 -30.5 49.5t-43 41 t-43.5 29.5t-35 19l-13 6l104 115q39 -17 78 -52t59 -61l19 -27q1 48 -18.5 103.5t-40.5 87.5l-20 31l161 183l160 -181q-33 -46 -52.5 -102.5t-22.5 -90.5l-4 -33q22 37 61.5 72.5t67.5 52.5l28 17l103 -115q-44 -14 -85 -50t-60 -65l-19 -29q-31 -56 -48 -133.5t-7 -170 t57 -156.5q33 -45 77.5 -60.5t85 -5.5t76 26.5t57.5 33.5l21 16q60 53 96.5 115t48.5 121.5t10 121.5t-18 118t-37 107.5t-45.5 93t-45 72t-34.5 47.5l-13 17q-14 13 -7 13l10 -3q40 -29 62.5 -46t62 -50t64 -58t58.5 -65t55.5 -77t45.5 -88t38 -103t23.5 -117t10.5 -136 q3 -259 -108 -465t-312 -321t-456 -115q-185 0 -351 74t-283.5 198t-184 293t-60.5 353z" /> +<glyph unicode="" horiz-adv-x="1792" d="M874 -102v-66q-208 6 -385 109.5t-283 275.5l58 34q29 -49 73 -99l65 57q148 -168 368 -212l-17 -86q65 -12 121 -13zM276 428l-83 -28q22 -60 49 -112l-57 -33q-98 180 -98 385t98 385l57 -33q-30 -56 -49 -112l82 -28q-35 -100 -35 -212q0 -109 36 -212zM1528 251 l58 -34q-106 -172 -283 -275.5t-385 -109.5v66q56 1 121 13l-17 86q220 44 368 212l65 -57q44 50 73 99zM1377 805l-233 -80q14 -42 14 -85t-14 -85l232 -80q-31 -92 -98 -169l-185 162q-57 -67 -147 -85l48 -241q-52 -10 -98 -10t-98 10l48 241q-90 18 -147 85l-185 -162 q-67 77 -98 169l232 80q-14 42 -14 85t14 85l-233 80q33 93 99 169l185 -162q59 68 147 86l-48 240q44 10 98 10t98 -10l-48 -240q88 -18 147 -86l185 162q66 -76 99 -169zM874 1448v-66q-65 -2 -121 -13l17 -86q-220 -42 -368 -211l-65 56q-38 -42 -73 -98l-57 33 q106 172 282 275.5t385 109.5zM1705 640q0 -205 -98 -385l-57 33q27 52 49 112l-83 28q36 103 36 212q0 112 -35 212l82 28q-19 56 -49 112l57 33q98 -180 98 -385zM1585 1063l-57 -33q-35 56 -73 98l-65 -56q-148 169 -368 211l17 86q-56 11 -121 13v66q209 -6 385 -109.5 t282 -275.5zM1748 640q0 173 -67.5 331t-181.5 272t-272 181.5t-331 67.5t-331 -67.5t-272 -181.5t-181.5 -272t-67.5 -331t67.5 -331t181.5 -272t272 -181.5t331 -67.5t331 67.5t272 181.5t181.5 272t67.5 331zM1792 640q0 -182 -71 -348t-191 -286t-286 -191t-348 -71 t-348 71t-286 191t-191 286t-71 348t71 348t191 286t286 191t348 71t348 -71t286 -191t191 -286t71 -348z" /> +<glyph unicode="" d="M582 228q0 -66 -93 -66q-107 0 -107 63q0 64 98 64q102 0 102 -61zM546 694q0 -85 -74 -85q-77 0 -77 84q0 90 77 90q36 0 55 -25.5t19 -63.5zM712 769v125q-78 -29 -135 -29q-50 29 -110 29q-86 0 -145 -57t-59 -143q0 -50 29.5 -102t73.5 -67v-3q-38 -17 -38 -85 q0 -53 41 -77v-3q-113 -37 -113 -139q0 -45 20 -78.5t54 -51t72 -25.5t81 -8q224 0 224 188q0 67 -48 99t-126 46q-27 5 -51.5 20.5t-24.5 39.5q0 44 49 52q77 15 122 70t45 134q0 24 -10 52q37 9 49 13zM771 350h137q-2 27 -2 82v387q0 46 2 69h-137q3 -23 3 -71v-392 q0 -50 -3 -75zM1280 366v121q-30 -21 -68 -21q-53 0 -53 82v225h52q9 0 26.5 -1t26.5 -1v117h-105q0 82 3 102h-140q4 -24 4 -55v-47h-60v-117q36 3 37 3q3 0 11 -0.5t12 -0.5v-2h-2v-217q0 -37 2.5 -64t11.5 -56.5t24.5 -48.5t43.5 -31t66 -12q64 0 108 24zM924 1072 q0 36 -24 63.5t-60 27.5t-60.5 -27t-24.5 -64q0 -36 25 -62.5t60 -26.5t59.5 27t24.5 62zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M595 22q0 100 -165 100q-158 0 -158 -104q0 -101 172 -101q151 0 151 105zM536 777q0 61 -30 102t-89 41q-124 0 -124 -145q0 -135 124 -135q119 0 119 137zM805 1101v-202q-36 -12 -79 -22q16 -43 16 -84q0 -127 -73 -216.5t-197 -112.5q-40 -8 -59.5 -27t-19.5 -58 q0 -31 22.5 -51.5t58 -32t78.5 -22t86 -25.5t78.5 -37.5t58 -64t22.5 -98.5q0 -304 -363 -304q-69 0 -130 12.5t-116 41t-87.5 82t-32.5 127.5q0 165 182 225v4q-67 41 -67 126q0 109 63 137v4q-72 24 -119.5 108.5t-47.5 165.5q0 139 95 231.5t235 92.5q96 0 178 -47 q98 0 218 47zM1123 220h-222q4 45 4 134v609q0 94 -4 128h222q-4 -33 -4 -124v-613q0 -89 4 -134zM1724 442v-196q-71 -39 -174 -39q-62 0 -107 20t-70 50t-39.5 78t-18.5 92t-4 103v351h2v4q-7 0 -19 1t-18 1q-21 0 -59 -6v190h96v76q0 54 -6 89h227q-6 -41 -6 -165h171 v-190q-15 0 -43.5 2t-42.5 2h-85v-365q0 -131 87 -131q61 0 109 33zM1148 1389q0 -58 -39 -101.5t-96 -43.5q-58 0 -98 43.5t-40 101.5q0 59 39.5 103t98.5 44q58 0 96.5 -44.5t38.5 -102.5z" /> +<glyph unicode="" d="M825 547l343 588h-150q-21 -39 -63.5 -118.5t-68 -128.5t-59.5 -118.5t-60 -128.5h-3q-21 48 -44.5 97t-52 105.5t-46.5 92t-54 104.5t-49 95h-150l323 -589v-435h134v436zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960 q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1280" d="M842 964q0 -80 -57 -136.5t-136 -56.5q-60 0 -111 35q-62 -67 -115 -146q-247 -371 -202 -859q1 -22 -12.5 -38.5t-34.5 -18.5h-5q-20 0 -35 13.5t-17 33.5q-14 126 -3.5 247.5t29.5 217t54 186t69 155.5t74 125q61 90 132 165q-16 35 -16 77q0 80 56.5 136.5t136.5 56.5 t136.5 -56.5t56.5 -136.5zM1223 953q0 -158 -78 -292t-212.5 -212t-292.5 -78q-64 0 -131 14q-21 5 -32.5 23.5t-6.5 39.5q5 20 23 31.5t39 7.5q51 -13 108 -13q97 0 186 38t153 102t102 153t38 186t-38 186t-102 153t-153 102t-186 38t-186 -38t-153 -102t-102 -153 t-38 -186q0 -114 52 -218q10 -20 3.5 -40t-25.5 -30t-39.5 -3t-30.5 26q-64 123 -64 265q0 119 46.5 227t124.5 186t186 124t226 46q158 0 292.5 -78t212.5 -212.5t78 -292.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M270 730q-8 19 -8 52q0 20 11 49t24 45q-1 22 7.5 53t22.5 43q0 139 92.5 288.5t217.5 209.5q139 66 324 66q133 0 266 -55q49 -21 90 -48t71 -56t55 -68t42 -74t32.5 -84.5t25.5 -89.5t22 -98l1 -5q55 -83 55 -150q0 -14 -9 -40t-9 -38q0 -1 1.5 -3.5t3.5 -5t2 -3.5 q77 -114 120.5 -214.5t43.5 -208.5q0 -43 -19.5 -100t-55.5 -57q-9 0 -19.5 7.5t-19 17.5t-19 26t-16 26.5t-13.5 26t-9 17.5q-1 1 -3 1l-5 -4q-59 -154 -132 -223q20 -20 61.5 -38.5t69 -41.5t35.5 -65q-2 -4 -4 -16t-7 -18q-64 -97 -302 -97q-53 0 -110.5 9t-98 20 t-104.5 30q-15 5 -23 7q-14 4 -46 4.5t-40 1.5q-41 -45 -127.5 -65t-168.5 -20q-35 0 -69 1.5t-93 9t-101 20.5t-74.5 40t-32.5 64q0 40 10 59.5t41 48.5q11 2 40.5 13t49.5 12q4 0 14 2q2 2 2 4l-2 3q-48 11 -108 105.5t-73 156.5l-5 3q-4 0 -12 -20q-18 -41 -54.5 -74.5 t-77.5 -37.5h-1q-4 0 -6 4.5t-5 5.5q-23 54 -23 100q0 275 252 466z" /> +<glyph unicode="" horiz-adv-x="2048" d="M580 1075q0 41 -25 66t-66 25q-43 0 -76 -25.5t-33 -65.5q0 -39 33 -64.5t76 -25.5q41 0 66 24.5t25 65.5zM1323 568q0 28 -25.5 50t-65.5 22q-27 0 -49.5 -22.5t-22.5 -49.5q0 -28 22.5 -50.5t49.5 -22.5q40 0 65.5 22t25.5 51zM1087 1075q0 41 -24.5 66t-65.5 25 q-43 0 -76 -25.5t-33 -65.5q0 -39 33 -64.5t76 -25.5q41 0 65.5 24.5t24.5 65.5zM1722 568q0 28 -26 50t-65 22q-27 0 -49.5 -22.5t-22.5 -49.5q0 -28 22.5 -50.5t49.5 -22.5q39 0 65 22t26 51zM1456 965q-31 4 -70 4q-169 0 -311 -77t-223.5 -208.5t-81.5 -287.5 q0 -78 23 -152q-35 -3 -68 -3q-26 0 -50 1.5t-55 6.5t-44.5 7t-54.5 10.5t-50 10.5l-253 -127l72 218q-290 203 -290 490q0 169 97.5 311t264 223.5t363.5 81.5q176 0 332.5 -66t262 -182.5t136.5 -260.5zM2048 404q0 -117 -68.5 -223.5t-185.5 -193.5l55 -181l-199 109 q-150 -37 -218 -37q-169 0 -311 70.5t-223.5 191.5t-81.5 264t81.5 264t223.5 191.5t311 70.5q161 0 303 -70.5t227.5 -192t85.5 -263.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1764 1525q33 -24 27 -64l-256 -1536q-5 -29 -32 -45q-14 -8 -31 -8q-11 0 -24 5l-453 185l-242 -295q-18 -23 -49 -23q-13 0 -22 4q-19 7 -30.5 23.5t-11.5 36.5v349l864 1059l-1069 -925l-395 162q-37 14 -40 55q-2 40 32 59l1664 960q15 9 32 9q20 0 36 -11z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1764 1525q33 -24 27 -64l-256 -1536q-5 -29 -32 -45q-14 -8 -31 -8q-11 0 -24 5l-527 215l-298 -327q-18 -21 -47 -21q-14 0 -23 4q-19 7 -30 23.5t-11 36.5v452l-472 193q-37 14 -40 55q-3 39 32 59l1664 960q35 21 68 -2zM1422 26l221 1323l-1434 -827l336 -137 l863 639l-478 -797z" /> +<glyph unicode="" d="M1536 640q0 -156 -61 -298t-164 -245t-245 -164t-298 -61q-172 0 -327 72.5t-264 204.5q-7 10 -6.5 22.5t8.5 20.5l137 138q10 9 25 9q16 -2 23 -12q73 -95 179 -147t225 -52q104 0 198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5t-40.5 198.5t-109.5 163.5 t-163.5 109.5t-198.5 40.5q-98 0 -188 -35.5t-160 -101.5l137 -138q31 -30 14 -69q-17 -40 -59 -40h-448q-26 0 -45 19t-19 45v448q0 42 40 59q39 17 69 -14l130 -129q107 101 244.5 156.5t284.5 55.5q156 0 298 -61t245 -164t164 -245t61 -298zM896 928v-448q0 -14 -9 -23 t-23 -9h-320q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h224v352q0 14 9 23t23 9h64q14 0 23 -9t9 -23z" /> +<glyph unicode="" d="M768 1280q-130 0 -248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5t-51 248.5t-136.5 204t-204 136.5t-248.5 51zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103 t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1682 -128q-44 0 -132.5 3.5t-133.5 3.5q-44 0 -132 -3.5t-132 -3.5q-24 0 -37 20.5t-13 45.5q0 31 17 46t39 17t51 7t45 15q33 21 33 140l-1 391q0 21 -1 31q-13 4 -50 4h-675q-38 0 -51 -4q-1 -10 -1 -31l-1 -371q0 -142 37 -164q16 -10 48 -13t57 -3.5t45 -15 t20 -45.5q0 -26 -12.5 -48t-36.5 -22q-47 0 -139.5 3.5t-138.5 3.5q-43 0 -128 -3.5t-127 -3.5q-23 0 -35.5 21t-12.5 45q0 30 15.5 45t36 17.5t47.5 7.5t42 15q33 23 33 143l-1 57v813q0 3 0.5 26t0 36.5t-1.5 38.5t-3.5 42t-6.5 36.5t-11 31.5t-16 18q-15 10 -45 12t-53 2 t-41 14t-18 45q0 26 12 48t36 22q46 0 138.5 -3.5t138.5 -3.5q42 0 126.5 3.5t126.5 3.5q25 0 37.5 -22t12.5 -48q0 -30 -17 -43.5t-38.5 -14.5t-49.5 -4t-43 -13q-35 -21 -35 -160l1 -320q0 -21 1 -32q13 -3 39 -3h699q25 0 38 3q1 11 1 32l1 320q0 139 -35 160 q-18 11 -58.5 12.5t-66 13t-25.5 49.5q0 26 12.5 48t37.5 22q44 0 132 -3.5t132 -3.5q43 0 129 3.5t129 3.5q25 0 37.5 -22t12.5 -48q0 -30 -17.5 -44t-40 -14.5t-51.5 -3t-44 -12.5q-35 -23 -35 -161l1 -943q0 -119 34 -140q16 -10 46 -13.5t53.5 -4.5t41.5 -15.5t18 -44.5 q0 -26 -12 -48t-36 -22z" /> +<glyph unicode="" horiz-adv-x="1280" d="M1278 1347v-73q0 -29 -18.5 -61t-42.5 -32q-50 0 -54 -1q-26 -6 -32 -31q-3 -11 -3 -64v-1152q0 -25 -18 -43t-43 -18h-108q-25 0 -43 18t-18 43v1218h-143v-1218q0 -25 -17.5 -43t-43.5 -18h-108q-26 0 -43.5 18t-17.5 43v496q-147 12 -245 59q-126 58 -192 179 q-64 117 -64 259q0 166 88 286q88 118 209 159q111 37 417 37h479q25 0 43 -18t18 -43z" /> +<glyph unicode="" d="M352 128v-128h-352v128h352zM704 256q26 0 45 -19t19 -45v-256q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h256zM864 640v-128h-864v128h864zM224 1152v-128h-224v128h224zM1536 128v-128h-736v128h736zM576 1280q26 0 45 -19t19 -45v-256 q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h256zM1216 768q26 0 45 -19t19 -45v-256q0 -26 -19 -45t-45 -19h-256q-26 0 -45 19t-19 45v256q0 26 19 45t45 19h256zM1536 640v-128h-224v128h224zM1536 1152v-128h-864v128h864z" /> +<glyph unicode="" d="M1216 512q133 0 226.5 -93.5t93.5 -226.5t-93.5 -226.5t-226.5 -93.5t-226.5 93.5t-93.5 226.5q0 12 2 34l-360 180q-92 -86 -218 -86q-133 0 -226.5 93.5t-93.5 226.5t93.5 226.5t226.5 93.5q126 0 218 -86l360 180q-2 22 -2 34q0 133 93.5 226.5t226.5 93.5 t226.5 -93.5t93.5 -226.5t-93.5 -226.5t-226.5 -93.5q-126 0 -218 86l-360 -180q2 -22 2 -34t-2 -34l360 -180q92 86 218 86z" /> +<glyph unicode="" d="M1280 341q0 88 -62.5 151t-150.5 63q-84 0 -145 -58l-241 120q2 16 2 23t-2 23l241 120q61 -58 145 -58q88 0 150.5 63t62.5 151t-62.5 150.5t-150.5 62.5t-151 -62.5t-63 -150.5q0 -7 2 -23l-241 -120q-62 57 -145 57q-88 0 -150.5 -62.5t-62.5 -150.5t62.5 -150.5 t150.5 -62.5q83 0 145 57l241 -120q-2 -16 -2 -23q0 -88 63 -150.5t151 -62.5t150.5 62.5t62.5 150.5zM1536 1120v-960q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M571 947q-10 25 -34 35t-49 0q-108 -44 -191 -127t-127 -191q-10 -25 0 -49t35 -34q13 -5 24 -5q42 0 60 40q34 84 98.5 148.5t148.5 98.5q25 11 35 35t0 49zM1513 1303l46 -46l-244 -243l68 -68q19 -19 19 -45.5t-19 -45.5l-64 -64q89 -161 89 -343q0 -143 -55.5 -273.5 t-150 -225t-225 -150t-273.5 -55.5t-273.5 55.5t-225 150t-150 225t-55.5 273.5t55.5 273.5t150 225t225 150t273.5 55.5q182 0 343 -89l64 64q19 19 45.5 19t45.5 -19l68 -68zM1521 1359q-10 -10 -22 -10q-13 0 -23 10l-91 90q-9 10 -9 23t9 23q10 9 23 9t23 -9l90 -91 q10 -9 10 -22.5t-10 -22.5zM1751 1129q-11 -9 -23 -9t-23 9l-90 91q-10 9 -10 22.5t10 22.5q9 10 22.5 10t22.5 -10l91 -90q9 -10 9 -23t-9 -23zM1792 1312q0 -14 -9 -23t-23 -9h-96q-14 0 -23 9t-9 23t9 23t23 9h96q14 0 23 -9t9 -23zM1600 1504v-96q0 -14 -9 -23t-23 -9 t-23 9t-9 23v96q0 14 9 23t23 9t23 -9t9 -23zM1751 1449l-91 -90q-10 -10 -22 -10q-13 0 -23 10q-10 9 -10 22.5t10 22.5l90 91q10 9 23 9t23 -9q9 -10 9 -23t-9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M609 720l287 208l287 -208l-109 -336h-355zM896 1536q182 0 348 -71t286 -191t191 -286t71 -348t-71 -348t-191 -286t-286 -191t-348 -71t-348 71t-286 191t-191 286t-71 348t71 348t191 286t286 191t348 71zM1515 186q149 203 149 454v3l-102 -89l-240 224l63 323 l134 -12q-150 206 -389 282l53 -124l-287 -159l-287 159l53 124q-239 -76 -389 -282l135 12l62 -323l-240 -224l-102 89v-3q0 -251 149 -454l30 132l326 -40l139 -298l-116 -69q117 -39 240 -39t240 39l-116 69l139 298l326 40z" /> +<glyph unicode="" horiz-adv-x="1792" d="M448 224v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM256 608v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM832 224v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23 v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM640 608v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM66 768q-28 0 -47 19t-19 46v129h514v-129q0 -27 -19 -46t-46 -19h-383zM1216 224v-192q0 -14 -9 -23t-23 -9h-192 q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1024 608v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1600 224v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23 zM1408 608v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1792 1016v-13h-514v10q0 104 -382 102q-382 -1 -382 -102v-10h-514v13q0 17 8.5 43t34 64t65.5 75.5t110.5 76t160 67.5t224 47.5t293.5 18.5t293 -18.5t224 -47.5 t160.5 -67.5t110.5 -76t65.5 -75.5t34 -64t8.5 -43zM1792 608v-192q0 -14 -9 -23t-23 -9h-192q-14 0 -23 9t-9 23v192q0 14 9 23t23 9h192q14 0 23 -9t9 -23zM1792 962v-129q0 -27 -19 -46t-46 -19h-384q-27 0 -46 19t-19 46v129h514z" /> +<glyph unicode="" horiz-adv-x="1792" d="M704 1216v-768q0 -26 -19 -45t-45 -19v-576q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v512l249 873q7 23 31 23h424zM1024 1216v-704h-256v704h256zM1792 320v-512q0 -26 -19 -45t-45 -19h-512q-26 0 -45 19t-19 45v576q-26 0 -45 19t-19 45v768h424q24 0 31 -23z M736 1504v-224h-352v224q0 14 9 23t23 9h288q14 0 23 -9t9 -23zM1408 1504v-224h-352v224q0 14 9 23t23 9h288q14 0 23 -9t9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1755 1083q37 -37 37 -90t-37 -91l-401 -400l150 -150l-160 -160q-163 -163 -389.5 -186.5t-411.5 100.5l-362 -362h-181v181l362 362q-124 185 -100.5 411.5t186.5 389.5l160 160l150 -150l400 401q38 37 91 37t90 -37t37 -90.5t-37 -90.5l-400 -401l234 -234l401 400 q38 37 91 37t90 -37z" /> +<glyph unicode="" horiz-adv-x="1792" d="M873 796q0 -83 -63.5 -142.5t-152.5 -59.5t-152.5 59.5t-63.5 142.5q0 84 63.5 143t152.5 59t152.5 -59t63.5 -143zM1375 796q0 -83 -63 -142.5t-153 -59.5q-89 0 -152.5 59.5t-63.5 142.5q0 84 63.5 143t152.5 59q90 0 153 -59t63 -143zM1600 616v667q0 87 -32 123.5 t-111 36.5h-1112q-83 0 -112.5 -34t-29.5 -126v-673q43 -23 88.5 -40t81 -28t81 -18.5t71 -11t70 -4t58.5 -0.5t56.5 2t44.5 2q68 1 95 -27q6 -6 10 -9q26 -25 61 -51q7 91 118 87q5 0 36.5 -1.5t43 -2t45.5 -1t53 1t54.5 4.5t61 8.5t62 13.5t67 19.5t67.5 27t72 34.5z M1763 621q-121 -149 -372 -252q84 -285 -23 -465q-66 -113 -183 -148q-104 -32 -182 15q-86 51 -82 164l-1 326v1q-8 2 -24.5 6t-23.5 5l-1 -338q4 -114 -83 -164q-79 -47 -183 -15q-117 36 -182 150q-105 180 -22 463q-251 103 -372 252q-25 37 -4 63t60 -1q3 -2 11 -7 t11 -8v694q0 72 47 123t114 51h1257q67 0 114 -51t47 -123v-694l21 15q39 27 60 1t-4 -63z" /> +<glyph unicode="" horiz-adv-x="1792" d="M896 1102v-434h-145v434h145zM1294 1102v-434h-145v434h145zM1294 342l253 254v795h-1194v-1049h326v-217l217 217h398zM1692 1536v-1013l-434 -434h-326l-217 -217h-217v217h-398v1158l109 289h1483z" /> +<glyph unicode="" d="M773 217v-127q-1 -292 -6 -305q-12 -32 -51 -40q-54 -9 -181.5 38t-162.5 89q-13 15 -17 36q-1 12 4 26q4 10 34 47t181 216q1 0 60 70q15 19 39.5 24.5t49.5 -3.5q24 -10 37.5 -29t12.5 -42zM624 468q-3 -55 -52 -70l-120 -39q-275 -88 -292 -88q-35 2 -54 36 q-12 25 -17 75q-8 76 1 166.5t30 124.5t56 32q13 0 202 -77q70 -29 115 -47l84 -34q23 -9 35.5 -30.5t11.5 -48.5zM1450 171q-7 -54 -91.5 -161t-135.5 -127q-37 -14 -63 7q-14 10 -184 287l-47 77q-14 21 -11.5 46t19.5 46q35 43 83 26q1 -1 119 -40q203 -66 242 -79.5 t47 -20.5q28 -22 22 -61zM778 803q5 -102 -54 -122q-58 -17 -114 71l-378 598q-8 35 19 62q41 43 207.5 89.5t224.5 31.5q40 -10 49 -45q3 -18 22 -305.5t24 -379.5zM1440 695q3 -39 -26 -59q-15 -10 -329 -86q-67 -15 -91 -23l1 2q-23 -6 -46 4t-37 32q-30 47 0 87 q1 1 75 102q125 171 150 204t34 39q28 19 65 2q48 -23 123 -133.5t81 -167.5v-3z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1024 1024h-384v-384h384v384zM1152 384v-128h-640v128h640zM1152 1152v-640h-640v640h640zM1792 384v-128h-512v128h512zM1792 640v-128h-512v128h512zM1792 896v-128h-512v128h512zM1792 1152v-128h-512v128h512zM256 192v960h-128v-960q0 -26 19 -45t45 -19t45 19 t19 45zM1920 192v1088h-1536v-1088q0 -33 -11 -64h1483q26 0 45 19t19 45zM2048 1408v-1216q0 -80 -56 -136t-136 -56h-1664q-80 0 -136 56t-56 136v1088h256v128h1792z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1024 13q-20 0 -93 73.5t-73 93.5q0 32 62.5 54t103.5 22t103.5 -22t62.5 -54q0 -20 -73 -93.5t-93 -73.5zM1294 284q-2 0 -40 25t-101.5 50t-128.5 25t-128.5 -25t-101 -50t-40.5 -25q-18 0 -93.5 75t-75.5 93q0 13 10 23q78 77 196 121t233 44t233 -44t196 -121 q10 -10 10 -23q0 -18 -75.5 -93t-93.5 -75zM1567 556q-11 0 -23 8q-136 105 -252 154.5t-268 49.5q-85 0 -170.5 -22t-149 -53t-113.5 -62t-79 -53t-31 -22q-17 0 -92 75t-75 93q0 12 10 22q132 132 320 205t380 73t380 -73t320 -205q10 -10 10 -22q0 -18 -75 -93t-92 -75z M1838 827q-11 0 -22 9q-179 157 -371.5 236.5t-420.5 79.5t-420.5 -79.5t-371.5 -236.5q-11 -9 -22 -9q-17 0 -92.5 75t-75.5 93q0 13 10 23q187 186 445 288t527 102t527 -102t445 -288q10 -10 10 -23q0 -18 -75.5 -93t-92.5 -75z" /> +<glyph unicode="" horiz-adv-x="1792" d="M384 0q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM768 0q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM384 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5 t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1152 0q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM768 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5 t37.5 90.5zM384 768q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1152 384q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM768 768q0 53 -37.5 90.5t-90.5 37.5 t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1536 0v384q0 52 -38 90t-90 38t-90 -38t-38 -90v-384q0 -52 38 -90t90 -38t90 38t38 90zM1152 768q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5z M1536 1088v256q0 26 -19 45t-45 19h-1280q-26 0 -45 -19t-19 -45v-256q0 -26 19 -45t45 -19h1280q26 0 45 19t19 45zM1536 768q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1664 1408v-1536q0 -52 -38 -90t-90 -38 h-1408q-52 0 -90 38t-38 90v1536q0 52 38 90t90 38h1408q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1112 1090q0 159 -237 159h-70q-32 0 -59.5 -21.5t-34.5 -52.5l-63 -276q-2 -5 -2 -16q0 -24 17 -39.5t41 -15.5h53q69 0 128.5 13t112.5 41t83.5 81.5t30.5 126.5zM1716 938q0 -265 -220 -428q-219 -161 -612 -161h-61q-32 0 -59 -21.5t-34 -52.5l-73 -316 q-8 -36 -40.5 -61.5t-69.5 -25.5h-213q-31 0 -53 20t-22 51q0 10 13 65h151q34 0 64 23.5t38 56.5l73 316q8 33 37.5 57t63.5 24h61q390 0 607 160t217 421q0 129 -51 207q183 -92 183 -335zM1533 1123q0 -264 -221 -428q-218 -161 -612 -161h-60q-32 0 -59.5 -22t-34.5 -53 l-73 -315q-8 -36 -40 -61.5t-69 -25.5h-214q-31 0 -52.5 19.5t-21.5 51.5q0 8 2 20l300 1301q8 36 40.5 61.5t69.5 25.5h444q68 0 125 -4t120.5 -15t113.5 -30t96.5 -50.5t77.5 -74t49.5 -103.5t18.5 -136z" /> +<glyph unicode="" horiz-adv-x="1792" d="M602 949q19 -61 31 -123.5t17 -141.5t-14 -159t-62 -145q-21 81 -67 157t-95.5 127t-99 90.5t-78.5 57.5t-33 19q-62 34 -81.5 100t14.5 128t101 81.5t129 -14.5q138 -83 238 -177zM927 1236q11 -25 20.5 -46t36.5 -100.5t42.5 -150.5t25.5 -179.5t0 -205.5t-47.5 -209.5 t-105.5 -208.5q-51 -72 -138 -72q-54 0 -98 31q-57 40 -69 109t28 127q60 85 81 195t13 199.5t-32 180.5t-39 128t-22 52q-31 63 -8.5 129.5t85.5 97.5q34 17 75 17q47 0 88.5 -25t63.5 -69zM1248 567q-17 -160 -72 -311q-17 131 -63 246q25 174 -5 361q-27 178 -94 342 q114 -90 212 -211q9 -37 15 -80q26 -179 7 -347zM1520 1440q9 -17 23.5 -49.5t43.5 -117.5t50.5 -178t34 -227.5t5 -269t-47 -300t-112.5 -323.5q-22 -48 -66 -75.5t-95 -27.5q-39 0 -74 16q-67 31 -92.5 100t4.5 136q58 126 90 257.5t37.5 239.5t-3.5 213.5t-26.5 180.5 t-38.5 138.5t-32.5 90t-15.5 32.5q-34 65 -11.5 135.5t87.5 104.5q37 20 81 20q49 0 91.5 -25.5t66.5 -70.5z" /> +<glyph unicode="" horiz-adv-x="2304" d="M1975 546h-138q14 37 66 179l3 9q4 10 10 26t9 26l12 -55zM531 611l-58 295q-11 54 -75 54h-268l-2 -13q311 -79 403 -336zM710 960l-162 -438l-17 89q-26 70 -85 129.5t-131 88.5l135 -510h175l261 641h-176zM849 318h166l104 642h-166zM1617 944q-69 27 -149 27 q-123 0 -201 -59t-79 -153q-1 -102 145 -174q48 -23 67 -41t19 -39q0 -30 -30 -46t-69 -16q-86 0 -156 33l-22 11l-23 -144q74 -34 185 -34q130 -1 208.5 59t80.5 160q0 106 -140 174q-49 25 -71 42t-22 38q0 22 24.5 38.5t70.5 16.5q70 1 124 -24l15 -8zM2042 960h-128 q-65 0 -87 -54l-246 -588h174l35 96h212q5 -22 20 -96h154zM2304 1280v-1280q0 -52 -38 -90t-90 -38h-2048q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h2048q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="2304" d="M671 603h-13q-47 0 -47 -32q0 -22 20 -22q17 0 28 15t12 39zM1066 639h62v3q1 4 0.5 6.5t-1 7t-2 8t-4.5 6.5t-7.5 5t-11.5 2q-28 0 -36 -38zM1606 603h-12q-48 0 -48 -32q0 -22 20 -22q17 0 28 15t12 39zM1925 629q0 41 -30 41q-19 0 -31 -20t-12 -51q0 -42 28 -42 q20 0 32.5 20t12.5 52zM480 770h87l-44 -262h-56l32 201l-71 -201h-39l-4 200l-34 -200h-53l44 262h81l2 -163zM733 663q0 -6 -4 -42q-16 -101 -17 -113h-47l1 22q-20 -26 -58 -26q-23 0 -37.5 16t-14.5 42q0 39 26 60.5t73 21.5q14 0 23 -1q0 3 0.5 5.5t1 4.5t0.5 3 q0 20 -36 20q-29 0 -59 -10q0 4 7 48q38 11 67 11q74 0 74 -62zM889 721l-8 -49q-22 3 -41 3q-27 0 -27 -17q0 -8 4.5 -12t21.5 -11q40 -19 40 -60q0 -72 -87 -71q-34 0 -58 6q0 2 7 49q29 -8 51 -8q32 0 32 19q0 7 -4.5 11.5t-21.5 12.5q-43 20 -43 59q0 72 84 72 q30 0 50 -4zM977 721h28l-7 -52h-29q-2 -17 -6.5 -40.5t-7 -38.5t-2.5 -18q0 -16 19 -16q8 0 16 2l-8 -47q-21 -7 -40 -7q-43 0 -45 47q0 12 8 56q3 20 25 146h55zM1180 648q0 -23 -7 -52h-111q-3 -22 10 -33t38 -11q30 0 58 14l-9 -54q-30 -8 -57 -8q-95 0 -95 95 q0 55 27.5 90.5t69.5 35.5q35 0 55.5 -21t20.5 -56zM1319 722q-13 -23 -22 -62q-22 2 -31 -24t-25 -128h-56l3 14q22 130 29 199h51l-3 -33q14 21 25.5 29.5t28.5 4.5zM1506 763l-9 -57q-28 14 -50 14q-31 0 -51 -27.5t-20 -70.5q0 -30 13.5 -47t38.5 -17q21 0 48 13 l-10 -59q-28 -8 -50 -8q-45 0 -71.5 30.5t-26.5 82.5q0 70 35.5 114.5t91.5 44.5q26 0 61 -13zM1668 663q0 -18 -4 -42q-13 -79 -17 -113h-46l1 22q-20 -26 -59 -26q-23 0 -37 16t-14 42q0 39 25.5 60.5t72.5 21.5q15 0 23 -1q2 7 2 13q0 20 -36 20q-29 0 -59 -10q0 4 8 48 q38 11 67 11q73 0 73 -62zM1809 722q-14 -24 -21 -62q-23 2 -31.5 -23t-25.5 -129h-56l3 14q19 104 29 199h52q0 -11 -4 -33q15 21 26.5 29.5t27.5 4.5zM1950 770h56l-43 -262h-53l3 19q-23 -23 -52 -23q-31 0 -49.5 24t-18.5 64q0 53 27.5 92t64.5 39q31 0 53 -29z M2061 640q0 148 -72.5 273t-198 198t-273.5 73q-181 0 -328 -110q127 -116 171 -284h-50q-44 150 -158 253q-114 -103 -158 -253h-50q44 168 171 284q-147 110 -328 110q-148 0 -273.5 -73t-198 -198t-72.5 -273t72.5 -273t198 -198t273.5 -73q181 0 328 110 q-120 111 -165 264h50q46 -138 152 -233q106 95 152 233h50q-45 -153 -165 -264q147 -110 328 -110q148 0 273.5 73t198 198t72.5 273zM2304 1280v-1280q0 -52 -38 -90t-90 -38h-2048q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h2048q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="2304" d="M313 759q0 -51 -36 -84q-29 -26 -89 -26h-17v220h17q61 0 89 -27q36 -31 36 -83zM2089 824q0 -52 -64 -52h-19v101h20q63 0 63 -49zM380 759q0 74 -50 120.5t-129 46.5h-95v-333h95q74 0 119 38q60 51 60 128zM410 593h65v333h-65v-333zM730 694q0 40 -20.5 62t-75.5 42 q-29 10 -39.5 19t-10.5 23q0 16 13.5 26.5t34.5 10.5q29 0 53 -27l34 44q-41 37 -98 37q-44 0 -74 -27.5t-30 -67.5q0 -35 18 -55.5t64 -36.5q37 -13 45 -19q19 -12 19 -34q0 -20 -14 -33.5t-36 -13.5q-48 0 -71 44l-42 -40q44 -64 115 -64q51 0 83 30.5t32 79.5zM1008 604 v77q-37 -37 -78 -37q-49 0 -80.5 32.5t-31.5 82.5q0 48 31.5 81.5t77.5 33.5q43 0 81 -38v77q-40 20 -80 20q-74 0 -125.5 -50.5t-51.5 -123.5t51 -123.5t125 -50.5q42 0 81 19zM2240 0v527q-65 -40 -144.5 -84t-237.5 -117t-329.5 -137.5t-417.5 -134.5t-504 -118h1569 q26 0 45 19t19 45zM1389 757q0 75 -53 128t-128 53t-128 -53t-53 -128t53 -128t128 -53t128 53t53 128zM1541 584l144 342h-71l-90 -224l-89 224h-71l142 -342h35zM1714 593h184v56h-119v90h115v56h-115v74h119v57h-184v-333zM2105 593h80l-105 140q76 16 76 94q0 47 -31 73 t-87 26h-97v-333h65v133h9zM2304 1274v-1268q0 -56 -38.5 -95t-93.5 -39h-2040q-55 0 -93.5 39t-38.5 95v1268q0 56 38.5 95t93.5 39h2040q55 0 93.5 -39t38.5 -95z" /> +<glyph unicode="" horiz-adv-x="2304" d="M119 854h89l-45 108zM740 328l74 79l-70 79h-163v-49h142v-55h-142v-54h159zM898 406l99 -110v217zM1186 453q0 33 -40 33h-84v-69h83q41 0 41 36zM1475 457q0 29 -42 29h-82v-61h81q43 0 43 32zM1197 923q0 29 -42 29h-82v-60h81q43 0 43 31zM1656 854h89l-44 108z M699 1009v-271h-66v212l-94 -212h-57l-94 212v-212h-132l-25 60h-135l-25 -60h-70l116 271h96l110 -257v257h106l85 -184l77 184h108zM1255 453q0 -20 -5.5 -35t-14 -25t-22.5 -16.5t-26 -10t-31.5 -4.5t-31.5 -1t-32.5 0.5t-29.5 0.5v-91h-126l-80 90l-83 -90h-256v271h260 l80 -89l82 89h207q109 0 109 -89zM964 794v-56h-217v271h217v-57h-152v-49h148v-55h-148v-54h152zM2304 235v-229q0 -55 -38.5 -94.5t-93.5 -39.5h-2040q-55 0 -93.5 39.5t-38.5 94.5v678h111l25 61h55l25 -61h218v46l19 -46h113l20 47v-47h541v99l10 1q10 0 10 -14v-86h279 v23q23 -12 55 -18t52.5 -6.5t63 0.5t51.5 1l25 61h56l25 -61h227v58l34 -58h182v378h-180v-44l-25 44h-185v-44l-23 44h-249q-69 0 -109 -22v22h-172v-22q-24 22 -73 22h-628l-43 -97l-43 97h-198v-44l-22 44h-169l-78 -179v391q0 55 38.5 94.5t93.5 39.5h2040 q55 0 93.5 -39.5t38.5 -94.5v-678h-120q-51 0 -81 -22v22h-177q-55 0 -78 -22v22h-316v-22q-31 22 -87 22h-209v-22q-23 22 -91 22h-234l-54 -58l-50 58h-349v-378h343l55 59l52 -59h211v89h21q59 0 90 13v-102h174v99h8q8 0 10 -2t2 -10v-87h529q57 0 88 24v-24h168 q60 0 95 17zM1546 469q0 -23 -12 -43t-34 -29q25 -9 34 -26t9 -46v-54h-65v45q0 33 -12 43.5t-46 10.5h-69v-99h-65v271h154q48 0 77 -15t29 -58zM1269 936q0 -24 -12.5 -44t-33.5 -29q26 -9 34.5 -25.5t8.5 -46.5v-53h-65q0 9 0.5 26.5t0 25t-3 18.5t-8.5 16t-17.5 8.5 t-29.5 3.5h-70v-98h-64v271l153 -1q49 0 78 -14.5t29 -57.5zM1798 327v-56h-216v271h216v-56h-151v-49h148v-55h-148v-54zM1372 1009v-271h-66v271h66zM2065 357q0 -86 -102 -86h-126v58h126q34 0 34 25q0 16 -17 21t-41.5 5t-49.5 3.5t-42 22.5t-17 55q0 39 26 60t66 21 h130v-57h-119q-36 0 -36 -25q0 -16 17.5 -20.5t42 -4t49 -2.5t42 -21.5t17.5 -54.5zM2304 407v-101q-24 -35 -88 -35h-125v58h125q33 0 33 25q0 13 -12.5 19t-31 5.5t-40 2t-40 8t-31 24t-12.5 48.5q0 39 26.5 60t66.5 21h129v-57h-118q-36 0 -36 -25q0 -20 29 -22t68.5 -5 t56.5 -26zM2139 1008v-270h-92l-122 203v-203h-132l-26 60h-134l-25 -60h-75q-129 0 -129 133q0 138 133 138h63v-59q-7 0 -28 1t-28.5 0.5t-23 -2t-21.5 -6.5t-14.5 -13.5t-11.5 -23t-3 -33.5q0 -38 13.5 -58t49.5 -20h29l92 213h97l109 -256v256h99l114 -188v188h66z" /> +<glyph unicode="" horiz-adv-x="2304" d="M322 689h-15q-19 0 -19 18q0 28 19 85q5 15 15 19.5t28 4.5q77 0 77 -49q0 -41 -30.5 -59.5t-74.5 -18.5zM664 528q-47 0 -47 29q0 62 123 62l3 -3q-5 -88 -79 -88zM1438 687h-15q-19 0 -19 19q0 28 19 85q5 15 14.5 19t28.5 4q77 0 77 -49q0 -41 -30.5 -59.5 t-74.5 -18.5zM1780 527q-47 0 -47 30q0 62 123 62l3 -3q-5 -89 -79 -89zM373 894h-128q-8 0 -14.5 -4t-8.5 -7.5t-7 -12.5q-3 -7 -45 -190t-42 -192q0 -7 5.5 -12.5t13.5 -5.5h62q25 0 32.5 34.5l15 69t32.5 34.5q47 0 87.5 7.5t80.5 24.5t63.5 52.5t23.5 84.5 q0 36 -14.5 61t-41 36.5t-53.5 15.5t-62 4zM719 798q-38 0 -74 -6q-2 0 -8.5 -1t-9 -1.5l-7.5 -1.5t-7.5 -2t-6.5 -3t-6.5 -4t-5 -5t-4.5 -7t-4 -9q-9 -29 -9 -39t9 -10q5 0 21.5 5t19.5 6q30 8 58 8q74 0 74 -36q0 -11 -10 -14q-8 -2 -18 -3t-21.5 -1.5t-17.5 -1.5 q-38 -4 -64.5 -10t-56.5 -19.5t-45.5 -39t-15.5 -62.5q0 -38 26 -59.5t64 -21.5q24 0 45.5 6.5t33 13t38.5 23.5q-3 -7 -3 -15t5.5 -13.5t12.5 -5.5h56q1 1 7 3.5t7.5 3.5t5 3.5t5 5.5t2.5 8l45 194q4 13 4 30q0 81 -145 81zM1247 793h-74q-22 0 -39 -23q-5 -7 -29.5 -51 t-46.5 -81.5t-26 -38.5l-5 4q0 77 -27 166q-1 5 -3.5 8.5t-6 6.5t-6.5 5t-8.5 3t-8.5 1.5t-9.5 1t-9 0.5h-10h-8.5q-38 0 -38 -21l1 -5q5 -53 25 -151t25 -143q2 -16 2 -24q0 -19 -30.5 -61.5t-30.5 -58.5q0 -13 40 -13q61 0 76 25l245 415q10 20 10 26q0 9 -8 9zM1489 892 h-129q-18 0 -29 -23q-6 -13 -46.5 -191.5t-40.5 -190.5q0 -20 43 -20h7.5h9h9t9.5 1t8.5 2t8.5 3t6.5 4.5t5.5 6t3 8.5l21 91q2 10 10.5 17t19.5 7q47 0 87.5 7t80.5 24.5t63.5 52.5t23.5 84q0 36 -14.5 61t-41 36.5t-53.5 15.5t-62 4zM1835 798q-26 0 -74 -6 q-38 -6 -48 -16q-7 -8 -11 -19q-8 -24 -8 -39q0 -10 8 -10q1 0 41 12q30 8 58 8q74 0 74 -36q0 -12 -10 -14q-4 -1 -57 -7q-38 -4 -64.5 -10t-56.5 -19.5t-45.5 -39t-15.5 -62.5t26 -58.5t64 -21.5q24 0 45 6t34 13t38 24q-3 -15 -3 -16q0 -5 2 -8.5t6.5 -5.5t8 -3.5 t10.5 -2t9.5 -0.5h9.5h8q42 0 48 25l45 194q3 15 3 31q0 81 -145 81zM2157 889h-55q-25 0 -33 -40q-10 -44 -36.5 -167t-42.5 -190v-5q0 -16 16 -18h1h57q10 0 18.5 6.5t10.5 16.5l83 374h-1l1 5q0 7 -5.5 12.5t-13.5 5.5zM2304 1280v-1280q0 -52 -38 -90t-90 -38h-2048 q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h2048q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="2304" d="M1597 633q0 -69 -21 -106q-19 -35 -52 -35q-23 0 -41 9v224q29 30 57 30q57 0 57 -122zM2035 669h-110q6 98 56 98q51 0 54 -98zM476 534q0 59 -33 91.5t-101 57.5q-36 13 -52 24t-16 25q0 26 38 26q58 0 124 -33l18 112q-67 32 -149 32q-77 0 -123 -38q-48 -39 -48 -109 q0 -58 32.5 -90.5t99.5 -56.5q39 -14 54.5 -25.5t15.5 -27.5q0 -31 -48 -31q-29 0 -70 12.5t-72 30.5l-18 -113q72 -41 168 -41q81 0 129 37q51 41 51 117zM771 749l19 111h-96v135l-129 -21l-18 -114l-46 -8l-17 -103h62v-219q0 -84 44 -120q38 -30 111 -30q32 0 79 11v118 q-32 -7 -44 -7q-42 0 -42 50v197h77zM1087 724v139q-15 3 -28 3q-32 0 -55.5 -16t-33.5 -46l-10 56h-131v-471h150v306q26 31 82 31q16 0 26 -2zM1124 389h150v471h-150v-471zM1746 638q0 122 -45 179q-40 52 -111 52q-64 0 -117 -56l-8 47h-132v-645l150 25v151 q36 -11 68 -11q83 0 134 56q61 65 61 202zM1278 986q0 33 -23 56t-56 23t-56 -23t-23 -56t23 -56.5t56 -23.5t56 23.5t23 56.5zM2176 629q0 113 -48 176q-50 64 -144 64q-96 0 -151.5 -66t-55.5 -180q0 -128 63 -188q55 -55 161 -55q101 0 160 40l-16 103q-57 -31 -128 -31 q-43 0 -63 19q-23 19 -28 66h248q2 14 2 52zM2304 1280v-1280q0 -52 -38 -90t-90 -38h-2048q-52 0 -90 38t-38 90v1280q0 52 38 90t90 38h2048q52 0 90 -38t38 -90z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1558 684q61 -356 298 -556q0 -52 -38 -90t-90 -38h-448q0 -106 -75 -181t-181 -75t-180.5 74.5t-75.5 180.5zM1024 -176q16 0 16 16t-16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5zM2026 1424q8 -10 7.5 -23.5t-10.5 -22.5 l-1872 -1622q-10 -8 -23.5 -7t-21.5 11l-84 96q-8 10 -7.5 23.5t10.5 21.5l186 161q-19 32 -19 66q50 42 91 88t85 119.5t74.5 158.5t50 206t19.5 260q0 152 117 282.5t307 158.5q-8 19 -8 39q0 40 28 68t68 28t68 -28t28 -68q0 -20 -8 -39q124 -18 219 -82.5t148 -157.5 l418 363q10 8 23.5 7t21.5 -11z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1040 -160q0 16 -16 16q-59 0 -101.5 42.5t-42.5 101.5q0 16 -16 16t-16 -16q0 -73 51.5 -124.5t124.5 -51.5q16 0 16 16zM503 315l877 760q-42 88 -132.5 146.5t-223.5 58.5q-93 0 -169.5 -31.5t-121.5 -80.5t-69 -103t-24 -105q0 -384 -137 -645zM1856 128 q0 -52 -38 -90t-90 -38h-448q0 -106 -75 -181t-181 -75t-180.5 74.5t-75.5 180.5l149 129h757q-166 187 -227 459l111 97q61 -356 298 -556zM1942 1520l84 -96q8 -10 7.5 -23.5t-10.5 -22.5l-1872 -1622q-10 -8 -23.5 -7t-21.5 11l-84 96q-8 10 -7.5 23.5t10.5 21.5l186 161 q-19 32 -19 66q50 42 91 88t85 119.5t74.5 158.5t50 206t19.5 260q0 152 117 282.5t307 158.5q-8 19 -8 39q0 40 28 68t68 28t68 -28t28 -68q0 -20 -8 -39q124 -18 219 -82.5t148 -157.5l418 363q10 8 23.5 7t21.5 -11z" /> +<glyph unicode="" horiz-adv-x="1408" d="M512 160v704q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-704q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM768 160v704q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-704q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM1024 160v704q0 14 -9 23t-23 9h-64q-14 0 -23 -9t-9 -23v-704 q0 -14 9 -23t23 -9h64q14 0 23 9t9 23zM480 1152h448l-48 117q-7 9 -17 11h-317q-10 -2 -17 -11zM1408 1120v-64q0 -14 -9 -23t-23 -9h-96v-948q0 -83 -47 -143.5t-113 -60.5h-832q-66 0 -113 58.5t-47 141.5v952h-96q-14 0 -23 9t-9 23v64q0 14 9 23t23 9h309l70 167 q15 37 54 63t79 26h320q40 0 79 -26t54 -63l70 -167h309q14 0 23 -9t9 -23z" /> +<glyph unicode="" d="M1150 462v-109q0 -50 -36.5 -89t-94 -60.5t-118 -32.5t-117.5 -11q-205 0 -342.5 139t-137.5 346q0 203 136 339t339 136q34 0 75.5 -4.5t93 -18t92.5 -34t69 -56.5t28 -81v-109q0 -16 -16 -16h-118q-16 0 -16 16v70q0 43 -65.5 67.5t-137.5 24.5q-140 0 -228.5 -91.5 t-88.5 -237.5q0 -151 91.5 -249.5t233.5 -98.5q68 0 138 24t70 66v70q0 7 4.5 11.5t10.5 4.5h119q6 0 11 -4.5t5 -11.5zM768 1280q-130 0 -248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51t248.5 51t204 136.5t136.5 204t51 248.5 t-51 248.5t-136.5 204t-204 136.5t-248.5 51zM1536 640q0 -209 -103 -385.5t-279.5 -279.5t-385.5 -103t-385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103t385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" d="M972 761q0 108 -53.5 169t-147.5 61q-63 0 -124 -30.5t-110 -84.5t-79.5 -137t-30.5 -180q0 -112 53.5 -173t150.5 -61q96 0 176 66.5t122.5 166t42.5 203.5zM1536 640q0 -111 -37 -197t-98.5 -135t-131.5 -74.5t-145 -27.5q-6 0 -15.5 -0.5t-16.5 -0.5q-95 0 -142 53 q-28 33 -33 83q-52 -66 -131.5 -110t-173.5 -44q-161 0 -249.5 95.5t-88.5 269.5q0 157 66 290t179 210.5t246 77.5q87 0 155 -35.5t106 -99.5l2 19l11 56q1 6 5.5 12t9.5 6h118q5 0 13 -11q5 -5 3 -16l-120 -614q-5 -24 -5 -48q0 -39 12.5 -52t44.5 -13q28 1 57 5.5t73 24 t77 50t57 89.5t24 137q0 292 -174 466t-466 174q-130 0 -248.5 -51t-204 -136.5t-136.5 -204t-51 -248.5t51 -248.5t136.5 -204t204 -136.5t248.5 -51q228 0 405 144q11 9 24 8t21 -12l41 -49q8 -12 7 -24q-2 -13 -12 -22q-102 -83 -227.5 -128t-258.5 -45q-156 0 -298 61 t-245 164t-164 245t-61 298t61 298t164 245t245 164t298 61q344 0 556 -212t212 -556z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1698 1442q94 -94 94 -226.5t-94 -225.5l-225 -223l104 -104q10 -10 10 -23t-10 -23l-210 -210q-10 -10 -23 -10t-23 10l-105 105l-603 -603q-37 -37 -90 -37h-203l-256 -128l-64 64l128 256v203q0 53 37 90l603 603l-105 105q-10 10 -10 23t10 23l210 210q10 10 23 10 t23 -10l104 -104l223 225q93 94 225.5 94t226.5 -94zM512 64l576 576l-192 192l-576 -576v-192h192z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1615 1536q70 0 122.5 -46.5t52.5 -116.5q0 -63 -45 -151q-332 -629 -465 -752q-97 -91 -218 -91q-126 0 -216.5 92.5t-90.5 219.5q0 128 92 212l638 579q59 54 130 54zM706 502q39 -76 106.5 -130t150.5 -76l1 -71q4 -213 -129.5 -347t-348.5 -134q-123 0 -218 46.5 t-152.5 127.5t-86.5 183t-29 220q7 -5 41 -30t62 -44.5t59 -36.5t46 -17q41 0 55 37q25 66 57.5 112.5t69.5 76t88 47.5t103 25.5t125 10.5z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1792 128v-384h-1792v384q45 0 85 14t59 27.5t47 37.5q30 27 51.5 38t56.5 11t55.5 -11t52.5 -38q29 -25 47 -38t58 -27t86 -14q45 0 85 14.5t58 27t48 37.5q21 19 32.5 27t31 15t43.5 7q35 0 56.5 -11t51.5 -38q28 -24 47 -37.5t59 -27.5t85 -14t85 14t59 27.5t47 37.5 q30 27 51.5 38t56.5 11q34 0 55.5 -11t51.5 -38q28 -24 47 -37.5t59 -27.5t85 -14zM1792 448v-192q-35 0 -55.5 11t-52.5 38q-29 25 -47 38t-58 27t-85 14q-46 0 -86 -14t-58 -27t-47 -38q-22 -19 -33 -27t-31 -15t-44 -7q-35 0 -56.5 11t-51.5 38q-29 25 -47 38t-58 27 t-86 14q-45 0 -85 -14.5t-58 -27t-48 -37.5q-21 -19 -32.5 -27t-31 -15t-43.5 -7q-35 0 -56.5 11t-51.5 38q-28 24 -47 37.5t-59 27.5t-85 14q-46 0 -86 -14t-58 -27t-47 -38q-30 -27 -51.5 -38t-56.5 -11v192q0 80 56 136t136 56h64v448h256v-448h256v448h256v-448h256v448 h256v-448h64q80 0 136 -56t56 -136zM512 1312q0 -77 -36 -118.5t-92 -41.5q-53 0 -90.5 37.5t-37.5 90.5q0 29 9.5 51t23.5 34t31 28t31 31.5t23.5 44.5t9.5 67q38 0 83 -74t45 -150zM1024 1312q0 -77 -36 -118.5t-92 -41.5q-53 0 -90.5 37.5t-37.5 90.5q0 29 9.5 51 t23.5 34t31 28t31 31.5t23.5 44.5t9.5 67q38 0 83 -74t45 -150zM1536 1312q0 -77 -36 -118.5t-92 -41.5q-53 0 -90.5 37.5t-37.5 90.5q0 29 9.5 51t23.5 34t31 28t31 31.5t23.5 44.5t9.5 67q38 0 83 -74t45 -150z" /> +<glyph unicode="" horiz-adv-x="2048" d="M2048 0v-128h-2048v1536h128v-1408h1920zM1664 1024l256 -896h-1664v576l448 576l576 -576z" /> +<glyph unicode="" horiz-adv-x="1792" d="M768 646l546 -546q-106 -108 -247.5 -168t-298.5 -60q-209 0 -385.5 103t-279.5 279.5t-103 385.5t103 385.5t279.5 279.5t385.5 103v-762zM955 640h773q0 -157 -60 -298.5t-168 -247.5zM1664 768h-768v768q209 0 385.5 -103t279.5 -279.5t103 -385.5z" /> +<glyph unicode="" horiz-adv-x="2048" d="M2048 0v-128h-2048v1536h128v-1408h1920zM1920 1248v-435q0 -21 -19.5 -29.5t-35.5 7.5l-121 121l-633 -633q-10 -10 -23 -10t-23 10l-233 233l-416 -416l-192 192l585 585q10 10 23 10t23 -10l233 -233l464 464l-121 121q-16 16 -7.5 35.5t29.5 19.5h435q14 0 23 -9 t9 -23z" /> +<glyph unicode="" horiz-adv-x="1792" d="M1292 832q0 -6 10 -41q10 -29 25 -49.5t41 -34t44 -20t55 -16.5q325 -91 325 -332q0 -146 -105.5 -242.5t-254.5 -96.5q-59 0 -111.5 18.5t-91.5 45.5t-77 74.5t-63 87.5t-53.5 103.5t-43.5 103t-39.5 106.5t-35.5 95q-32 81 -61.5 133.5t-73.5 96.5t-104 64t-142 20 q-96 0 -183 -55.5t-138 -144.5t-51 -185q0 -160 106.5 -279.5t263.5 -119.5q177 0 258 95q56 63 83 116l84 -152q-15 -34 -44 -70l1 -1q-131 -152 -388 -152q-147 0 -269.5 79t-190.5 207.5t-68 274.5q0 105 43.5 206t116 176.5t172 121.5t204.5 46q87 0 159 -19t123.5 -50 t95 -80t72.5 -99t58.5 -117t50.5 -124.5t50 -130.5t55 -127q96 -200 233 -200q81 0 138.5 48.5t57.5 128.5q0 42 -19 72t-50.5 46t-72.5 31.5t-84.5 27t-87.5 34t-81 52t-65 82t-39 122.5q-3 16 -3 33q0 110 87.5 192t198.5 78q78 -3 120.5 -14.5t90.5 -53.5h-1 q12 -11 23 -24.5t26 -36t19 -27.5l-129 -99q-26 49 -54 70v1q-23 21 -97 21q-49 0 -84 -33t-35 -83z" /> +<glyph unicode="" d="M1432 484q0 173 -234 239q-35 10 -53 16.5t-38 25t-29 46.5q0 2 -2 8.5t-3 12t-1 7.5q0 36 24.5 59.5t60.5 23.5q54 0 71 -15h-1q20 -15 39 -51l93 71q-39 54 -49 64q-33 29 -67.5 39t-85.5 10q-80 0 -142 -57.5t-62 -137.5q0 -7 2 -23q16 -96 64.5 -140t148.5 -73 q29 -8 49 -15.5t45 -21.5t38.5 -34.5t13.5 -46.5v-5q1 -58 -40.5 -93t-100.5 -35q-97 0 -167 144q-23 47 -51.5 121.5t-48 125.5t-54 110.5t-74 95.5t-103.5 60.5t-147 24.5q-101 0 -192 -56t-144 -148t-50 -192v-1q4 -108 50.5 -199t133.5 -147.5t196 -56.5q186 0 279 110 q20 27 31 51l-60 109q-42 -80 -99 -116t-146 -36q-115 0 -191 87t-76 204q0 105 82 189t186 84q112 0 170 -53.5t104 -172.5q8 -21 25.5 -68.5t28.5 -76.5t31.5 -74.5t38.5 -74t45.5 -62.5t55.5 -53.5t66 -33t80 -13.5q107 0 183 69.5t76 174.5zM1536 1120v-960 q0 -119 -84.5 -203.5t-203.5 -84.5h-960q-119 0 -203.5 84.5t-84.5 203.5v960q0 119 84.5 203.5t203.5 84.5h960q119 0 203.5 -84.5t84.5 -203.5z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1152 640q0 104 -40.5 198.5t-109.5 163.5t-163.5 109.5t-198.5 40.5t-198.5 -40.5t-163.5 -109.5t-109.5 -163.5t-40.5 -198.5t40.5 -198.5t109.5 -163.5t163.5 -109.5t198.5 -40.5t198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5zM1920 640q0 104 -40.5 198.5 t-109.5 163.5t-163.5 109.5t-198.5 40.5h-386q119 -90 188.5 -224t69.5 -288t-69.5 -288t-188.5 -224h386q104 0 198.5 40.5t163.5 109.5t109.5 163.5t40.5 198.5zM2048 640q0 -130 -51 -248.5t-136.5 -204t-204 -136.5t-248.5 -51h-768q-130 0 -248.5 51t-204 136.5 t-136.5 204t-51 248.5t51 248.5t136.5 204t204 136.5t248.5 51h768q130 0 248.5 -51t204 -136.5t136.5 -204t51 -248.5z" /> +<glyph unicode="" horiz-adv-x="2048" d="M0 640q0 130 51 248.5t136.5 204t204 136.5t248.5 51h768q130 0 248.5 -51t204 -136.5t136.5 -204t51 -248.5t-51 -248.5t-136.5 -204t-204 -136.5t-248.5 -51h-768q-130 0 -248.5 51t-204 136.5t-136.5 204t-51 248.5zM1408 128q104 0 198.5 40.5t163.5 109.5 t109.5 163.5t40.5 198.5t-40.5 198.5t-109.5 163.5t-163.5 109.5t-198.5 40.5t-198.5 -40.5t-163.5 -109.5t-109.5 -163.5t-40.5 -198.5t40.5 -198.5t109.5 -163.5t163.5 -109.5t198.5 -40.5z" /> +<glyph unicode="" horiz-adv-x="2304" d="M762 384h-314q-40 0 -57.5 35t6.5 67l188 251q-65 31 -137 31q-132 0 -226 -94t-94 -226t94 -226t226 -94q115 0 203 72.5t111 183.5zM576 512h186q-18 85 -75 148zM1056 512l288 384h-480l-99 -132q105 -103 126 -252h165zM2176 448q0 132 -94 226t-226 94 q-60 0 -121 -24l174 -260q15 -23 10 -49t-27 -40q-15 -11 -36 -11q-35 0 -53 29l-174 260q-93 -95 -93 -225q0 -132 94 -226t226 -94t226 94t94 226zM2304 448q0 -185 -131.5 -316.5t-316.5 -131.5t-316.5 131.5t-131.5 316.5q0 97 39.5 183.5t109.5 149.5l-65 98l-353 -469 q-18 -26 -51 -26h-197q-23 -164 -149 -274t-294 -110q-185 0 -316.5 131.5t-131.5 316.5t131.5 316.5t316.5 131.5q114 0 215 -55l137 183h-224q-26 0 -45 19t-19 45t19 45t45 19h384v-128h435l-85 128h-222q-26 0 -45 19t-19 45t19 45t45 19h256q33 0 53 -28l267 -400 q91 44 192 44q185 0 316.5 -131.5t131.5 -316.5z" /> +<glyph unicode="" d="M384 320q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1408 320q0 53 -37.5 90.5t-90.5 37.5t-90.5 -37.5t-37.5 -90.5t37.5 -90.5t90.5 -37.5t90.5 37.5t37.5 90.5zM1362 716l-72 384q-5 23 -22.5 37.5t-40.5 14.5 h-918q-23 0 -40.5 -14.5t-22.5 -37.5l-72 -384q-5 -30 14 -53t49 -23h1062q30 0 49 23t14 53zM1136 1328q0 20 -14 34t-34 14h-640q-20 0 -34 -14t-14 -34t14 -34t34 -14h640q20 0 34 14t14 34zM1536 603v-603h-128v-128q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5 t-37.5 90.5v128h-768v-128q0 -53 -37.5 -90.5t-90.5 -37.5t-90.5 37.5t-37.5 90.5v128h-128v603q0 112 25 223l103 454q9 78 97.5 137t230 89t312.5 30t312.5 -30t230 -89t97.5 -137l105 -454q23 -102 23 -223z" /> +<glyph unicode="" horiz-adv-x="2048" d="M1463 704q0 -35 -25 -60.5t-61 -25.5h-702q-36 0 -61 25.5t-25 60.5t25 60.5t61 25.5h702q36 0 61 -25.5t25 -60.5zM1677 704q0 86 -23 170h-982q-36 0 -61 25t-25 60q0 36 25 61t61 25h908q-88 143 -235 227t-320 84q-177 0 -327.5 -87.5t-238 -237.5t-87.5 -327 q0 -86 23 -170h982q36 0 61 -25t25 -60q0 -36 -25 -61t-61 -25h-908q88 -143 235.5 -227t320.5 -84q132 0 253 51.5t208 139t139 208t52 253.5zM2048 959q0 -35 -25 -60t-61 -25h-131q17 -85 17 -170q0 -167 -65.5 -319.5t-175.5 -263t-262.5 -176t-319.5 -65.5 q-246 0 -448.5 133t-301.5 350h-189q-36 0 -61 25t-25 61q0 35 25 60t61 25h132q-17 85 -17 170q0 167 65.5 319.5t175.5 263t262.5 176t320.5 65.5q245 0 447.5 -133t301.5 -350h188q36 0 61 -25t25 -61z" /> +<glyph unicode="" horiz-adv-x="1280" d="M953 1158l-114 -328l117 -21q165 451 165 518q0 56 -38 56q-57 0 -130 -225zM654 471l33 -88q37 42 71 67l-33 5.5t-38.5 7t-32.5 8.5zM362 1367q0 -98 159 -521q18 10 49 10q15 0 75 -5l-121 351q-75 220 -123 220q-19 0 -29 -17.5t-10 -37.5zM283 608q0 -36 51.5 -119 t117.5 -153t100 -70q14 0 25.5 13t11.5 27q0 24 -32 102q-13 32 -32 72t-47.5 89t-61.5 81t-62 32q-20 0 -45.5 -27t-25.5 -47zM125 273q0 -41 25 -104q59 -145 183.5 -227t281.5 -82q227 0 382 170q152 169 152 427q0 43 -1 67t-11.5 62t-30.5 56q-56 49 -211.5 75.5 t-270.5 26.5q-37 0 -49 -11q-12 -5 -12 -35q0 -34 21.5 -60t55.5 -40t77.5 -23.5t87.5 -11.5t85 -4t70 0h23q24 0 40 -19q15 -19 19 -55q-28 -28 -96 -54q-61 -22 -93 -46q-64 -46 -108.5 -114t-44.5 -137q0 -31 18.5 -88.5t18.5 -87.5l-3 -12q-4 -12 -4 -14 q-137 10 -146 216q-8 -2 -41 -2q2 -7 2 -21q0 -53 -40.5 -89.5t-94.5 -36.5q-82 0 -166.5 78t-84.5 159q0 34 33 67q52 -64 60 -76q77 -104 133 -104q12 0 26.5 8.5t14.5 20.5q0 34 -87.5 145t-116.5 111q-43 0 -70 -44.5t-27 -90.5zM11 264q0 101 42.5 163t136.5 88 q-28 74 -28 104q0 62 61 123t122 61q29 0 70 -15q-163 462 -163 567q0 80 41 130.5t119 50.5q131 0 325 -581q6 -17 8 -23q6 16 29 79.5t43.5 118.5t54 127.5t64.5 123t70.5 86.5t76.5 36q71 0 112 -49t41 -122q0 -108 -159 -550q61 -15 100.5 -46t58.5 -78t26 -93.5 t7 -110.5q0 -150 -47 -280t-132 -225t-211 -150t-278 -55q-111 0 -223 42q-149 57 -258 191.5t-109 286.5z" /> +<glyph unicode="" horiz-adv-x="2048" d="M785 528h207q-14 -158 -98.5 -248.5t-214.5 -90.5q-162 0 -254.5 116t-92.5 316q0 194 93 311.5t233 117.5q148 0 232 -87t97 -247h-203q-5 64 -35.5 99t-81.5 35q-57 0 -88.5 -60.5t-31.5 -177.5q0 -48 5 -84t18 -69.5t40 -51.5t66 -18q95 0 109 139zM1497 528h206 q-14 -158 -98 -248.5t-214 -90.5q-162 0 -254.5 116t-92.5 316q0 194 93 311.5t233 117.5q148 0 232 -87t97 -247h-204q-4 64 -35 99t-81 35q-57 0 -88.5 -60.5t-31.5 -177.5q0 -48 5 -84t18 -69.5t39.5 -51.5t65.5 -18q49 0 76.5 38t33.5 101zM1856 647q0 207 -15.5 307 t-60.5 161q-6 8 -13.5 14t-21.5 15t-16 11q-86 63 -697 63q-625 0 -710 -63q-5 -4 -17.5 -11.5t-21 -14t-14.5 -14.5q-45 -60 -60 -159.5t-15 -308.5q0 -208 15 -307.5t60 -160.5q6 -8 15 -15t20.5 -14t17.5 -12q44 -33 239.5 -49t470.5 -16q610 0 697 65q5 4 17 11t20.5 14 t13.5 16q46 60 61 159t15 309zM2048 1408v-1536h-2048v1536h2048z" /> +<glyph unicode="" d="M992 912v-496q0 -14 -9 -23t-23 -9h-160q-14 0 -23 9t-9 23v496q0 112 -80 192t-192 80h-272v-1152q0 -14 -9 -23t-23 -9h-160q-14 0 -23 9t-9 23v1344q0 14 9 23t23 9h464q135 0 249 -66.5t180.5 -180.5t66.5 -249zM1376 1376v-880q0 -135 -66.5 -249t-180.5 -180.5 t-249 -66.5h-464q-14 0 -23 9t-9 23v960q0 14 9 23t23 9h160q14 0 23 -9t9 -23v-768h272q112 0 192 80t80 192v880q0 14 9 23t23 9h160q14 0 23 -9t9 -23z" /> +<glyph unicode="" d="M1311 694v-114q0 -24 -13.5 -38t-37.5 -14h-202q-24 0 -38 14t-14 38v114q0 24 14 38t38 14h202q24 0 37.5 -14t13.5 -38zM821 464v250q0 53 -32.5 85.5t-85.5 32.5h-133q-68 0 -96 -52q-28 52 -96 52h-130q-53 0 -85.5 -32.5t-32.5 -85.5v-250q0 -22 21 -22h55 q22 0 22 22v230q0 24 13.5 38t38.5 14h94q24 0 38 -14t14 -38v-230q0 -22 21 -22h54q22 0 22 22v230q0 24 14 38t38 14h97q24 0 37.5 -14t13.5 -38v-230q0 -22 22 -22h55q21 0 21 22zM1410 560v154q0 53 -33 85.5t-86 32.5h-264q-53 0 -86 -32.5t-33 -85.5v-410 q0 -21 22 -21h55q21 0 21 21v180q31 -42 94 -42h191q53 0 86 32.5t33 85.5zM1536 1176v-1072q0 -96 -68 -164t-164 -68h-1072q-96 0 -164 68t-68 164v1072q0 96 68 164t164 68h1072q96 0 164 -68t68 -164z" /> +<glyph unicode="" horiz-adv-x="1792" /> +<glyph unicode="" horiz-adv-x="1792" /> +<glyph unicode="" horiz-adv-x="1792" /> +</font> +</defs></svg> \ No newline at end of file diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.ttf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.ttf new file mode 100644 index 0000000000000000000000000000000000000000..96a3639cdde5e8ab459c6380e3b9524ee81641dc Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.ttf differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.woff b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.woff new file mode 100644 index 0000000000000000000000000000000000000000..628b6a52a87e62c6f22426e17c01f6a303aa194e Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/fonts/fontawesome-webfont.woff differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/img/ti_logo.png b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/img/ti_logo.png new file mode 100644 index 0000000000000000000000000000000000000000..5f45933524af35b579734baa58dfa81a5d335fb1 Binary files /dev/null and b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/img/ti_logo.png differ diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/js/modernizr.min.js b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/js/modernizr.min.js new file mode 100644 index 0000000000000000000000000000000000000000..f65d47974786ee51c258f680bd9be621629244f5 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/js/modernizr.min.js @@ -0,0 +1,4 @@ +/* Modernizr 2.6.2 (Custom Build) | MIT & BSD + * Build: http://modernizr.com/download/#-fontface-backgroundsize-borderimage-borderradius-boxshadow-flexbox-hsla-multiplebgs-opacity-rgba-textshadow-cssanimations-csscolumns-generatedcontent-cssgradients-cssreflections-csstransforms-csstransforms3d-csstransitions-applicationcache-canvas-canvastext-draganddrop-hashchange-history-audio-video-indexeddb-input-inputtypes-localstorage-postmessage-sessionstorage-websockets-websqldatabase-webworkers-geolocation-inlinesvg-smil-svg-svgclippaths-touch-webgl-shiv-mq-cssclasses-addtest-prefixed-teststyles-testprop-testallprops-hasevent-prefixes-domprefixes-load + */ +;window.Modernizr=function(a,b,c){function D(a){j.cssText=a}function E(a,b){return D(n.join(a+";")+(b||""))}function F(a,b){return typeof a===b}function G(a,b){return!!~(""+a).indexOf(b)}function H(a,b){for(var d in a){var e=a[d];if(!G(e,"-")&&j[e]!==c)return b=="pfx"?e:!0}return!1}function I(a,b,d){for(var e in a){var f=b[a[e]];if(f!==c)return d===!1?a[e]:F(f,"function")?f.bind(d||b):f}return!1}function J(a,b,c){var d=a.charAt(0).toUpperCase()+a.slice(1),e=(a+" "+p.join(d+" ")+d).split(" ");return F(b,"string")||F(b,"undefined")?H(e,b):(e=(a+" "+q.join(d+" ")+d).split(" "),I(e,b,c))}function K(){e.input=function(c){for(var d=0,e=c.length;d<e;d++)u[c[d]]=c[d]in k;return u.list&&(u.list=!!b.createElement("datalist")&&!!a.HTMLDataListElement),u}("autocomplete autofocus list placeholder max min multiple pattern required step".split(" ")),e.inputtypes=function(a){for(var d=0,e,f,h,i=a.length;d<i;d++)k.setAttribute("type",f=a[d]),e=k.type!=="text",e&&(k.value=l,k.style.cssText="position:absolute;visibility:hidden;",/^range$/.test(f)&&k.style.WebkitAppearance!==c?(g.appendChild(k),h=b.defaultView,e=h.getComputedStyle&&h.getComputedStyle(k,null).WebkitAppearance!=="textfield"&&k.offsetHeight!==0,g.removeChild(k)):/^(search|tel)$/.test(f)||(/^(url|email)$/.test(f)?e=k.checkValidity&&k.checkValidity()===!1:e=k.value!=l)),t[a[d]]=!!e;return t}("search tel url email datetime date month week time datetime-local number range color".split(" "))}var d="2.6.2",e={},f=!0,g=b.documentElement,h="modernizr",i=b.createElement(h),j=i.style,k=b.createElement("input"),l=":)",m={}.toString,n=" -webkit- -moz- -o- -ms- ".split(" "),o="Webkit Moz O ms",p=o.split(" "),q=o.toLowerCase().split(" "),r={svg:"http://www.w3.org/2000/svg"},s={},t={},u={},v=[],w=v.slice,x,y=function(a,c,d,e){var f,i,j,k,l=b.createElement("div"),m=b.body,n=m||b.createElement("body");if(parseInt(d,10))while(d--)j=b.createElement("div"),j.id=e?e[d]:h+(d+1),l.appendChild(j);return f=["­",'<style id="s',h,'">',a,"</style>"].join(""),l.id=h,(m?l:n).innerHTML+=f,n.appendChild(l),m||(n.style.background="",n.style.overflow="hidden",k=g.style.overflow,g.style.overflow="hidden",g.appendChild(n)),i=c(l,a),m?l.parentNode.removeChild(l):(n.parentNode.removeChild(n),g.style.overflow=k),!!i},z=function(b){var c=a.matchMedia||a.msMatchMedia;if(c)return c(b).matches;var d;return y("@media "+b+" { #"+h+" { position: absolute; } }",function(b){d=(a.getComputedStyle?getComputedStyle(b,null):b.currentStyle)["position"]=="absolute"}),d},A=function(){function d(d,e){e=e||b.createElement(a[d]||"div"),d="on"+d;var f=d in e;return f||(e.setAttribute||(e=b.createElement("div")),e.setAttribute&&e.removeAttribute&&(e.setAttribute(d,""),f=F(e[d],"function"),F(e[d],"undefined")||(e[d]=c),e.removeAttribute(d))),e=null,f}var a={select:"input",change:"input",submit:"form",reset:"form",error:"img",load:"img",abort:"img"};return d}(),B={}.hasOwnProperty,C;!F(B,"undefined")&&!F(B.call,"undefined")?C=function(a,b){return B.call(a,b)}:C=function(a,b){return b in a&&F(a.constructor.prototype[b],"undefined")},Function.prototype.bind||(Function.prototype.bind=function(b){var c=this;if(typeof c!="function")throw new TypeError;var d=w.call(arguments,1),e=function(){if(this instanceof e){var a=function(){};a.prototype=c.prototype;var f=new a,g=c.apply(f,d.concat(w.call(arguments)));return Object(g)===g?g:f}return c.apply(b,d.concat(w.call(arguments)))};return e}),s.flexbox=function(){return J("flexWrap")},s.canvas=function(){var a=b.createElement("canvas");return!!a.getContext&&!!a.getContext("2d")},s.canvastext=function(){return!!e.canvas&&!!F(b.createElement("canvas").getContext("2d").fillText,"function")},s.webgl=function(){return!!a.WebGLRenderingContext},s.touch=function(){var c;return"ontouchstart"in a||a.DocumentTouch&&b instanceof DocumentTouch?c=!0:y(["@media (",n.join("touch-enabled),("),h,")","{#modernizr{top:9px;position:absolute}}"].join(""),function(a){c=a.offsetTop===9}),c},s.geolocation=function(){return"geolocation"in navigator},s.postmessage=function(){return!!a.postMessage},s.websqldatabase=function(){return!!a.openDatabase},s.indexedDB=function(){return!!J("indexedDB",a)},s.hashchange=function(){return A("hashchange",a)&&(b.documentMode===c||b.documentMode>7)},s.history=function(){return!!a.history&&!!history.pushState},s.draganddrop=function(){var a=b.createElement("div");return"draggable"in a||"ondragstart"in a&&"ondrop"in a},s.websockets=function(){return"WebSocket"in a||"MozWebSocket"in a},s.rgba=function(){return D("background-color:rgba(150,255,150,.5)"),G(j.backgroundColor,"rgba")},s.hsla=function(){return D("background-color:hsla(120,40%,100%,.5)"),G(j.backgroundColor,"rgba")||G(j.backgroundColor,"hsla")},s.multiplebgs=function(){return D("background:url(https://),url(https://),red url(https://)"),/(url\s*\(.*?){3}/.test(j.background)},s.backgroundsize=function(){return J("backgroundSize")},s.borderimage=function(){return J("borderImage")},s.borderradius=function(){return J("borderRadius")},s.boxshadow=function(){return J("boxShadow")},s.textshadow=function(){return b.createElement("div").style.textShadow===""},s.opacity=function(){return E("opacity:.55"),/^0.55$/.test(j.opacity)},s.cssanimations=function(){return J("animationName")},s.csscolumns=function(){return J("columnCount")},s.cssgradients=function(){var a="background-image:",b="gradient(linear,left top,right bottom,from(#9f9),to(white));",c="linear-gradient(left top,#9f9, white);";return D((a+"-webkit- ".split(" ").join(b+a)+n.join(c+a)).slice(0,-a.length)),G(j.backgroundImage,"gradient")},s.cssreflections=function(){return J("boxReflect")},s.csstransforms=function(){return!!J("transform")},s.csstransforms3d=function(){var a=!!J("perspective");return a&&"webkitPerspective"in g.style&&y("@media (transform-3d),(-webkit-transform-3d){#modernizr{left:9px;position:absolute;height:3px;}}",function(b,c){a=b.offsetLeft===9&&b.offsetHeight===3}),a},s.csstransitions=function(){return J("transition")},s.fontface=function(){var a;return y('@font-face {font-family:"font";src:url("https://")}',function(c,d){var e=b.getElementById("smodernizr"),f=e.sheet||e.styleSheet,g=f?f.cssRules&&f.cssRules[0]?f.cssRules[0].cssText:f.cssText||"":"";a=/src/i.test(g)&&g.indexOf(d.split(" ")[0])===0}),a},s.generatedcontent=function(){var a;return y(["#",h,"{font:0/0 a}#",h,':after{content:"',l,'";visibility:hidden;font:3px/1 a}'].join(""),function(b){a=b.offsetHeight>=3}),a},s.video=function(){var a=b.createElement("video"),c=!1;try{if(c=!!a.canPlayType)c=new Boolean(c),c.ogg=a.canPlayType('video/ogg; codecs="theora"').replace(/^no$/,""),c.h264=a.canPlayType('video/mp4; codecs="avc1.42E01E"').replace(/^no$/,""),c.webm=a.canPlayType('video/webm; codecs="vp8, vorbis"').replace(/^no$/,"")}catch(d){}return c},s.audio=function(){var a=b.createElement("audio"),c=!1;try{if(c=!!a.canPlayType)c=new Boolean(c),c.ogg=a.canPlayType('audio/ogg; codecs="vorbis"').replace(/^no$/,""),c.mp3=a.canPlayType("audio/mpeg;").replace(/^no$/,""),c.wav=a.canPlayType('audio/wav; codecs="1"').replace(/^no$/,""),c.m4a=(a.canPlayType("audio/x-m4a;")||a.canPlayType("audio/aac;")).replace(/^no$/,"")}catch(d){}return c},s.localstorage=function(){try{return localStorage.setItem(h,h),localStorage.removeItem(h),!0}catch(a){return!1}},s.sessionstorage=function(){try{return sessionStorage.setItem(h,h),sessionStorage.removeItem(h),!0}catch(a){return!1}},s.webworkers=function(){return!!a.Worker},s.applicationcache=function(){return!!a.applicationCache},s.svg=function(){return!!b.createElementNS&&!!b.createElementNS(r.svg,"svg").createSVGRect},s.inlinesvg=function(){var a=b.createElement("div");return a.innerHTML="<svg/>",(a.firstChild&&a.firstChild.namespaceURI)==r.svg},s.smil=function(){return!!b.createElementNS&&/SVGAnimate/.test(m.call(b.createElementNS(r.svg,"animate")))},s.svgclippaths=function(){return!!b.createElementNS&&/SVGClipPath/.test(m.call(b.createElementNS(r.svg,"clipPath")))};for(var L in s)C(s,L)&&(x=L.toLowerCase(),e[x]=s[L](),v.push((e[x]?"":"no-")+x));return e.input||K(),e.addTest=function(a,b){if(typeof a=="object")for(var d in a)C(a,d)&&e.addTest(d,a[d]);else{a=a.toLowerCase();if(e[a]!==c)return e;b=typeof b=="function"?b():b,typeof f!="undefined"&&f&&(g.className+=" "+(b?"":"no-")+a),e[a]=b}return e},D(""),i=k=null,function(a,b){function k(a,b){var c=a.createElement("p"),d=a.getElementsByTagName("head")[0]||a.documentElement;return c.innerHTML="x<style>"+b+"</style>",d.insertBefore(c.lastChild,d.firstChild)}function l(){var a=r.elements;return typeof a=="string"?a.split(" "):a}function m(a){var b=i[a[g]];return b||(b={},h++,a[g]=h,i[h]=b),b}function n(a,c,f){c||(c=b);if(j)return c.createElement(a);f||(f=m(c));var g;return f.cache[a]?g=f.cache[a].cloneNode():e.test(a)?g=(f.cache[a]=f.createElem(a)).cloneNode():g=f.createElem(a),g.canHaveChildren&&!d.test(a)?f.frag.appendChild(g):g}function o(a,c){a||(a=b);if(j)return a.createDocumentFragment();c=c||m(a);var d=c.frag.cloneNode(),e=0,f=l(),g=f.length;for(;e<g;e++)d.createElement(f[e]);return d}function p(a,b){b.cache||(b.cache={},b.createElem=a.createElement,b.createFrag=a.createDocumentFragment,b.frag=b.createFrag()),a.createElement=function(c){return r.shivMethods?n(c,a,b):b.createElem(c)},a.createDocumentFragment=Function("h,f","return function(){var n=f.cloneNode(),c=n.createElement;h.shivMethods&&("+l().join().replace(/\w+/g,function(a){return b.createElem(a),b.frag.createElement(a),'c("'+a+'")'})+");return n}")(r,b.frag)}function q(a){a||(a=b);var c=m(a);return r.shivCSS&&!f&&!c.hasCSS&&(c.hasCSS=!!k(a,"article,aside,figcaption,figure,footer,header,hgroup,nav,section{display:block}mark{background:#FF0;color:#000}")),j||p(a,c),a}var c=a.html5||{},d=/^<|^(?:button|map|select|textarea|object|iframe|option|optgroup)$/i,e=/^(?:a|b|code|div|fieldset|h1|h2|h3|h4|h5|h6|i|label|li|ol|p|q|span|strong|style|table|tbody|td|th|tr|ul)$/i,f,g="_html5shiv",h=0,i={},j;(function(){try{var a=b.createElement("a");a.innerHTML="<xyz></xyz>",f="hidden"in a,j=a.childNodes.length==1||function(){b.createElement("a");var a=b.createDocumentFragment();return typeof a.cloneNode=="undefined"||typeof a.createDocumentFragment=="undefined"||typeof a.createElement=="undefined"}()}catch(c){f=!0,j=!0}})();var r={elements:c.elements||"abbr article aside audio bdi canvas data datalist details figcaption figure footer header hgroup mark meter nav output progress section summary time video",shivCSS:c.shivCSS!==!1,supportsUnknownElements:j,shivMethods:c.shivMethods!==!1,type:"default",shivDocument:q,createElement:n,createDocumentFragment:o};a.html5=r,q(b)}(this,b),e._version=d,e._prefixes=n,e._domPrefixes=q,e._cssomPrefixes=p,e.mq=z,e.hasEvent=A,e.testProp=function(a){return H([a])},e.testAllProps=J,e.testStyles=y,e.prefixed=function(a,b,c){return b?J(a,b,c):J(a,"pfx")},g.className=g.className.replace(/(^|\s)no-js(\s|$)/,"$1$2")+(f?" js "+v.join(" "):""),e}(this,this.document),function(a,b,c){function d(a){return"[object Function]"==o.call(a)}function e(a){return"string"==typeof a}function f(){}function g(a){return!a||"loaded"==a||"complete"==a||"uninitialized"==a}function h(){var a=p.shift();q=1,a?a.t?m(function(){("c"==a.t?B.injectCss:B.injectJs)(a.s,0,a.a,a.x,a.e,1)},0):(a(),h()):q=0}function i(a,c,d,e,f,i,j){function k(b){if(!o&&g(l.readyState)&&(u.r=o=1,!q&&h(),l.onload=l.onreadystatechange=null,b)){"img"!=a&&m(function(){t.removeChild(l)},50);for(var d in y[c])y[c].hasOwnProperty(d)&&y[c][d].onload()}}var j=j||B.errorTimeout,l=b.createElement(a),o=0,r=0,u={t:d,s:c,e:f,a:i,x:j};1===y[c]&&(r=1,y[c]=[]),"object"==a?l.data=c:(l.src=c,l.type=a),l.width=l.height="0",l.onerror=l.onload=l.onreadystatechange=function(){k.call(this,r)},p.splice(e,0,u),"img"!=a&&(r||2===y[c]?(t.insertBefore(l,s?null:n),m(k,j)):y[c].push(l))}function j(a,b,c,d,f){return q=0,b=b||"j",e(a)?i("c"==b?v:u,a,b,this.i++,c,d,f):(p.splice(this.i++,0,a),1==p.length&&h()),this}function k(){var a=B;return a.loader={load:j,i:0},a}var l=b.documentElement,m=a.setTimeout,n=b.getElementsByTagName("script")[0],o={}.toString,p=[],q=0,r="MozAppearance"in l.style,s=r&&!!b.createRange().compareNode,t=s?l:n.parentNode,l=a.opera&&"[object Opera]"==o.call(a.opera),l=!!b.attachEvent&&!l,u=r?"object":l?"script":"img",v=l?"script":u,w=Array.isArray||function(a){return"[object Array]"==o.call(a)},x=[],y={},z={timeout:function(a,b){return b.length&&(a.timeout=b[0]),a}},A,B;B=function(a){function b(a){var a=a.split("!"),b=x.length,c=a.pop(),d=a.length,c={url:c,origUrl:c,prefixes:a},e,f,g;for(f=0;f<d;f++)g=a[f].split("="),(e=z[g.shift()])&&(c=e(c,g));for(f=0;f<b;f++)c=x[f](c);return c}function g(a,e,f,g,h){var i=b(a),j=i.autoCallback;i.url.split(".").pop().split("?").shift(),i.bypass||(e&&(e=d(e)?e:e[a]||e[g]||e[a.split("/").pop().split("?")[0]]),i.instead?i.instead(a,e,f,g,h):(y[i.url]?i.noexec=!0:y[i.url]=1,f.load(i.url,i.forceCSS||!i.forceJS&&"css"==i.url.split(".").pop().split("?").shift()?"c":c,i.noexec,i.attrs,i.timeout),(d(e)||d(j))&&f.load(function(){k(),e&&e(i.origUrl,h,g),j&&j(i.origUrl,h,g),y[i.url]=2})))}function h(a,b){function c(a,c){if(a){if(e(a))c||(j=function(){var a=[].slice.call(arguments);k.apply(this,a),l()}),g(a,j,b,0,h);else if(Object(a)===a)for(n in m=function(){var b=0,c;for(c in a)a.hasOwnProperty(c)&&b++;return b}(),a)a.hasOwnProperty(n)&&(!c&&!--m&&(d(j)?j=function(){var a=[].slice.call(arguments);k.apply(this,a),l()}:j[n]=function(a){return function(){var b=[].slice.call(arguments);a&&a.apply(this,b),l()}}(k[n])),g(a[n],j,b,n,h))}else!c&&l()}var h=!!a.test,i=a.load||a.both,j=a.callback||f,k=j,l=a.complete||f,m,n;c(h?a.yep:a.nope,!!i),i&&c(i)}var i,j,l=this.yepnope.loader;if(e(a))g(a,0,l,0);else if(w(a))for(i=0;i<a.length;i++)j=a[i],e(j)?g(j,0,l,0):w(j)?B(j):Object(j)===j&&h(j,l);else Object(a)===a&&h(a,l)},B.addPrefix=function(a,b){z[a]=b},B.addFilter=function(a){x.push(a)},B.errorTimeout=1e4,null==b.readyState&&b.addEventListener&&(b.readyState="loading",b.addEventListener("DOMContentLoaded",A=function(){b.removeEventListener("DOMContentLoaded",A,0),b.readyState="complete"},0)),a.yepnope=k(),a.yepnope.executeStack=h,a.yepnope.injectJs=function(a,c,d,e,i,j){var k=b.createElement("script"),l,o,e=e||B.errorTimeout;k.src=a;for(o in d)k.setAttribute(o,d[o]);c=j?h:c||f,k.onreadystatechange=k.onload=function(){!l&&g(k.readyState)&&(l=1,c(),k.onload=k.onreadystatechange=null)},m(function(){l||(l=1,c(1))},e),i?k.onload():n.parentNode.insertBefore(k,n)},a.yepnope.injectCss=function(a,c,d,e,g,i){var e=b.createElement("link"),j,c=i?h:c||f;e.href=a,e.rel="stylesheet",e.type="text/css";for(j in d)e.setAttribute(j,d[j]);g||(n.parentNode.insertBefore(e,n),m(c,0))}}(this,document),Modernizr.load=function(){yepnope.apply(window,[].slice.call(arguments,0))}; diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/js/theme.js b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/js/theme.js new file mode 100644 index 0000000000000000000000000000000000000000..48a9f06b5b893afd1afca05d880273eb6cbaf941 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/static/js/theme.js @@ -0,0 +1,153 @@ +require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"sphinx-rtd-theme":[function(require,module,exports){ +var jQuery = (typeof(window) != 'undefined') ? window.jQuery : require('jquery'); + +// Sphinx theme nav state +function ThemeNav () { + + var nav = { + navBar: null, + win: null, + winScroll: false, + winResize: false, + linkScroll: false, + winPosition: 0, + winHeight: null, + docHeight: null, + isRunning: null + }; + + nav.enable = function () { + var self = this; + + jQuery(function ($) { + self.init($); + + self.reset(); + self.win.on('hashchange', self.reset); + + // Set scroll monitor + self.win.on('scroll', function () { + if (!self.linkScroll) { + self.winScroll = true; + } + }); + setInterval(function () { if (self.winScroll) self.onScroll(); }, 25); + + // Set resize monitor + self.win.on('resize', function () { + self.winResize = true; + }); + setInterval(function () { if (self.winResize) self.onResize(); }, 25); + self.onResize(); + }); + }; + + nav.init = function ($) { + var doc = $(document), + self = this; + + this.navBar = $('div.wy-side-scroll:first'); + this.win = $(window); + + // Set up javascript UX bits + $(document) + // Shift nav in mobile when clicking the menu. + .on('click', "[data-toggle='wy-nav-top']", function() { + $("[data-toggle='wy-nav-shift']").toggleClass("shift"); + $("[data-toggle='rst-versions']").toggleClass("shift"); + }) + + // Nav menu link click operations + .on('click', ".wy-menu-vertical .current ul li a", function() { + var target = $(this); + // Close menu when you click a link. + $("[data-toggle='wy-nav-shift']").removeClass("shift"); + $("[data-toggle='rst-versions']").toggleClass("shift"); + // Handle dynamic display of l3 and l4 nav lists + self.toggleCurrent(target); + self.hashChange(); + }) + .on('click', "[data-toggle='rst-current-version']", function() { + $("[data-toggle='rst-versions']").toggleClass("shift-up"); + }) + + // Make tables responsive + $("table.docutils:not(.field-list)") + .wrap("<div class='wy-table-responsive'></div>"); + + // Add expand links to all parents of nested ul + $('.wy-menu-vertical ul').not('.simple').siblings('a').each(function () { + var link = $(this); + expand = $('<span class="toctree-expand"></span>'); + expand.on('click', function (ev) { + self.toggleCurrent(link); + ev.stopPropagation(); + return false; + }); + link.prepend(expand); + }); + }; + + nav.reset = function () { + // Get anchor from URL and open up nested nav + var anchor = encodeURI(window.location.hash); + if (anchor) { + try { + var link = $('.wy-menu-vertical') + .find('[href="' + anchor + '"]'); + $('.wy-menu-vertical li.toctree-l1 li.current') + .removeClass('current'); + link.closest('li.toctree-l2').addClass('current'); + link.closest('li.toctree-l3').addClass('current'); + link.closest('li.toctree-l4').addClass('current'); + } + catch (err) { + console.log("Error expanding nav for anchor", err); + } + } + }; + + nav.onScroll = function () { + this.winScroll = false; + var newWinPosition = this.win.scrollTop(), + winBottom = newWinPosition + this.winHeight, + navPosition = this.navBar.scrollTop(), + newNavPosition = navPosition + (newWinPosition - this.winPosition); + if (newWinPosition < 0 || winBottom > this.docHeight) { + return; + } + this.navBar.scrollTop(newNavPosition); + this.winPosition = newWinPosition; + }; + + nav.onResize = function () { + this.winResize = false; + this.winHeight = this.win.height(); + this.docHeight = $(document).height(); + }; + + nav.hashChange = function () { + this.linkScroll = true; + this.win.one('hashchange', function () { + this.linkScroll = false; + }); + }; + + nav.toggleCurrent = function (elem) { + var parent_li = elem.closest('li'); + parent_li.siblings('li.current').removeClass('current'); + parent_li.siblings().find('li.current').removeClass('current'); + parent_li.find('> ul li.current').removeClass('current'); + parent_li.toggleClass('current'); + } + + return nav; +}; + +module.exports.ThemeNav = ThemeNav(); + +if (typeof(window) != 'undefined') { + window.SphinxRtdTheme = { StickyNav: module.exports.ThemeNav }; +} + +},{"jquery":"jquery"}]},{},["sphinx-rtd-theme"]); diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/theme.conf b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/theme.conf new file mode 100644 index 0000000000000000000000000000000000000000..3896d4d38fbdde84e29f9a37ed8c32448553f882 --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/theme.conf @@ -0,0 +1,11 @@ +[theme] +inherit = basic +stylesheet = css/theme.css + +[options] +typekit_id = hiw1hhg +analytics_id = +sticky_navigation = False +logo_only = +collapse_navigation = False +display_version = True diff --git a/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/versions.html b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/versions.html new file mode 100644 index 0000000000000000000000000000000000000000..8b3eb79d2592e8aae24d868c7b2832a855bbc96e --- /dev/null +++ b/beaglebone-ai-64/edge_ai_apps/themes/sphinx_rtd_theme_ti/versions.html @@ -0,0 +1,37 @@ +{% if READTHEDOCS %} +{# Add rst-badge after rst-versions for small badge style. #} + <div class="rst-versions" data-toggle="rst-versions" role="note" aria-label="versions"> + <span class="rst-current-version" data-toggle="rst-current-version"> + <span class="fa fa-book"> Read the Docs</span> + v: {{ current_version }} + <span class="fa fa-caret-down"></span> + </span> + <div class="rst-other-versions"> + <dl> + <dt>Versions</dt> + {% for slug, url in versions %} + <dd><a href="{{ url }}">{{ slug }}</a></dd> + {% endfor %} + </dl> + <dl> + <dt>Downloads</dt> + {% for type, url in downloads %} + <dd><a href="{{ url }}">{{ type }}</a></dd> + {% endfor %} + </dl> + <dl> + <dt>On Read the Docs</dt> + <dd> + <a href="//{{ PRODUCTION_DOMAIN }}/projects/{{ slug }}/?fromdocs={{ slug }}">Project Home</a> + </dd> + <dd> + <a href="//{{ PRODUCTION_DOMAIN }}/builds/{{ slug }}/?fromdocs={{ slug }}">Builds</a> + </dd> + </dl> + <hr/> + Free document hosting provided by <a href="http://www.readthedocs.org">Read the Docs</a>. + + </div> + </div> +{% endif %} + diff --git a/beaglebone-ai-64/index.rst b/beaglebone-ai-64/index.rst index e07ca87c0730c89bda8eada1abe702f31340afd0..21e1063fe6df6867b6676f741e64f715495ec93d 100644 --- a/beaglebone-ai-64/index.rst +++ b/beaglebone-ai-64/index.rst @@ -16,3 +16,4 @@ BeagleBone® AI-64 brings a complete system for developing artificial intelligen ch09.rst ch10.rst ch11.rst + edge_ai_apps/index.rst diff --git a/bone-cook-book/01basics/basics.rst b/beaglebone-cookbook/01basics/basics.rst similarity index 92% rename from bone-cook-book/01basics/basics.rst rename to beaglebone-cookbook/01basics/basics.rst index 9dc35327fda0b322d41307158cf549921a04dd72..b85bd045ab1fe60ba6651a9c310c8c4c960f60e8 100644 --- a/bone-cook-book/01basics/basics.rst +++ b/beaglebone-cookbook/01basics/basics.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-basics: +.. _beaglebone-cookbook-basics: Basics ####### @@ -9,8 +9,8 @@ Introduction When you buy BeagleBone Black, pretty much everything you need to get going comes with it. You can just plug it into the USB of a host computer, and it works. The goal of this chapter is to show what you can do with your Bone, right out of the box. It has enough -information to carry through the next three chapters on sensors (:ref:`sensors <bone-cook-book-sensors>`), -displays (:ref:`display <bone-cook-book-display>`), and motors (:ref:`motors <bone-cook-book-motors>`). +information to carry through the next three chapters on sensors (:ref:`sensors <beaglebone-cookbook-sensors>`), +displays (:ref:`display <beaglebone-cookbook-display>`), and motors (:ref:`motors <beaglebone-cookbook-motors>`). Picking Your Beagle --------------------- @@ -57,7 +57,8 @@ a new USB drive appear on your host computer. :ref:`figure below <basics_01getti shows how it will appear on a Windows host, and Linux and Mac hosts will look similar. The Bone acting like a USB drive and the files you see are located on the Bone. -.. todo:: Update +.. todo:: + Update .. _basics_01gettingStarted_fig: @@ -172,6 +173,7 @@ perform physical computing tasks without first learning Linux. Solution *********** + Plug your board into the USB of your host computer and browse to http://192.168.7.2:3000 using Google Chrome or Firefox (as shown in :ref:`basics of out of the box <basics_out_of_the_box>`). In the left @@ -346,7 +348,8 @@ with *am57xx-debian-* is for programming the Beagle AI's. The onboard flash is often called the *eMMC* memory. We just call it *onboard flash*, but you'll often see *eMMC* appearing in filenames of images used to update the onboard flash. -Click the image you want to use and it will download. The images are some 500M, so it might take a while. +Click the image you want to use and it will download. +The images are some 500M, so it might take a while. Discussion ************ @@ -370,9 +373,16 @@ If you boot the Bone with a microSD card inserted with a valid boot image, it will boot from the microSD card. If you boot without the microSD card installed, it will boot from the onboard flash. -.. tip:: If you want to reflash the onboard flash memory, see :ref:`basic onboard flash <basics_onboard_flash>`. +.. tip:: + If you want to reflash the onboard flash memory, + see :ref:`basic onboard flash <basics_onboard_flash>`. -.. note:: I instruct my students to use the microSD for booting. I suggest they keep an extra microSD flashed with the current OS. If they mess up the one on the Bone, it takes only a moment to swap in the extra microSD, boot up, and continue running. If they are running off the onboard flash, it will take much longer to reflash and boot from it. +.. note:: + I instruct my students to use the microSD for booting. I suggest they + keep an extra microSD flashed with the current OS. If they mess up the + one on the Bone, it takes only a moment to swap in the extra microSD, + boot up, and continue running. If they are running off the onboard flash, + it will take much longer to reflash and boot from it. Download the image you found in :ref:`basic find image <basics_find_image>`. It's more than 500 MB, so be sure to have a fast Internet connection. Then go to http://beagleboard.org/getting-started#update and diff --git a/bone-cook-book/01basics/figures/01GettingStarted.png b/beaglebone-cookbook/01basics/figures/01GettingStarted.png similarity index 100% rename from bone-cook-book/01basics/figures/01GettingStarted.png rename to beaglebone-cookbook/01basics/figures/01GettingStarted.png diff --git a/bone-cook-book/01basics/figures/02GettingStarted.png b/beaglebone-cookbook/01basics/figures/02GettingStarted.png similarity index 100% rename from bone-cook-book/01basics/figures/02GettingStarted.png rename to beaglebone-cookbook/01basics/figures/02GettingStarted.png diff --git a/bone-cook-book/01basics/figures/03GettingStarted.png b/beaglebone-cookbook/01basics/figures/03GettingStarted.png similarity index 100% rename from bone-cook-book/01basics/figures/03GettingStarted.png rename to beaglebone-cookbook/01basics/figures/03GettingStarted.png diff --git a/bone-cook-book/01basics/figures/04GettingStarted.png b/beaglebone-cookbook/01basics/figures/04GettingStarted.png similarity index 100% rename from bone-cook-book/01basics/figures/04GettingStarted.png rename to beaglebone-cookbook/01basics/figures/04GettingStarted.png diff --git a/bone-cook-book/01basics/figures/05GettingStarted.png b/beaglebone-cookbook/01basics/figures/05GettingStarted.png similarity index 100% rename from bone-cook-book/01basics/figures/05GettingStarted.png rename to beaglebone-cookbook/01basics/figures/05GettingStarted.png diff --git a/bone-cook-book/01basics/figures/05GettingStartedVScode.png b/beaglebone-cookbook/01basics/figures/05GettingStartedVScode.png similarity index 100% rename from bone-cook-book/01basics/figures/05GettingStartedVScode.png rename to beaglebone-cookbook/01basics/figures/05GettingStartedVScode.png diff --git a/bone-cook-book/01basics/figures/BeagleBoard_described.jpg b/beaglebone-cookbook/01basics/figures/BeagleBoard_described.jpg similarity index 100% rename from bone-cook-book/01basics/figures/BeagleBoard_described.jpg rename to beaglebone-cookbook/01basics/figures/BeagleBoard_described.jpg diff --git a/bone-cook-book/01basics/figures/BeagleBoard_xM.JPG b/beaglebone-cookbook/01basics/figures/BeagleBoard_xM.JPG similarity index 100% rename from bone-cook-book/01basics/figures/BeagleBoard_xM.JPG rename to beaglebone-cookbook/01basics/figures/BeagleBoard_xM.JPG diff --git a/bone-cook-book/01basics/figures/Beagle_Board_big.jpg b/beaglebone-cookbook/01basics/figures/Beagle_Board_big.jpg similarity index 100% rename from bone-cook-book/01basics/figures/Beagle_Board_big.jpg rename to beaglebone-cookbook/01basics/figures/Beagle_Board_big.jpg diff --git a/bone-cook-book/01basics/figures/BeaglexM_details.jpg b/beaglebone-cookbook/01basics/figures/BeaglexM_details.jpg similarity index 100% rename from bone-cook-book/01basics/figures/BeaglexM_details.jpg rename to beaglebone-cookbook/01basics/figures/BeaglexM_details.jpg diff --git a/bone-cook-book/01basics/figures/X15_Beta_Front.jpg b/beaglebone-cookbook/01basics/figures/X15_Beta_Front.jpg similarity index 100% rename from bone-cook-book/01basics/figures/X15_Beta_Front.jpg rename to beaglebone-cookbook/01basics/figures/X15_Beta_Front.jpg diff --git a/bone-cook-book/01basics/figures/beaglebone-white.jpg b/beaglebone-cookbook/01basics/figures/beaglebone-white.jpg similarity index 100% rename from bone-cook-book/01basics/figures/beaglebone-white.jpg rename to beaglebone-cookbook/01basics/figures/beaglebone-white.jpg diff --git a/bone-cook-book/01basics/figures/bone-black-details.jpg b/beaglebone-cookbook/01basics/figures/bone-black-details.jpg similarity index 100% rename from bone-cook-book/01basics/figures/bone-black-details.jpg rename to beaglebone-cookbook/01basics/figures/bone-black-details.jpg diff --git a/bone-cook-book/01basics/figures/bone-black.jpg b/beaglebone-cookbook/01basics/figures/bone-black.jpg similarity index 100% rename from bone-cook-book/01basics/figures/bone-black.jpg rename to beaglebone-cookbook/01basics/figures/bone-black.jpg diff --git a/bone-cook-book/01basics/figures/bone-white.jpg b/beaglebone-cookbook/01basics/figures/bone-white.jpg similarity index 100% rename from bone-cook-book/01basics/figures/bone-white.jpg rename to beaglebone-cookbook/01basics/figures/bone-white.jpg diff --git a/bone-cook-book/01basics/figures/cloud9.png b/beaglebone-cookbook/01basics/figures/cloud9.png similarity index 100% rename from bone-cook-book/01basics/figures/cloud9.png rename to beaglebone-cookbook/01basics/figures/cloud9.png diff --git a/bone-cook-book/01basics/figures/cloud9Debugger.png b/beaglebone-cookbook/01basics/figures/cloud9Debugger.png similarity index 100% rename from bone-cook-book/01basics/figures/cloud9Debugger.png rename to beaglebone-cookbook/01basics/figures/cloud9Debugger.png diff --git a/bone-cook-book/01basics/figures/cloud9withBash.png b/beaglebone-cookbook/01basics/figures/cloud9withBash.png similarity index 100% rename from bone-cook-book/01basics/figures/cloud9withBash.png rename to beaglebone-cookbook/01basics/figures/cloud9withBash.png diff --git a/bone-cook-book/01basics/figures/deb1.png b/beaglebone-cookbook/01basics/figures/deb1.png similarity index 100% rename from bone-cook-book/01basics/figures/deb1.png rename to beaglebone-cookbook/01basics/figures/deb1.png diff --git a/bone-cook-book/01basics/figures/deb2.png b/beaglebone-cookbook/01basics/figures/deb2.png similarity index 100% rename from bone-cook-book/01basics/figures/deb2.png rename to beaglebone-cookbook/01basics/figures/deb2.png diff --git a/bone-cook-book/01basics/figures/deb3.png b/beaglebone-cookbook/01basics/figures/deb3.png similarity index 100% rename from bone-cook-book/01basics/figures/deb3.png rename to beaglebone-cookbook/01basics/figures/deb3.png diff --git a/bone-cook-book/01basics/figures/forkMe.png b/beaglebone-cookbook/01basics/figures/forkMe.png similarity index 100% rename from bone-cook-book/01basics/figures/forkMe.png rename to beaglebone-cookbook/01basics/figures/forkMe.png diff --git a/bone-cook-book/01basics/figures/pluggingIn.jpg b/beaglebone-cookbook/01basics/figures/pluggingIn.jpg similarity index 100% rename from bone-cook-book/01basics/figures/pluggingIn.jpg rename to beaglebone-cookbook/01basics/figures/pluggingIn.jpg diff --git a/bone-cook-book/01basics/figures/template.fzz b/beaglebone-cookbook/01basics/figures/template.fzz similarity index 100% rename from bone-cook-book/01basics/figures/template.fzz rename to beaglebone-cookbook/01basics/figures/template.fzz diff --git a/bone-cook-book/01basics/figures/template_bb.png b/beaglebone-cookbook/01basics/figures/template_bb.png similarity index 100% rename from bone-cook-book/01basics/figures/template_bb.png rename to beaglebone-cookbook/01basics/figures/template_bb.png diff --git a/bone-cook-book/01basics/figures/vscBash.png b/beaglebone-cookbook/01basics/figures/vscBash.png similarity index 100% rename from bone-cook-book/01basics/figures/vscBash.png rename to beaglebone-cookbook/01basics/figures/vscBash.png diff --git a/bone-cook-book/02sensors/code/GPS.js b/beaglebone-cookbook/02sensors/code/GPS.js similarity index 100% rename from bone-cook-book/02sensors/code/GPS.js rename to beaglebone-cookbook/02sensors/code/GPS.js diff --git a/bone-cook-book/02sensors/code/analogIn.js b/beaglebone-cookbook/02sensors/code/analogIn.js similarity index 100% rename from bone-cook-book/02sensors/code/analogIn.js rename to beaglebone-cookbook/02sensors/code/analogIn.js diff --git a/bone-cook-book/02sensors/code/analogIn.py b/beaglebone-cookbook/02sensors/code/analogIn.py similarity index 100% rename from bone-cook-book/02sensors/code/analogIn.py rename to beaglebone-cookbook/02sensors/code/analogIn.py diff --git a/bone-cook-book/02sensors/code/audio.asoundrc b/beaglebone-cookbook/02sensors/code/audio.asoundrc similarity index 100% rename from bone-cook-book/02sensors/code/audio.asoundrc rename to beaglebone-cookbook/02sensors/code/audio.asoundrc diff --git a/bone-cook-book/02sensors/code/audio.js b/beaglebone-cookbook/02sensors/code/audio.js similarity index 100% rename from bone-cook-book/02sensors/code/audio.js rename to beaglebone-cookbook/02sensors/code/audio.js diff --git a/bone-cook-book/02sensors/code/bone_eqep2b.dts b/beaglebone-cookbook/02sensors/code/bone_eqep2b.dts similarity index 100% rename from bone-cook-book/02sensors/code/bone_eqep2b.dts rename to beaglebone-cookbook/02sensors/code/bone_eqep2b.dts diff --git a/bone-cook-book/02sensors/code/encoder.js b/beaglebone-cookbook/02sensors/code/encoder.js similarity index 100% rename from bone-cook-book/02sensors/code/encoder.js rename to beaglebone-cookbook/02sensors/code/encoder.js diff --git a/bone-cook-book/02sensors/code/gpiod/.gitignore b/beaglebone-cookbook/02sensors/code/gpiod/.gitignore similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/.gitignore rename to beaglebone-cookbook/02sensors/code/gpiod/.gitignore diff --git a/bone-cook-book/02sensors/code/gpiod/Makefile b/beaglebone-cookbook/02sensors/code/gpiod/Makefile similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/Makefile rename to beaglebone-cookbook/02sensors/code/gpiod/Makefile diff --git a/bone-cook-book/02sensors/code/gpiod/ReadMe.md b/beaglebone-cookbook/02sensors/code/gpiod/ReadMe.md similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/ReadMe.md rename to beaglebone-cookbook/02sensors/code/gpiod/ReadMe.md diff --git a/bone-cook-book/02sensors/code/gpiod/aggregatorSetup.sh b/beaglebone-cookbook/02sensors/code/gpiod/aggregatorSetup.sh similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/aggregatorSetup.sh rename to beaglebone-cookbook/02sensors/code/gpiod/aggregatorSetup.sh diff --git a/bone-cook-book/02sensors/code/gpiod/bulk_blink.py b/beaglebone-cookbook/02sensors/code/gpiod/bulk_blink.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/bulk_blink.py rename to beaglebone-cookbook/02sensors/code/gpiod/bulk_blink.py diff --git a/bone-cook-book/02sensors/code/gpiod/get.c b/beaglebone-cookbook/02sensors/code/gpiod/get.c similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/get.c rename to beaglebone-cookbook/02sensors/code/gpiod/get.c diff --git a/bone-cook-book/02sensors/code/gpiod/get.py b/beaglebone-cookbook/02sensors/code/gpiod/get.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/get.py rename to beaglebone-cookbook/02sensors/code/gpiod/get.py diff --git a/bone-cook-book/02sensors/code/gpiod/get.sh b/beaglebone-cookbook/02sensors/code/gpiod/get.sh similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/get.sh rename to beaglebone-cookbook/02sensors/code/gpiod/get.sh diff --git a/bone-cook-book/02sensors/code/gpiod/getset.c b/beaglebone-cookbook/02sensors/code/gpiod/getset.c similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/getset.c rename to beaglebone-cookbook/02sensors/code/gpiod/getset.c diff --git a/bone-cook-book/02sensors/code/gpiod/getset.py b/beaglebone-cookbook/02sensors/code/gpiod/getset.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/getset.py rename to beaglebone-cookbook/02sensors/code/gpiod/getset.py diff --git a/bone-cook-book/02sensors/code/gpiod/getsetEvent.c b/beaglebone-cookbook/02sensors/code/gpiod/getsetEvent.c similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/getsetEvent.c rename to beaglebone-cookbook/02sensors/code/gpiod/getsetEvent.c diff --git a/bone-cook-book/02sensors/code/gpiod/getsetEvent.py b/beaglebone-cookbook/02sensors/code/gpiod/getsetEvent.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/getsetEvent.py rename to beaglebone-cookbook/02sensors/code/gpiod/getsetEvent.py diff --git a/bone-cook-book/02sensors/code/gpiod/toggle1.c b/beaglebone-cookbook/02sensors/code/gpiod/toggle1.c similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggle1.c rename to beaglebone-cookbook/02sensors/code/gpiod/toggle1.c diff --git a/bone-cook-book/02sensors/code/gpiod/toggle1.py b/beaglebone-cookbook/02sensors/code/gpiod/toggle1.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggle1.py rename to beaglebone-cookbook/02sensors/code/gpiod/toggle1.py diff --git a/bone-cook-book/02sensors/code/gpiod/toggle1.sh b/beaglebone-cookbook/02sensors/code/gpiod/toggle1.sh similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggle1.sh rename to beaglebone-cookbook/02sensors/code/gpiod/toggle1.sh diff --git a/bone-cook-book/02sensors/code/gpiod/toggle2.c b/beaglebone-cookbook/02sensors/code/gpiod/toggle2.c similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggle2.c rename to beaglebone-cookbook/02sensors/code/gpiod/toggle2.c diff --git a/bone-cook-book/02sensors/code/gpiod/toggle2.py b/beaglebone-cookbook/02sensors/code/gpiod/toggle2.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggle2.py rename to beaglebone-cookbook/02sensors/code/gpiod/toggle2.py diff --git a/bone-cook-book/02sensors/code/gpiod/toggleLED.c b/beaglebone-cookbook/02sensors/code/gpiod/toggleLED.c similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggleLED.c rename to beaglebone-cookbook/02sensors/code/gpiod/toggleLED.c diff --git a/bone-cook-book/02sensors/code/gpiod/toggleLED.py b/beaglebone-cookbook/02sensors/code/gpiod/toggleLED.py similarity index 100% rename from bone-cook-book/02sensors/code/gpiod/toggleLED.py rename to beaglebone-cookbook/02sensors/code/gpiod/toggleLED.py diff --git a/bone-cook-book/02sensors/code/hc-sr04-ultraSonic.js b/beaglebone-cookbook/02sensors/code/hc-sr04-ultraSonic.js similarity index 100% rename from bone-cook-book/02sensors/code/hc-sr04-ultraSonic.js rename to beaglebone-cookbook/02sensors/code/hc-sr04-ultraSonic.js diff --git a/bone-cook-book/02sensors/code/i2c-scan.js b/beaglebone-cookbook/02sensors/code/i2c-scan.js similarity index 100% rename from bone-cook-book/02sensors/code/i2c-scan.js rename to beaglebone-cookbook/02sensors/code/i2c-scan.js diff --git a/bone-cook-book/02sensors/code/i2c-test.js b/beaglebone-cookbook/02sensors/code/i2c-test.js similarity index 100% rename from bone-cook-book/02sensors/code/i2c-test.js rename to beaglebone-cookbook/02sensors/code/i2c-test.js diff --git a/bone-cook-book/02sensors/code/i2cTemp.js b/beaglebone-cookbook/02sensors/code/i2cTemp.js similarity index 100% rename from bone-cook-book/02sensors/code/i2cTemp.js rename to beaglebone-cookbook/02sensors/code/i2cTemp.js diff --git a/bone-cook-book/02sensors/code/i2cTemp.py b/beaglebone-cookbook/02sensors/code/i2cTemp.py similarity index 100% rename from bone-cook-book/02sensors/code/i2cTemp.py rename to beaglebone-cookbook/02sensors/code/i2cTemp.py diff --git a/bone-cook-book/02sensors/code/i2ctmp101.py b/beaglebone-cookbook/02sensors/code/i2ctmp101.py similarity index 100% rename from bone-cook-book/02sensors/code/i2ctmp101.py rename to beaglebone-cookbook/02sensors/code/i2ctmp101.py diff --git a/bone-cook-book/02sensors/code/pushbutton.js b/beaglebone-cookbook/02sensors/code/pushbutton.js similarity index 100% rename from bone-cook-book/02sensors/code/pushbutton.js rename to beaglebone-cookbook/02sensors/code/pushbutton.js diff --git a/bone-cook-book/02sensors/code/pushbutton.py b/beaglebone-cookbook/02sensors/code/pushbutton.py similarity index 100% rename from bone-cook-book/02sensors/code/pushbutton.py rename to beaglebone-cookbook/02sensors/code/pushbutton.py diff --git a/bone-cook-book/02sensors/code/pushbutton2.js b/beaglebone-cookbook/02sensors/code/pushbutton2.js similarity index 100% rename from bone-cook-book/02sensors/code/pushbutton2.js rename to beaglebone-cookbook/02sensors/code/pushbutton2.js diff --git a/bone-cook-book/02sensors/code/pushbuttonPullup.js b/beaglebone-cookbook/02sensors/code/pushbuttonPullup.js similarity index 100% rename from bone-cook-book/02sensors/code/pushbuttonPullup.js rename to beaglebone-cookbook/02sensors/code/pushbuttonPullup.js diff --git a/bone-cook-book/02sensors/code/pushbutton_digitalRead.js b/beaglebone-cookbook/02sensors/code/pushbutton_digitalRead.js similarity index 100% rename from bone-cook-book/02sensors/code/pushbutton_digitalRead.js rename to beaglebone-cookbook/02sensors/code/pushbutton_digitalRead.js diff --git a/bone-cook-book/02sensors/code/rotaryEncoder.js b/beaglebone-cookbook/02sensors/code/rotaryEncoder.js similarity index 100% rename from bone-cook-book/02sensors/code/rotaryEncoder.js rename to beaglebone-cookbook/02sensors/code/rotaryEncoder.js diff --git a/bone-cook-book/02sensors/code/rotaryEncoder.py b/beaglebone-cookbook/02sensors/code/rotaryEncoder.py similarity index 100% rename from bone-cook-book/02sensors/code/rotaryEncoder.py rename to beaglebone-cookbook/02sensors/code/rotaryEncoder.py diff --git a/bone-cook-book/02sensors/code/sensorTag.js b/beaglebone-cookbook/02sensors/code/sensorTag.js similarity index 100% rename from bone-cook-book/02sensors/code/sensorTag.js rename to beaglebone-cookbook/02sensors/code/sensorTag.js diff --git a/bone-cook-book/02sensors/code/stop.js b/beaglebone-cookbook/02sensors/code/stop.js similarity index 100% rename from bone-cook-book/02sensors/code/stop.js rename to beaglebone-cookbook/02sensors/code/stop.js diff --git a/bone-cook-book/02sensors/code/testHC-SR04.js b/beaglebone-cookbook/02sensors/code/testHC-SR04.js similarity index 100% rename from bone-cook-book/02sensors/code/testHC-SR04.js rename to beaglebone-cookbook/02sensors/code/testHC-SR04.js diff --git a/bone-cook-book/02sensors/code/ultrasonicRange.js b/beaglebone-cookbook/02sensors/code/ultrasonicRange.js similarity index 100% rename from bone-cook-book/02sensors/code/ultrasonicRange.js rename to beaglebone-cookbook/02sensors/code/ultrasonicRange.js diff --git a/bone-cook-book/02sensors/code/ultrasonicRange.py b/beaglebone-cookbook/02sensors/code/ultrasonicRange.py similarity index 100% rename from bone-cook-book/02sensors/code/ultrasonicRange.py rename to beaglebone-cookbook/02sensors/code/ultrasonicRange.py diff --git a/bone-cook-book/02sensors/code/w1.js b/beaglebone-cookbook/02sensors/code/w1.js similarity index 100% rename from bone-cook-book/02sensors/code/w1.js rename to beaglebone-cookbook/02sensors/code/w1.js diff --git a/bone-cook-book/02sensors/code/w1.py b/beaglebone-cookbook/02sensors/code/w1.py similarity index 100% rename from bone-cook-book/02sensors/code/w1.py rename to beaglebone-cookbook/02sensors/code/w1.py diff --git a/bone-cook-book/02sensors/figures/Adafruit Xbee Adapter v1.1.fzpz b/beaglebone-cookbook/02sensors/figures/Adafruit Xbee Adapter v1.1.fzpz similarity index 100% rename from bone-cook-book/02sensors/figures/Adafruit Xbee Adapter v1.1.fzpz rename to beaglebone-cookbook/02sensors/figures/Adafruit Xbee Adapter v1.1.fzpz diff --git a/bone-cook-book/02sensors/figures/Adafruit Xbee Adapter v1.1.txt b/beaglebone-cookbook/02sensors/figures/Adafruit Xbee Adapter v1.1.txt similarity index 100% rename from bone-cook-book/02sensors/figures/Adafruit Xbee Adapter v1.1.txt rename to beaglebone-cookbook/02sensors/figures/Adafruit Xbee Adapter v1.1.txt diff --git a/bone-cook-book/02sensors/figures/GPS.fzz b/beaglebone-cookbook/02sensors/figures/GPS.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/GPS.fzz rename to beaglebone-cookbook/02sensors/figures/GPS.fzz diff --git a/bone-cook-book/02sensors/figures/GPS_bb.png b/beaglebone-cookbook/02sensors/figures/GPS_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/GPS_bb.png rename to beaglebone-cookbook/02sensors/figures/GPS_bb.png diff --git a/bone-cook-book/02sensors/figures/P8P9.fzz b/beaglebone-cookbook/02sensors/figures/P8P9.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/P8P9.fzz rename to beaglebone-cookbook/02sensors/figures/P8P9.fzz diff --git a/bone-cook-book/02sensors/figures/P8P9_bb.png b/beaglebone-cookbook/02sensors/figures/P8P9_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/P8P9_bb.png rename to beaglebone-cookbook/02sensors/figures/P8P9_bb.png diff --git a/bone-cook-book/02sensors/figures/analogIn.fzz b/beaglebone-cookbook/02sensors/figures/analogIn.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/analogIn.fzz rename to beaglebone-cookbook/02sensors/figures/analogIn.fzz diff --git a/bone-cook-book/02sensors/figures/analogIn_bb.png b/beaglebone-cookbook/02sensors/figures/analogIn_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/analogIn_bb.png rename to beaglebone-cookbook/02sensors/figures/analogIn_bb.png diff --git a/bone-cook-book/02sensors/figures/audioDongle.jpg b/beaglebone-cookbook/02sensors/figures/audioDongle.jpg similarity index 100% rename from bone-cook-book/02sensors/figures/audioDongle.jpg rename to beaglebone-cookbook/02sensors/figures/audioDongle.jpg diff --git a/bone-cook-book/02sensors/figures/audioDongle.txt b/beaglebone-cookbook/02sensors/figures/audioDongle.txt similarity index 100% rename from bone-cook-book/02sensors/figures/audioDongle.txt rename to beaglebone-cookbook/02sensors/figures/audioDongle.txt diff --git a/bone-cook-book/02sensors/figures/black_hardware_details.png b/beaglebone-cookbook/02sensors/figures/black_hardware_details.png similarity index 100% rename from bone-cook-book/02sensors/figures/black_hardware_details.png rename to beaglebone-cookbook/02sensors/figures/black_hardware_details.png diff --git a/bone-cook-book/02sensors/figures/cape-headers-analog.png b/beaglebone-cookbook/02sensors/figures/cape-headers-analog.png similarity index 100% rename from bone-cook-book/02sensors/figures/cape-headers-analog.png rename to beaglebone-cookbook/02sensors/figures/cape-headers-analog.png diff --git a/bone-cook-book/02sensors/figures/cape-headers-i2c.png b/beaglebone-cookbook/02sensors/figures/cape-headers-i2c.png similarity index 100% rename from bone-cook-book/02sensors/figures/cape-headers-i2c.png rename to beaglebone-cookbook/02sensors/figures/cape-headers-i2c.png diff --git a/bone-cook-book/02sensors/figures/cape-headers-serial.png b/beaglebone-cookbook/02sensors/figures/cape-headers-serial.png similarity index 100% rename from bone-cook-book/02sensors/figures/cape-headers-serial.png rename to beaglebone-cookbook/02sensors/figures/cape-headers-serial.png diff --git a/bone-cook-book/02sensors/figures/cape-headers.png b/beaglebone-cookbook/02sensors/figures/cape-headers.png similarity index 100% rename from bone-cook-book/02sensors/figures/cape-headers.png rename to beaglebone-cookbook/02sensors/figures/cape-headers.png diff --git a/bone-cook-book/02sensors/figures/flexResistor.fzz b/beaglebone-cookbook/02sensors/figures/flexResistor.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/flexResistor.fzz rename to beaglebone-cookbook/02sensors/figures/flexResistor.fzz diff --git a/bone-cook-book/02sensors/figures/flexResistor_bb.png b/beaglebone-cookbook/02sensors/figures/flexResistor_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/flexResistor_bb.png rename to beaglebone-cookbook/02sensors/figures/flexResistor_bb.png diff --git a/bone-cook-book/02sensors/figures/hc-sr04-ultraSonic.fzz b/beaglebone-cookbook/02sensors/figures/hc-sr04-ultraSonic.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/hc-sr04-ultraSonic.fzz rename to beaglebone-cookbook/02sensors/figures/hc-sr04-ultraSonic.fzz diff --git a/bone-cook-book/02sensors/figures/hc-sr04-ultraSonic_bb.png b/beaglebone-cookbook/02sensors/figures/hc-sr04-ultraSonic_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/hc-sr04-ultraSonic_bb.png rename to beaglebone-cookbook/02sensors/figures/hc-sr04-ultraSonic_bb.png diff --git a/bone-cook-book/02sensors/figures/hc-sr04.jpg b/beaglebone-cookbook/02sensors/figures/hc-sr04.jpg similarity index 100% rename from bone-cook-book/02sensors/figures/hc-sr04.jpg rename to beaglebone-cookbook/02sensors/figures/hc-sr04.jpg diff --git a/bone-cook-book/02sensors/figures/i2cTemp.fzz b/beaglebone-cookbook/02sensors/figures/i2cTemp.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/i2cTemp.fzz rename to beaglebone-cookbook/02sensors/figures/i2cTemp.fzz diff --git a/bone-cook-book/02sensors/figures/i2cTemp.js.old b/beaglebone-cookbook/02sensors/figures/i2cTemp.js.old similarity index 100% rename from bone-cook-book/02sensors/figures/i2cTemp.js.old rename to beaglebone-cookbook/02sensors/figures/i2cTemp.js.old diff --git a/bone-cook-book/02sensors/figures/i2cTemp_bb.png b/beaglebone-cookbook/02sensors/figures/i2cTemp_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/i2cTemp_bb.png rename to beaglebone-cookbook/02sensors/figures/i2cTemp_bb.png diff --git a/bone-cook-book/02sensors/figures/onewire.fzz b/beaglebone-cookbook/02sensors/figures/onewire.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/onewire.fzz rename to beaglebone-cookbook/02sensors/figures/onewire.fzz diff --git a/bone-cook-book/02sensors/figures/onewire_bb.png b/beaglebone-cookbook/02sensors/figures/onewire_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/onewire_bb.png rename to beaglebone-cookbook/02sensors/figures/onewire_bb.png diff --git a/bone-cook-book/02sensors/figures/onewire_setup.sh b/beaglebone-cookbook/02sensors/figures/onewire_setup.sh similarity index 100% rename from bone-cook-book/02sensors/figures/onewire_setup.sh rename to beaglebone-cookbook/02sensors/figures/onewire_setup.sh diff --git a/bone-cook-book/02sensors/figures/pot.fzz b/beaglebone-cookbook/02sensors/figures/pot.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/pot.fzz rename to beaglebone-cookbook/02sensors/figures/pot.fzz diff --git a/bone-cook-book/02sensors/figures/pot_schem.png b/beaglebone-cookbook/02sensors/figures/pot_schem.png similarity index 100% rename from bone-cook-book/02sensors/figures/pot_schem.png rename to beaglebone-cookbook/02sensors/figures/pot_schem.png diff --git a/bone-cook-book/02sensors/figures/pushbutton.fzz b/beaglebone-cookbook/02sensors/figures/pushbutton.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/pushbutton.fzz rename to beaglebone-cookbook/02sensors/figures/pushbutton.fzz diff --git a/bone-cook-book/02sensors/figures/pushbuttonPullup.fzz b/beaglebone-cookbook/02sensors/figures/pushbuttonPullup.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/pushbuttonPullup.fzz rename to beaglebone-cookbook/02sensors/figures/pushbuttonPullup.fzz diff --git a/bone-cook-book/02sensors/figures/pushbuttonPullup_bb.png b/beaglebone-cookbook/02sensors/figures/pushbuttonPullup_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/pushbuttonPullup_bb.png rename to beaglebone-cookbook/02sensors/figures/pushbuttonPullup_bb.png diff --git a/bone-cook-book/02sensors/figures/pushbutton_bb.png b/beaglebone-cookbook/02sensors/figures/pushbutton_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/pushbutton_bb.png rename to beaglebone-cookbook/02sensors/figures/pushbutton_bb.png diff --git a/bone-cook-book/02sensors/figures/rotaryEncoder.fzz b/beaglebone-cookbook/02sensors/figures/rotaryEncoder.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/rotaryEncoder.fzz rename to beaglebone-cookbook/02sensors/figures/rotaryEncoder.fzz diff --git a/bone-cook-book/02sensors/figures/rotaryEncoder.js.orig b/beaglebone-cookbook/02sensors/figures/rotaryEncoder.js.orig similarity index 100% rename from bone-cook-book/02sensors/figures/rotaryEncoder.js.orig rename to beaglebone-cookbook/02sensors/figures/rotaryEncoder.js.orig diff --git a/bone-cook-book/02sensors/figures/rotaryEncoder_bb.png b/beaglebone-cookbook/02sensors/figures/rotaryEncoder_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/rotaryEncoder_bb.png rename to beaglebone-cookbook/02sensors/figures/rotaryEncoder_bb.png diff --git a/bone-cook-book/02sensors/figures/sensorTag.jpg b/beaglebone-cookbook/02sensors/figures/sensorTag.jpg similarity index 100% rename from bone-cook-book/02sensors/figures/sensorTag.jpg rename to beaglebone-cookbook/02sensors/figures/sensorTag.jpg diff --git a/bone-cook-book/02sensors/figures/testSerial-2014-04-11.zip b/beaglebone-cookbook/02sensors/figures/testSerial-2014-04-11.zip similarity index 100% rename from bone-cook-book/02sensors/figures/testSerial-2014-04-11.zip rename to beaglebone-cookbook/02sensors/figures/testSerial-2014-04-11.zip diff --git a/bone-cook-book/02sensors/figures/ultrasonicRange.fzz b/beaglebone-cookbook/02sensors/figures/ultrasonicRange.fzz similarity index 100% rename from bone-cook-book/02sensors/figures/ultrasonicRange.fzz rename to beaglebone-cookbook/02sensors/figures/ultrasonicRange.fzz diff --git a/bone-cook-book/02sensors/figures/ultrasonicRange_bb.png b/beaglebone-cookbook/02sensors/figures/ultrasonicRange_bb.png similarity index 100% rename from bone-cook-book/02sensors/figures/ultrasonicRange_bb.png rename to beaglebone-cookbook/02sensors/figures/ultrasonicRange_bb.png diff --git a/bone-cook-book/02sensors/figures/vsc-bash-tab.png b/beaglebone-cookbook/02sensors/figures/vsc-bash-tab.png similarity index 100% rename from bone-cook-book/02sensors/figures/vsc-bash-tab.png rename to beaglebone-cookbook/02sensors/figures/vsc-bash-tab.png diff --git a/bone-cook-book/02sensors/sensors.rst b/beaglebone-cookbook/02sensors/sensors.rst similarity index 96% rename from bone-cook-book/02sensors/sensors.rst rename to beaglebone-cookbook/02sensors/sensors.rst index 5c3007acc48d1236d49e2f29b6ddaca05ed1d5ba..1a92078cb45a0772206330d45cb921ecff2c2ca5 100644 --- a/bone-cook-book/02sensors/sensors.rst +++ b/beaglebone-cookbook/02sensors/sensors.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-sensors: +.. _beaglebone-cookbook-sensors: Sensors ######## @@ -57,7 +57,8 @@ You want to acquire and attach a sensor and need to understand your basic option Solution ********* -:ref:`sensor cape headers<sensors_cape_headers>` shows many of the possibilities for connecting a sensor. +:ref:`sensor cape headers<sensors_cape_headers>` +shows many of the possibilities for connecting a sensor. .. _sensors_cape_headers: @@ -170,7 +171,8 @@ or both on the Bone, as shown in :ref:`figure below <js_pushbutton_fig>`. Bone with pushbutton -The code in :ref:`js pushbutton code<js_pushbutton_code>` reads GPIO port *P9_42*, which is attached to the pushbutton. +The code in :ref:`js pushbutton code<js_pushbutton_code>` +reads GPIO port *P9_42*, which is attached to the pushbutton. .. _py_pushbutton_code: @@ -528,6 +530,7 @@ Here, we'll use the *eQEP2* encoder via the Linux +count+ subsystem. Then run the following commands: + .. code-block:: bash bone$ config-pin P8_11 qep diff --git a/bone-cook-book/03displays/code/externLED.js b/beaglebone-cookbook/03displays/code/externLED.js similarity index 100% rename from bone-cook-book/03displays/code/externLED.js rename to beaglebone-cookbook/03displays/code/externLED.js diff --git a/bone-cook-book/03displays/code/externLED.py b/beaglebone-cookbook/03displays/code/externLED.py similarity index 100% rename from bone-cook-book/03displays/code/externLED.py rename to beaglebone-cookbook/03displays/code/externLED.py diff --git a/bone-cook-book/03displays/code/fadeLED.js b/beaglebone-cookbook/03displays/code/fadeLED.js similarity index 100% rename from bone-cook-book/03displays/code/fadeLED.js rename to beaglebone-cookbook/03displays/code/fadeLED.js diff --git a/bone-cook-book/03displays/code/fadeLED.py b/beaglebone-cookbook/03displays/code/fadeLED.py similarity index 100% rename from bone-cook-book/03displays/code/fadeLED.py rename to beaglebone-cookbook/03displays/code/fadeLED.py diff --git a/bone-cook-book/03displays/code/internLED.js b/beaglebone-cookbook/03displays/code/internLED.js similarity index 100% rename from bone-cook-book/03displays/code/internLED.js rename to beaglebone-cookbook/03displays/code/internLED.js diff --git a/bone-cook-book/03displays/code/internLED.py b/beaglebone-cookbook/03displays/code/internLED.py similarity index 100% rename from bone-cook-book/03displays/code/internLED.py rename to beaglebone-cookbook/03displays/code/internLED.py diff --git a/bone-cook-book/03displays/code/matrixLEDi2c.js b/beaglebone-cookbook/03displays/code/matrixLEDi2c.js similarity index 100% rename from bone-cook-book/03displays/code/matrixLEDi2c.js rename to beaglebone-cookbook/03displays/code/matrixLEDi2c.js diff --git a/bone-cook-book/03displays/code/matrixLEDi2c.py b/beaglebone-cookbook/03displays/code/matrixLEDi2c.py similarity index 100% rename from bone-cook-book/03displays/code/matrixLEDi2c.py rename to beaglebone-cookbook/03displays/code/matrixLEDi2c.py diff --git a/bone-cook-book/03displays/code/neoPixel.sh b/beaglebone-cookbook/03displays/code/neoPixel.sh similarity index 100% rename from bone-cook-book/03displays/code/neoPixel.sh rename to beaglebone-cookbook/03displays/code/neoPixel.sh diff --git a/bone-cook-book/03displays/code/nokia5110.js b/beaglebone-cookbook/03displays/code/nokia5110.js similarity index 100% rename from bone-cook-book/03displays/code/nokia5110.js rename to beaglebone-cookbook/03displays/code/nokia5110.js diff --git a/bone-cook-book/03displays/code/nokia5110Test.js b/beaglebone-cookbook/03displays/code/nokia5110Test.js similarity index 100% rename from bone-cook-book/03displays/code/nokia5110Test.js rename to beaglebone-cookbook/03displays/code/nokia5110Test.js diff --git a/bone-cook-book/03displays/code/pwmTest.sh b/beaglebone-cookbook/03displays/code/pwmTest.sh similarity index 100% rename from bone-cook-book/03displays/code/pwmTest.sh rename to beaglebone-cookbook/03displays/code/pwmTest.sh diff --git a/bone-cook-book/03displays/code/speak.js b/beaglebone-cookbook/03displays/code/speak.js similarity index 100% rename from bone-cook-book/03displays/code/speak.js rename to beaglebone-cookbook/03displays/code/speak.js diff --git a/bone-cook-book/03displays/displays.rst b/beaglebone-cookbook/03displays/displays.rst similarity index 84% rename from bone-cook-book/03displays/displays.rst rename to beaglebone-cookbook/03displays/displays.rst index 1078ffb476c72eb6333e781b2298982d62af73c3..c796ba35bbdd089bf0de0f0f125d2ac637d35321 100644 --- a/bone-cook-book/03displays/displays.rst +++ b/beaglebone-cookbook/03displays/displays.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-displays: +.. _beaglebone-cookbook-displays: Displays and Other Outputs ########################### @@ -8,7 +8,8 @@ Introduction In this chapter, you will learn how to control physical hardware via BeagleBone Black's general-purpose input/output (GPIO) pins. The Bone has -65 GPIO pins that are brought out on two 46-pin headers, called +P8+ and +P9+, as shown in :ref:`<js_P8P9_fig>>. +65 GPIO pins that are brought out on two 46-pin headers, called ++P8+ and +P9+, as shown in :ref:`<js_P8P9_fig>>. .. note:: All the examples in the book assume you have cloned the @@ -42,6 +43,7 @@ You want to know how to flash the four LEDs that are next to the Ethernet port o Solution ************* + Locate the four onboard LEDs shown in :ref:`<js_internLED_fig>>. They are labeled +USR0+ through +USR3+, but we'll refer to them as the +USER+ LEDs. @@ -80,8 +82,9 @@ Using an internal LED (internLED.js) In the +bash+ command window, enter the following commands: .. code-block:: bash + bone$ cd ~/BoneCookbook/docs/03displays/code -bone$ ./internLED.js + bone$ ./internLED.js The +USER0+ LED should now be flashing. @@ -108,10 +111,17 @@ Connect an LED to one of the GPIO pins using a series resistor to limit the curr * 220 Ω to 470 Ω resistor (see :ref:`app resistor <app_resistor>`) * LED (see :ref:`app opto <app_opto>`) -.. WARNING:: The value of the current limiting resistor depends on the LED you are using. The Bone can drive only 4 to 6 mA, so you might need a larger resistor to keep from pulling too much current. A 330 Ω or 470 Ω resistor might be better. +.. WARNING:: + The value of the current limiting resistor depends on the LED you are using. + The Bone can drive only 4 to 6 mA, so you might need a larger resistor to keep + from pulling too much current. A 330 Ω or 470 Ω resistor might be better. -:ref:`<displays_externLED_fig>` shows how you can wire the LED to pin 14 of the +P9+ header (+P9_14+). Every circuit in this book (:ref:`<basics_wire_breadboard>>) assumes you have already wired the rightmost bus to ground (+P9_1+) and the next bus to the left to the 3.3 V (+P9_3+) pins on the header. Be sure to get the polarity right on the LED. The _short_ lead always goes to ground. +:ref:`<displays_externLED_fig>` shows how you can wire the LED to pin 14 of +the +P9+ header (+P9_14+). Every circuit in this book (:ref:`<basics_wire_breadboard>>) +assumes you have already wired the rightmost bus to ground (+P9_1+) and the next bus to +the left to the 3.3 V (+P9_3+) pins on the header. Be sure to get the polarity right on +the LED. The _short_ lead always goes to ground. .. _displays_externLED_fig: @@ -121,7 +131,8 @@ Diagram for using an external LED :align: center :alt: External LED -After you've wired it, start VSC (see :ref:`basic vsc<basics_vsc>`) and find the code shown in :ref:`<py_externLED_code>>. +After you've wired it, start VSC (see :ref:`basic vsc<basics_vsc>`) +and find the code shown in :ref:`<py_externLED_code>>. .. _py_externLED_code: @@ -159,13 +170,15 @@ You want to control a device that runs at 120 V. Solution ************* -Working with 120 V can be tricky--even dangerous--if you aren't careful. Here's a safe way to do it. +Working with 120 V can be tricky--even dangerous--if +you aren't careful. Here's a safe way to do it. To make this recipe, you will need: * PowerSwitch Tail II (see :ref:`<app_misc>>) -:ref:`<displays_powerSwitch_fig>` shows how you can wire the PowerSwitch Tail II to pin +P9_14+. +:ref:`<displays_powerSwitch_fig>` shows how you can wire +the PowerSwitch Tail II to pin +P9_14+. .. _displays_powerSwitch_fig: @@ -221,6 +234,7 @@ Code for using an external LED (fadeLED.js) Discussion ************* + The Bone has several outputs that can be use as pwm's as shown in :ref:`<cape-headers-pwm_fig>>. There are three +EHRPWM+'s which each has a pair of pwm channels. Each pair must have the same period. @@ -237,6 +251,7 @@ The pwm's are accessed through +/dev/bone/pwm+ .. todo:: Should this be /dev/bone/pwm? .. code-block:: bash + bone$ cd /dev/bone/pwm bone$ ls 0 1 2 @@ -244,6 +259,7 @@ The pwm's are accessed through +/dev/bone/pwm+ Here we see six pwmchips that can be used, each has two channels. Explore one. .. code-block:: bash + bone$ cd 1 bone$ ls a b @@ -255,13 +271,15 @@ Here we see six pwmchips that can be used, each has two channels. Explore one. Attach in LED to P9_14 and if you set the period long enough you can see the LED flash. .. code-block:: bash + bone$ echo 1000000000 > period bone$ echo 500000000 > duty_cycle bone$ echo 1 > enable Your LED should now be flashing. -:ref:`<display_pwm_mapping>` are the mapping I've figured out so far. I don't know how to get to the timers. +:ref:`<display_pwm_mapping>` are the mapping I've figured out +so far. I don't know how to get to the timers. .. _display_pwm_mapping: @@ -296,6 +314,7 @@ You have an I^2^C-based LED matrix to interface. Solution ************* + There are a number of nice LED matrices that allow you to control several LEDs via one interface. This solution uses an `Adafruit Bicolor 8x8 LED Square Pixel Matrix w/I^2^C Backpack <http://www.adafruit.com/products/902>`_. @@ -336,11 +355,13 @@ Using I^2^C command-line tools to discover the address of the display 60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 70: 70 -- -- -- -- -- -- -- -Here, you can see a device at +0x49+ and +0x70+. I know I have a temperature sensor at +0x49+, so the LED matrix must be at +0.70+. +Here, you can see a device at +0x49+ and +0x70+. I know I have a temperature +sensor at +0x49+, so the LED matrix must be at +0.70+. Find the code in :ref:`<displays_matrix_i2c>` and run it by using the following command: .. code-block:: bash + bone$ pip install smbus # (Do this only once.) bone$ ./matrixLEDi2c.py @@ -379,16 +400,19 @@ You have a 5 V device to drive, and the Bone has 3.3 V outputs. Solution ************* -If you are lucky, you might be able to drive a 5 V device from the Bone's 3.3 V output. Try it and see if it works. If not, you need a level translator. +If you are lucky, you might be able to drive a 5 V device from the Bone's 3.3 V output. +Try it and see if it works. If not, you need a level translator. What you will need for this recipe: * A PCA9306 level translator (see :ref:`app ic<app_ic>`) * A 5 V power supply (if the Bone's 5 V power supply isn't enough) -The PCA9306 translates signals at 3.3 V to 5 V in both directions. It's meant to work with I^2^C devices that have a pull-up resistor, but it can work with anything needing translation. +The PCA9306 translates signals at 3.3 V to 5 V in both directions. It's meant to work with +I^2^C devices that have a pull-up resistor, but it can work with anything needing translation. -:ref:`<displays_i2cMatrixLevelTrans_fig>` shows how to wire a PCA9306 to an LED matrix. The left is the 3.3 V side and the right is the 5 V side. Notice that we are using the Bone's built-in 5 V power supply. +:ref:`<displays_i2cMatrixLevelTrans_fig>` shows how to wire a PCA9306 to an LED matrix. +The left is the 3.3 V side and the right is the 5 V side. Notice that we are using the Bone's built-in 5 V power supply. .. _displays_i2cMatrixLevelTrans_fig: @@ -410,12 +434,14 @@ Writing to a NeoPixel LED String Using the PRUs Problem ************* -You have an :ref:`Adafruit NeoPixel LED string <http://www.adafruit.com/products/1138>`_ or `Adafruit NeoPixel LED matrix <http://www.adafruit.com/products/1487>`_ and want to light it up. +You have an :ref:`Adafruit NeoPixel LED string <http://www.adafruit.com/products/1138>`_ or +`Adafruit NeoPixel LED matrix <http://www.adafruit.com/products/1487>`_ and want to light it up. Solution ************* -The PRU Cookbook has a nice discussion (https://markayoder.github.io/PRUCookbook/05blocks/blocks.html#blocks_ws2812[WS2812 (NeoPixel) driver]) on driving NeoPixels. +The PRU Cookbook has a nice discussion +(https://markayoder.github.io/PRUCookbook/05blocks/blocks.html#blocks_ws2812[WS2812 (NeoPixel) driver]) on driving NeoPixels. .. _py_neoPixelMatrix_fig: @@ -474,6 +500,7 @@ Solution Just install the _flite_ text-to-speech program: .. code-block:: bash + bone$ sudo apt install flite diff --git a/bone-cook-book/03displays/figures/LEDstring.fzz b/beaglebone-cookbook/03displays/figures/LEDstring.fzz similarity index 100% rename from bone-cook-book/03displays/figures/LEDstring.fzz rename to beaglebone-cookbook/03displays/figures/LEDstring.fzz diff --git a/bone-cook-book/03displays/figures/LEDstring_bb.png b/beaglebone-cookbook/03displays/figures/LEDstring_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/LEDstring_bb.png rename to beaglebone-cookbook/03displays/figures/LEDstring_bb.png diff --git a/bone-cook-book/03displays/figures/P8P9.fzz b/beaglebone-cookbook/03displays/figures/P8P9.fzz similarity index 100% rename from bone-cook-book/03displays/figures/P8P9.fzz rename to beaglebone-cookbook/03displays/figures/P8P9.fzz diff --git a/bone-cook-book/03displays/figures/P8P9_bb.png b/beaglebone-cookbook/03displays/figures/P8P9_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/P8P9_bb.png rename to beaglebone-cookbook/03displays/figures/P8P9_bb.png diff --git a/bone-cook-book/03displays/figures/cape-headers-pwm.png b/beaglebone-cookbook/03displays/figures/cape-headers-pwm.png similarity index 100% rename from bone-cook-book/03displays/figures/cape-headers-pwm.png rename to beaglebone-cookbook/03displays/figures/cape-headers-pwm.png diff --git a/bone-cook-book/03displays/figures/externLED.fzz b/beaglebone-cookbook/03displays/figures/externLED.fzz similarity index 100% rename from bone-cook-book/03displays/figures/externLED.fzz rename to beaglebone-cookbook/03displays/figures/externLED.fzz diff --git a/bone-cook-book/03displays/figures/externLED_bb.png b/beaglebone-cookbook/03displays/figures/externLED_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/externLED_bb.png rename to beaglebone-cookbook/03displays/figures/externLED_bb.png diff --git a/bone-cook-book/03displays/figures/i2cMatrix.fzz b/beaglebone-cookbook/03displays/figures/i2cMatrix.fzz similarity index 100% rename from bone-cook-book/03displays/figures/i2cMatrix.fzz rename to beaglebone-cookbook/03displays/figures/i2cMatrix.fzz diff --git a/bone-cook-book/03displays/figures/i2cMatrixLevelTrans.fzz b/beaglebone-cookbook/03displays/figures/i2cMatrixLevelTrans.fzz similarity index 100% rename from bone-cook-book/03displays/figures/i2cMatrixLevelTrans.fzz rename to beaglebone-cookbook/03displays/figures/i2cMatrixLevelTrans.fzz diff --git a/bone-cook-book/03displays/figures/i2cMatrixLevelTrans_bb.png b/beaglebone-cookbook/03displays/figures/i2cMatrixLevelTrans_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/i2cMatrixLevelTrans_bb.png rename to beaglebone-cookbook/03displays/figures/i2cMatrixLevelTrans_bb.png diff --git a/bone-cook-book/03displays/figures/i2cMatrixOLD.fzz b/beaglebone-cookbook/03displays/figures/i2cMatrixOLD.fzz similarity index 100% rename from bone-cook-book/03displays/figures/i2cMatrixOLD.fzz rename to beaglebone-cookbook/03displays/figures/i2cMatrixOLD.fzz diff --git a/bone-cook-book/03displays/figures/i2cMatrix_bb.png b/beaglebone-cookbook/03displays/figures/i2cMatrix_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/i2cMatrix_bb.png rename to beaglebone-cookbook/03displays/figures/i2cMatrix_bb.png diff --git a/bone-cook-book/03displays/figures/internLED.png b/beaglebone-cookbook/03displays/figures/internLED.png similarity index 100% rename from bone-cook-book/03displays/figures/internLED.png rename to beaglebone-cookbook/03displays/figures/internLED.png diff --git a/bone-cook-book/03displays/figures/matrix16x24.fzz b/beaglebone-cookbook/03displays/figures/matrix16x24.fzz similarity index 100% rename from bone-cook-book/03displays/figures/matrix16x24.fzz rename to beaglebone-cookbook/03displays/figures/matrix16x24.fzz diff --git a/bone-cook-book/03displays/figures/matrix16x24_bb.png b/beaglebone-cookbook/03displays/figures/matrix16x24_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/matrix16x24_bb.png rename to beaglebone-cookbook/03displays/figures/matrix16x24_bb.png diff --git a/bone-cook-book/03displays/figures/neo.fzz b/beaglebone-cookbook/03displays/figures/neo.fzz similarity index 100% rename from bone-cook-book/03displays/figures/neo.fzz rename to beaglebone-cookbook/03displays/figures/neo.fzz diff --git a/bone-cook-book/03displays/figures/neoPixel.fzz b/beaglebone-cookbook/03displays/figures/neoPixel.fzz similarity index 100% rename from bone-cook-book/03displays/figures/neoPixel.fzz rename to beaglebone-cookbook/03displays/figures/neoPixel.fzz diff --git a/bone-cook-book/03displays/figures/neoPixelMatrix.fzz b/beaglebone-cookbook/03displays/figures/neoPixelMatrix.fzz similarity index 100% rename from bone-cook-book/03displays/figures/neoPixelMatrix.fzz rename to beaglebone-cookbook/03displays/figures/neoPixelMatrix.fzz diff --git a/bone-cook-book/03displays/figures/neoPixelMatrix_bb.png b/beaglebone-cookbook/03displays/figures/neoPixelMatrix_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/neoPixelMatrix_bb.png rename to beaglebone-cookbook/03displays/figures/neoPixelMatrix_bb.png diff --git a/bone-cook-book/03displays/figures/neoPixel_bb.png b/beaglebone-cookbook/03displays/figures/neoPixel_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/neoPixel_bb.png rename to beaglebone-cookbook/03displays/figures/neoPixel_bb.png diff --git a/bone-cook-book/03displays/figures/neo_bb.png b/beaglebone-cookbook/03displays/figures/neo_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/neo_bb.png rename to beaglebone-cookbook/03displays/figures/neo_bb.png diff --git a/bone-cook-book/03displays/figures/nokia5110.fzz b/beaglebone-cookbook/03displays/figures/nokia5110.fzz similarity index 100% rename from bone-cook-book/03displays/figures/nokia5110.fzz rename to beaglebone-cookbook/03displays/figures/nokia5110.fzz diff --git a/bone-cook-book/03displays/figures/nokia5110_bb.png b/beaglebone-cookbook/03displays/figures/nokia5110_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/nokia5110_bb.png rename to beaglebone-cookbook/03displays/figures/nokia5110_bb.png diff --git a/bone-cook-book/03displays/figures/old/USERLEDs.png b/beaglebone-cookbook/03displays/figures/old/USERLEDs.png similarity index 100% rename from bone-cook-book/03displays/figures/old/USERLEDs.png rename to beaglebone-cookbook/03displays/figures/old/USERLEDs.png diff --git a/bone-cook-book/03displays/figures/old/externalLED.png b/beaglebone-cookbook/03displays/figures/old/externalLED.png similarity index 100% rename from bone-cook-book/03displays/figures/old/externalLED.png rename to beaglebone-cookbook/03displays/figures/old/externalLED.png diff --git a/bone-cook-book/03displays/figures/old/p8p9headers.png b/beaglebone-cookbook/03displays/figures/old/p8p9headers.png similarity index 100% rename from bone-cook-book/03displays/figures/old/p8p9headers.png rename to beaglebone-cookbook/03displays/figures/old/p8p9headers.png diff --git a/bone-cook-book/03displays/figures/powerSwitch.fzz b/beaglebone-cookbook/03displays/figures/powerSwitch.fzz similarity index 100% rename from bone-cook-book/03displays/figures/powerSwitch.fzz rename to beaglebone-cookbook/03displays/figures/powerSwitch.fzz diff --git a/bone-cook-book/03displays/figures/powerSwitch_bb.png b/beaglebone-cookbook/03displays/figures/powerSwitch_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/powerSwitch_bb.png rename to beaglebone-cookbook/03displays/figures/powerSwitch_bb.png diff --git a/bone-cook-book/03displays/figures/ws2801.fzz b/beaglebone-cookbook/03displays/figures/ws2801.fzz similarity index 100% rename from bone-cook-book/03displays/figures/ws2801.fzz rename to beaglebone-cookbook/03displays/figures/ws2801.fzz diff --git a/bone-cook-book/03displays/figures/ws2801_bb.png b/beaglebone-cookbook/03displays/figures/ws2801_bb.png similarity index 100% rename from bone-cook-book/03displays/figures/ws2801_bb.png rename to beaglebone-cookbook/03displays/figures/ws2801_bb.png diff --git a/bone-cook-book/04motors/code/bipolarStepperMotor.py b/beaglebone-cookbook/04motors/code/bipolarStepperMotor.py similarity index 100% rename from bone-cook-book/04motors/code/bipolarStepperMotor.py rename to beaglebone-cookbook/04motors/code/bipolarStepperMotor.py diff --git a/bone-cook-book/04motors/code/dcMotor.js b/beaglebone-cookbook/04motors/code/dcMotor.js similarity index 100% rename from bone-cook-book/04motors/code/dcMotor.js rename to beaglebone-cookbook/04motors/code/dcMotor.js diff --git a/bone-cook-book/04motors/code/dcMotor.py b/beaglebone-cookbook/04motors/code/dcMotor.py similarity index 100% rename from bone-cook-book/04motors/code/dcMotor.py rename to beaglebone-cookbook/04motors/code/dcMotor.py diff --git a/bone-cook-book/04motors/code/h-bridgeMotor.js b/beaglebone-cookbook/04motors/code/h-bridgeMotor.js similarity index 100% rename from bone-cook-book/04motors/code/h-bridgeMotor.js rename to beaglebone-cookbook/04motors/code/h-bridgeMotor.js diff --git a/bone-cook-book/04motors/code/ring.js b/beaglebone-cookbook/04motors/code/ring.js similarity index 100% rename from bone-cook-book/04motors/code/ring.js rename to beaglebone-cookbook/04motors/code/ring.js diff --git a/bone-cook-book/04motors/code/servoBird.js b/beaglebone-cookbook/04motors/code/servoBird.js similarity index 100% rename from bone-cook-book/04motors/code/servoBird.js rename to beaglebone-cookbook/04motors/code/servoBird.js diff --git a/bone-cook-book/04motors/code/servoEncoder.py b/beaglebone-cookbook/04motors/code/servoEncoder.py similarity index 100% rename from bone-cook-book/04motors/code/servoEncoder.py rename to beaglebone-cookbook/04motors/code/servoEncoder.py diff --git a/bone-cook-book/04motors/code/servoMotor.js b/beaglebone-cookbook/04motors/code/servoMotor.js similarity index 100% rename from bone-cook-book/04motors/code/servoMotor.js rename to beaglebone-cookbook/04motors/code/servoMotor.js diff --git a/bone-cook-book/04motors/code/servoMotor.py b/beaglebone-cookbook/04motors/code/servoMotor.py similarity index 100% rename from bone-cook-book/04motors/code/servoMotor.py rename to beaglebone-cookbook/04motors/code/servoMotor.py diff --git a/bone-cook-book/04motors/code/servoSense.js b/beaglebone-cookbook/04motors/code/servoSense.js similarity index 100% rename from bone-cook-book/04motors/code/servoSense.js rename to beaglebone-cookbook/04motors/code/servoSense.js diff --git a/bone-cook-book/04motors/code/stop.js b/beaglebone-cookbook/04motors/code/stop.js similarity index 100% rename from bone-cook-book/04motors/code/stop.js rename to beaglebone-cookbook/04motors/code/stop.js diff --git a/bone-cook-book/04motors/code/unipolarStepperMotor.js b/beaglebone-cookbook/04motors/code/unipolarStepperMotor.js similarity index 100% rename from bone-cook-book/04motors/code/unipolarStepperMotor.js rename to beaglebone-cookbook/04motors/code/unipolarStepperMotor.js diff --git a/bone-cook-book/04motors/code/unipolarStepperMotor.js.diff b/beaglebone-cookbook/04motors/code/unipolarStepperMotor.js.diff similarity index 100% rename from bone-cook-book/04motors/code/unipolarStepperMotor.js.diff rename to beaglebone-cookbook/04motors/code/unipolarStepperMotor.js.diff diff --git a/bone-cook-book/04motors/code/unipolarStepperMotor.py b/beaglebone-cookbook/04motors/code/unipolarStepperMotor.py similarity index 100% rename from bone-cook-book/04motors/code/unipolarStepperMotor.py rename to beaglebone-cookbook/04motors/code/unipolarStepperMotor.py diff --git a/bone-cook-book/04motors/code/unipolarStepperMotor.py.diff b/beaglebone-cookbook/04motors/code/unipolarStepperMotor.py.diff similarity index 100% rename from bone-cook-book/04motors/code/unipolarStepperMotor.py.diff rename to beaglebone-cookbook/04motors/code/unipolarStepperMotor.py.diff diff --git a/bone-cook-book/04motors/figures/H-bridge.ms13 b/beaglebone-cookbook/04motors/figures/H-bridge.ms13 similarity index 100% rename from bone-cook-book/04motors/figures/H-bridge.ms13 rename to beaglebone-cookbook/04motors/figures/H-bridge.ms13 diff --git a/bone-cook-book/04motors/figures/H-bridge.png b/beaglebone-cookbook/04motors/figures/H-bridge.png similarity index 100% rename from bone-cook-book/04motors/figures/H-bridge.png rename to beaglebone-cookbook/04motors/figures/H-bridge.png diff --git a/bone-cook-book/04motors/figures/bipolarStepperMotor.fzz b/beaglebone-cookbook/04motors/figures/bipolarStepperMotor.fzz similarity index 100% rename from bone-cook-book/04motors/figures/bipolarStepperMotor.fzz rename to beaglebone-cookbook/04motors/figures/bipolarStepperMotor.fzz diff --git a/bone-cook-book/04motors/figures/bipolarStepperMotor_bb.png b/beaglebone-cookbook/04motors/figures/bipolarStepperMotor_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/bipolarStepperMotor_bb.png rename to beaglebone-cookbook/04motors/figures/bipolarStepperMotor_bb.png diff --git a/bone-cook-book/04motors/figures/dcMotor.fzz b/beaglebone-cookbook/04motors/figures/dcMotor.fzz similarity index 100% rename from bone-cook-book/04motors/figures/dcMotor.fzz rename to beaglebone-cookbook/04motors/figures/dcMotor.fzz diff --git a/bone-cook-book/04motors/figures/dcMotor_bb.png b/beaglebone-cookbook/04motors/figures/dcMotor_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/dcMotor_bb.png rename to beaglebone-cookbook/04motors/figures/dcMotor_bb.png diff --git a/bone-cook-book/04motors/figures/h-bridgeMotor.fzz b/beaglebone-cookbook/04motors/figures/h-bridgeMotor.fzz similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotor.fzz rename to beaglebone-cookbook/04motors/figures/h-bridgeMotor.fzz diff --git a/bone-cook-book/04motors/figures/h-bridgeMotor5V.fzz b/beaglebone-cookbook/04motors/figures/h-bridgeMotor5V.fzz similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotor5V.fzz rename to beaglebone-cookbook/04motors/figures/h-bridgeMotor5V.fzz diff --git a/bone-cook-book/04motors/figures/h-bridgeMotor5V_bb.png b/beaglebone-cookbook/04motors/figures/h-bridgeMotor5V_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotor5V_bb.png rename to beaglebone-cookbook/04motors/figures/h-bridgeMotor5V_bb.png diff --git a/bone-cook-book/04motors/figures/h-bridgeMotorExternal.fzz b/beaglebone-cookbook/04motors/figures/h-bridgeMotorExternal.fzz similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotorExternal.fzz rename to beaglebone-cookbook/04motors/figures/h-bridgeMotorExternal.fzz diff --git a/bone-cook-book/04motors/figures/h-bridgeMotorExternal_bb.png b/beaglebone-cookbook/04motors/figures/h-bridgeMotorExternal_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotorExternal_bb.png rename to beaglebone-cookbook/04motors/figures/h-bridgeMotorExternal_bb.png diff --git a/bone-cook-book/04motors/figures/h-bridgeMotorOLD.fzz b/beaglebone-cookbook/04motors/figures/h-bridgeMotorOLD.fzz similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotorOLD.fzz rename to beaglebone-cookbook/04motors/figures/h-bridgeMotorOLD.fzz diff --git a/bone-cook-book/04motors/figures/h-bridgeMotor_bb.png b/beaglebone-cookbook/04motors/figures/h-bridgeMotor_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/h-bridgeMotor_bb.png rename to beaglebone-cookbook/04motors/figures/h-bridgeMotor_bb.png diff --git a/bone-cook-book/04motors/figures/servoMotor.fzz b/beaglebone-cookbook/04motors/figures/servoMotor.fzz similarity index 100% rename from bone-cook-book/04motors/figures/servoMotor.fzz rename to beaglebone-cookbook/04motors/figures/servoMotor.fzz diff --git a/bone-cook-book/04motors/figures/servoMotor5V.fzz b/beaglebone-cookbook/04motors/figures/servoMotor5V.fzz similarity index 100% rename from bone-cook-book/04motors/figures/servoMotor5V.fzz rename to beaglebone-cookbook/04motors/figures/servoMotor5V.fzz diff --git a/bone-cook-book/04motors/figures/servoMotor5V_bb.png b/beaglebone-cookbook/04motors/figures/servoMotor5V_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/servoMotor5V_bb.png rename to beaglebone-cookbook/04motors/figures/servoMotor5V_bb.png diff --git a/bone-cook-book/04motors/figures/servoMotor_bb.png b/beaglebone-cookbook/04motors/figures/servoMotor_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/servoMotor_bb.png rename to beaglebone-cookbook/04motors/figures/servoMotor_bb.png diff --git a/bone-cook-book/04motors/figures/stepExternalSupply.fzz b/beaglebone-cookbook/04motors/figures/stepExternalSupply.fzz similarity index 100% rename from bone-cook-book/04motors/figures/stepExternalSupply.fzz rename to beaglebone-cookbook/04motors/figures/stepExternalSupply.fzz diff --git a/bone-cook-book/04motors/figures/stepExternalSupply_bb.png b/beaglebone-cookbook/04motors/figures/stepExternalSupply_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/stepExternalSupply_bb.png rename to beaglebone-cookbook/04motors/figures/stepExternalSupply_bb.png diff --git a/bone-cook-book/04motors/figures/unipolarStepperMotor.fzz b/beaglebone-cookbook/04motors/figures/unipolarStepperMotor.fzz similarity index 100% rename from bone-cook-book/04motors/figures/unipolarStepperMotor.fzz rename to beaglebone-cookbook/04motors/figures/unipolarStepperMotor.fzz diff --git a/bone-cook-book/04motors/figures/unipolarStepperMotor_bb.png b/beaglebone-cookbook/04motors/figures/unipolarStepperMotor_bb.png similarity index 100% rename from bone-cook-book/04motors/figures/unipolarStepperMotor_bb.png rename to beaglebone-cookbook/04motors/figures/unipolarStepperMotor_bb.png diff --git a/bone-cook-book/04motors/motors.rst b/beaglebone-cookbook/04motors/motors.rst similarity index 80% rename from bone-cook-book/04motors/motors.rst rename to beaglebone-cookbook/04motors/motors.rst index 78f6e23c60fa380a0df7539c33c0d3fa49bf5a6b..b5fb69abef7b0a2711b1a49776e4882dcb269ee3 100644 --- a/bone-cook-book/04motors/motors.rst +++ b/beaglebone-cookbook/04motors/motors.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-motors: +.. _beaglebone-cookbook-motors: Motors ######## @@ -21,11 +21,19 @@ The type of motor you use depends on the type of motion you want: When you know which type of motor to use, interfacing is easy. This chapter shows how to interface with each of these motors. -.. note:: Motors come in many sizes and types. This chapter presents some of the more popular types and shows how they can interface easily to the Bone. If you need to turn on and off a 120 V motor, consider using something like the PowerSwitch presented in :ref:`<displays_powerSwitch>`. +.. note:: + Motors come in many sizes and types. This chapter presents some of the more popular + types and shows how they can interface easily to the Bone. If you need to turn on and + off a 120 V motor, consider using something like the PowerSwitch presented in :ref:`<displays_powerSwitch>`. -.. note:: The Bone has built-in 3.3 V and 5 V supplies, which can supply enough current to drive some small motors. Many motors, however, draw enough current that an external power supply is needed. Therefore, an external 5 V power supply is listed as optional in many of the recipes. +.. note:: + The Bone has built-in 3.3 V and 5 V supplies, which can supply enough current to drive + some small motors. Many motors, however, draw enough current that an external power + supply is needed. Therefore, an external 5 V power supply is listed as optional in many of the recipes. -.. note:: All the examples in the book assume you have cloned the Cookbook repository on www.github.com. Go here :ref:`<basics_repo>` for instructions. +.. note:: + All the examples in the book assume you have cloned the Cookbook repository on + www.github.com. Go here :ref:`<basics_repo>` for instructions. .. _motors_servo: @@ -50,11 +58,16 @@ To make the recipe, you will need: * 1 kΩ resistor (optional, see :ref:`<app_resistor>`) * 5 V power supply (optional, see :ref:`<app_misc>`) -The 1 kΩ resistor isn't required, but it provides some protection to the general-purpose input/output (GPIO) pin in case the servo fails and draws a large current. +The 1 kΩ resistor isn't required, but it provides some protection to the general-purpose +input/output (GPIO) pin in case the servo fails and draws a large current. Wire up your servo, as shown in :ref:`<motors_servoMotor>`. -.. note:: There is no standard for how servo motor wires are colored. One of my servos is wired like :ref:`<motors_servoMotor>`: red is 3.3 V, black is ground, and yellow is the control line. I have another servo that has red as 3.3 V and ground is brown, with the control line being orange. Generally, though, the 3.3 V is in the middle. Check the datasheet for your servo before wiring. +.. note:: + There is no standard for how servo motor wires are colored. One of my servos is wired + like :ref:`<motors_servoMotor>`: red is 3.3 V, black is ground, and yellow is the control line. + I have another servo that has red as 3.3 V and ground is brown, with the control line being orange. + Generally, though, the 3.3 V is in the middle. Check the datasheet for your servo before wiring. .. _motors_servoMotor: @@ -64,7 +77,8 @@ Wire up your servo, as shown in :ref:`<motors_servoMotor>`. Driving a servo motor with the 3.3 V power supply -The code for controlling the servo motor is in _servoMotor.py_, shown in :ref:`<py_servoMotor_code>`. You need to configure the pin for PWM. +The code for controlling the servo motor is in _servoMotor.py_, shown +in :ref:`<py_servoMotor_code>`. You need to configure the pin for PWM. .. code-block:: bash @@ -77,12 +91,14 @@ The code for controlling the servo motor is in _servoMotor.py_, shown in :ref:`< .. _py_servoMotor_code: Code for driving a servo motor (servoMotor.py) + .. code-block:: python include::code/servoMotor.py[] .. _motors_servoMotor_code: + Code for driving a servo motor (servoMotor.js) .. code-block:: JavaScript @@ -111,6 +127,7 @@ Combine the code from :ref:`<digital_rotaryEncoder_js>` and :ref:`<motors_servo> .. code-block:: bash + bone$ <strong>config-pin P9_16 pwm</strong> bone$ <strong>config-pin P8_11 eqep</strong> bone$ <strong>config-pin P8_12 eqep</strong> @@ -127,7 +144,7 @@ Code for driving a servo motor with a rotary encorder(servoEncoder.py) .. _motors_dcSpeed: Controlling the Speed of a DC Motor --------------------------------------------- +------------------------------------- Problem ************** @@ -137,9 +154,13 @@ You have a DC motor (or a solenoid) and want a simple way to control its speed, Solution ********** -It would be nice if you could just wire the DC motor to BeagleBone Black and have it work, but it won't. Most motors require more current than the GPIO ports on the Bone can supply. Our solution is to use a transistor to control the current to the bone. +It would be nice if you could just wire the DC motor to BeagleBone Black and have it work, +but it won't. Most motors require more current than the GPIO ports on the Bone can supply. +Our solution is to use a transistor to control the current to the bone. -Here we configure the encoder to returns value between 0 and 180 inclusive. This value is then mapped to a value between +min+ (0.6 ma) and +max+ (2.5 ms). This number is converted from milliseconds and nanoseconds (time 1000000) and sent to the servo motor via the pwm. +Here we configure the encoder to returns value between 0 and 180 inclusive. This value is then +mapped to a value between +min+ (0.6 ma) and +max+ (2.5 ms). This number is converted from +milliseconds and nanoseconds (time 1000000) and sent to the servo motor via the pwm. Here's what you will need: diff --git a/bone-cook-book/05tips/code/blinkLED.c b/beaglebone-cookbook/05tips/code/blinkLED.c similarity index 100% rename from bone-cook-book/05tips/code/blinkLED.c rename to beaglebone-cookbook/05tips/code/blinkLED.c diff --git a/bone-cook-book/05tips/code/blinkLED.py b/beaglebone-cookbook/05tips/code/blinkLED.py similarity index 100% rename from bone-cook-book/05tips/code/blinkLED.py rename to beaglebone-cookbook/05tips/code/blinkLED.py diff --git a/bone-cook-book/05tips/code/ipMasquerade.sh b/beaglebone-cookbook/05tips/code/ipMasquerade.sh similarity index 100% rename from bone-cook-book/05tips/code/ipMasquerade.sh rename to beaglebone-cookbook/05tips/code/ipMasquerade.sh diff --git a/bone-cook-book/05tips/code/setDNS.sh b/beaglebone-cookbook/05tips/code/setDNS.sh similarity index 100% rename from bone-cook-book/05tips/code/setDNS.sh rename to beaglebone-cookbook/05tips/code/setDNS.sh diff --git a/bone-cook-book/05tips/figures/FTDIPins.png b/beaglebone-cookbook/05tips/figures/FTDIPins.png similarity index 100% rename from bone-cook-book/05tips/figures/FTDIPins.png rename to beaglebone-cookbook/05tips/figures/FTDIPins.png diff --git a/bone-cook-book/05tips/figures/FTDIcable.jpg b/beaglebone-cookbook/05tips/figures/FTDIcable.jpg similarity index 100% rename from bone-cook-book/05tips/figures/FTDIcable.jpg rename to beaglebone-cookbook/05tips/figures/FTDIcable.jpg diff --git a/bone-cook-book/05tips/figures/FTDIconnector.jpg b/beaglebone-cookbook/05tips/figures/FTDIconnector.jpg similarity index 100% rename from bone-cook-book/05tips/figures/FTDIconnector.jpg rename to beaglebone-cookbook/05tips/figures/FTDIconnector.jpg diff --git a/bone-cook-book/05tips/figures/cape-headers-digital.png b/beaglebone-cookbook/05tips/figures/cape-headers-digital.png similarity index 100% rename from bone-cook-book/05tips/figures/cape-headers-digital.png rename to beaglebone-cookbook/05tips/figures/cape-headers-digital.png diff --git a/bone-cook-book/05tips/figures/gedit.png b/beaglebone-cookbook/05tips/figures/gedit.png similarity index 100% rename from bone-cook-book/05tips/figures/gedit.png rename to beaglebone-cookbook/05tips/figures/gedit.png diff --git a/bone-cook-book/05tips/figures/hdmiConverter.jpg b/beaglebone-cookbook/05tips/figures/hdmiConverter.jpg similarity index 100% rename from bone-cook-book/05tips/figures/hdmiConverter.jpg rename to beaglebone-cookbook/05tips/figures/hdmiConverter.jpg diff --git a/bone-cook-book/05tips/figures/internLED.png b/beaglebone-cookbook/05tips/figures/internLED.png similarity index 100% rename from bone-cook-book/05tips/figures/internLED.png rename to beaglebone-cookbook/05tips/figures/internLED.png diff --git a/bone-cook-book/05tips/figures/latest-images.png b/beaglebone-cookbook/05tips/figures/latest-images.png similarity index 100% rename from bone-cook-book/05tips/figures/latest-images.png rename to beaglebone-cookbook/05tips/figures/latest-images.png diff --git a/bone-cook-book/05tips/figures/leafpad.png b/beaglebone-cookbook/05tips/figures/leafpad.png similarity index 100% rename from bone-cook-book/05tips/figures/leafpad.png rename to beaglebone-cookbook/05tips/figures/leafpad.png diff --git a/bone-cook-book/05tips/figures/nano.png b/beaglebone-cookbook/05tips/figures/nano.png similarity index 100% rename from bone-cook-book/05tips/figures/nano.png rename to beaglebone-cookbook/05tips/figures/nano.png diff --git a/bone-cook-book/05tips/figures/network.png b/beaglebone-cookbook/05tips/figures/network.png similarity index 100% rename from bone-cook-book/05tips/figures/network.png rename to beaglebone-cookbook/05tips/figures/network.png diff --git a/bone-cook-book/05tips/figures/npm.png b/beaglebone-cookbook/05tips/figures/npm.png similarity index 100% rename from bone-cook-book/05tips/figures/npm.png rename to beaglebone-cookbook/05tips/figures/npm.png diff --git a/bone-cook-book/05tips/figures/vi.png b/beaglebone-cookbook/05tips/figures/vi.png similarity index 100% rename from bone-cook-book/05tips/figures/vi.png rename to beaglebone-cookbook/05tips/figures/vi.png diff --git a/bone-cook-book/05tips/figures/vnc1.png b/beaglebone-cookbook/05tips/figures/vnc1.png similarity index 100% rename from bone-cook-book/05tips/figures/vnc1.png rename to beaglebone-cookbook/05tips/figures/vnc1.png diff --git a/bone-cook-book/05tips/figures/vnc2.png b/beaglebone-cookbook/05tips/figures/vnc2.png similarity index 100% rename from bone-cook-book/05tips/figures/vnc2.png rename to beaglebone-cookbook/05tips/figures/vnc2.png diff --git a/bone-cook-book/05tips/figures/vnc3.png b/beaglebone-cookbook/05tips/figures/vnc3.png similarity index 100% rename from bone-cook-book/05tips/figures/vnc3.png rename to beaglebone-cookbook/05tips/figures/vnc3.png diff --git a/bone-cook-book/05tips/figures/wicd.png b/beaglebone-cookbook/05tips/figures/wicd.png similarity index 100% rename from bone-cook-book/05tips/figures/wicd.png rename to beaglebone-cookbook/05tips/figures/wicd.png diff --git a/bone-cook-book/05tips/figures/wicdConfig.png b/beaglebone-cookbook/05tips/figures/wicdConfig.png similarity index 100% rename from bone-cook-book/05tips/figures/wicdConfig.png rename to beaglebone-cookbook/05tips/figures/wicdConfig.png diff --git a/bone-cook-book/05tips/figures/wicdConnected.png b/beaglebone-cookbook/05tips/figures/wicdConnected.png similarity index 100% rename from bone-cook-book/05tips/figures/wicdConnected.png rename to beaglebone-cookbook/05tips/figures/wicdConnected.png diff --git a/bone-cook-book/05tips/figures/wicdPassword.png b/beaglebone-cookbook/05tips/figures/wicdPassword.png similarity index 100% rename from bone-cook-book/05tips/figures/wicdPassword.png rename to beaglebone-cookbook/05tips/figures/wicdPassword.png diff --git a/bone-cook-book/05tips/figures/wicdPrefs.png b/beaglebone-cookbook/05tips/figures/wicdPrefs.png similarity index 100% rename from bone-cook-book/05tips/figures/wicdPrefs.png rename to beaglebone-cookbook/05tips/figures/wicdPrefs.png diff --git a/bone-cook-book/05tips/figures/wicdSecure.png b/beaglebone-cookbook/05tips/figures/wicdSecure.png similarity index 100% rename from bone-cook-book/05tips/figures/wicdSecure.png rename to beaglebone-cookbook/05tips/figures/wicdSecure.png diff --git a/bone-cook-book/05tips/tips.rst b/beaglebone-cookbook/05tips/tips.rst similarity index 84% rename from bone-cook-book/05tips/tips.rst rename to beaglebone-cookbook/05tips/tips.rst index d78d3c7ffe2539a0aa0eab3f36201e836627a81a..89ff503d37fa6dc721d9c96e1284a56b6f5569a1 100644 --- a/bone-cook-book/05tips/tips.rst +++ b/beaglebone-cookbook/05tips/tips.rst @@ -1,21 +1,27 @@ -.. _bone-cook-book-tips: +.. _beaglebone-cookbook-tips: Beyond the Basics ################## Introduction ------------------------- -In :ref:`<basics>`, you learned how to set up BeagleBone Black, and :ref:`<sensors>`, :ref:`<displays>`, and :ref:`<motors>` showed how to interface to the physical world. The remainder of the book moves into some more exciting advanced topics, and this chapter gets you ready for them. +--------------- + +In :ref:`<basics>`, you learned how to set up BeagleBone Black, and +:ref:`<sensors>`, :ref:`<displays>`, and :ref:`<motors>` showed how to +interface to the physical world. The remainder of the book moves into some +more exciting advanced topics, and this chapter gets you ready for them. -The recipes in this chapter assume that you are running Linux on your host computer (:ref:`<tips_pick_os>`) and are comfortable with using Linux. We continue to assume that you are logged in as +debian+ on your Bone. +The recipes in this chapter assume that you are running Linux on your host +computer (:ref:`<tips_pick_os>`) and are comfortable with using Linux. We +continue to assume that you are logged in as +debian+ on your Bone. .. _tips_hdmi: Running Your Bone Standalone ------------------------- +----------------------------- Problem -************* +********* You want to use BeagleBone Black as a desktop computer with keyboard, mouse, and an HDMI display. @@ -32,9 +38,14 @@ To make this recipe, you will need: * USB keyboard and mouse * Powered USB hub (see :ref:`<app_misc>`) -.. note:: The microHDMI adapter is nice because it allows you to use a regular HDMI cable with the Bone. However, it will block other ports and can damage the Bone if you aren't careful. The microHDMI-to-HDMI cable won't have these problems. +.. note:: + The microHDMI adapter is nice because it allows you to use a regular HDMI cable + with the Bone. However, it will block other ports and can damage the Bone if you + aren't careful. The microHDMI-to-HDMI cable won't have these problems. -.. tip:: You can also use an HDMI-to-DVI cable (:ref:`<app_misc>`) and use your Bone with a DVI-D display. +.. tip:: + You can also use an HDMI-to-DVI cable (:ref:`<app_misc>`) + and use your Bone with a DVI-D display. The adapter looks something like :ref:`<tips_HDMI_adaptor_fig>`. @@ -60,24 +71,25 @@ If nothing appears after the reboot, edit the _/boot/uEnv.txt_ file. Search for Then reboot. -//// -PRODUCTION: in the following tip, we're trying to display the hash symbol (#), all by itself, in constant width. Using +#+ produces an empty space in the build, and I don't know how to escape special characters within what should be literal strings. +.. PRODUCTION: in the following tip, we're trying to display the hash symbol (#), all by itself, in constant width. Using +#+ produces an empty space in the build, and I don't know how to escape special characters within what should be literal strings. -Adding to my confusion, the # signs are dropped in the first paragraph of the tip, but not in the second, which is formatted in the same exact way. +.. Adding to my confusion, the # signs are dropped in the first paragraph of the tip, but not in the second, which is formatted in the same exact way. + +.. Also, using ## in the code italicizes the second # and everything after it in the line, which should not happen. -Also, using ## in the code italicizes the second # and everything after it in the line, which should not happen. -//// The _/boot/uEnv.txt_ file contains a number of configuration commands that are executed at boot time. The +#+ character is used to add comments; that is, everything to the right of a +# is ignored by the Bone and is assumed to be for humans to read. In the previous example, +###Disable auto loading+ is a comment that informs us the next line(s) are for disabling things. Two +disable_uboot_overlay+ commands follow. Both should be commented-out and won't be executed by the Bon Why not just remove the line? Later, you might decide you need more general-purpose input/output (GPIO) pins and don't need the HDMI display. If so, just remove the +#+ from the +disable_uboot_overlay_video=1+ command. If you had completely removed the line earlier, you would have to look up the details somewhere to re-create it. When in doubt, comment-out; don't delete. -.. note:: If you want to re-enable the HDMI audio, just comment-out the line you added. +.. note:: + If you want to re-enable the HDMI audio, just comment-out the line you added. The Bone has only one USB port, so you will need to get either a keyboard with a USB hub (see :ref:`<app_misc>`) or a USB hub. Plug the USB hub into the Bone and then plug your keyboard and mouse in to the hub. You now have a Beagle workstation; no host computer is needed. -.. tip:: A powered hub is recommended because USB can supply only 500 mA, and you'll want to plug many things into the Bone. +.. tip:: + A powered hub is recommended because USB can supply only 500 mA, and you'll want to plug many things into the Bone. Discussion ************* @@ -124,7 +136,9 @@ Solution ************* -:ref:`<basics_vsc_IDE>` shows how to run shell commands in the Visual Studio Code +bash+ tab. However, the Bone has Secure Shell (SSH) enabled right out of the box, so you can easily connect by using the following command to log in as user +debian+, (note the +$+ at the end of the prompt): +:ref:`<basics_vsc_IDE>` shows how to run shell commands in the Visual Studio Code +bash+ tab. +However, the Bone has Secure Shell (SSH) enabled right out of the box, so you can easily +connect by using the following command to log in as user +debian+, (note the +$+ at the end of the prompt): .. code-block:: bash @@ -184,7 +198,7 @@ Looks like I'm already in the group, but if you aren't, just add yourself to the .. code-block:: bash -host$ sudo adduser $USER dialout + host$ sudo adduser $USER dialout You have to run +adduser+ only once. Your host computer will remember the next time you boot up. Now, install and run the +screen+ command: @@ -228,9 +242,13 @@ To make this recipe, you will need: * 3.3 V FTDI cable (see :ref:`<app_misc>`) -.. warning:: Be sure to get a 3.3 V FTDI cable (shown in :ref:`<tips_FTDIcable_fig>`), because the 5 V cables won't work. +.. warning:: + Be sure to get a 3.3 V FTDI cable (shown in :ref:`<tips_FTDIcable_fig>`), + because the 5 V cables won't work. -.. tip:: The Bone's Serial Debug J1 connector has Pin 1 connected to ground, Pin 4 to receive, and Pin 5 to transmit. The other pins are not attached. +.. tip:: + The Bone's Serial Debug J1 connector has Pin 1 connected to ground, + Pin 4 to receive, and Pin 5 to transmit. The other pins are not attached. .. _tips_FTDIcable_fig: @@ -241,7 +259,8 @@ To make this recipe, you will need: FTDI cable -Look for a small triangle at the end of the FTDI cable (:ref:`<tips_FTDIconnector_fig>`). It's often connected to the black wire. +Look for a small triangle at the end of the FTDI cable (:ref:`<tips_FTDIconnector_fig>`). +It's often connected to the black wire. .. _tips_FTDIconnector_fig: @@ -252,9 +271,9 @@ Look for a small triangle at the end of the FTDI cable (:ref:`<tips_FTDIconnecto FTDI connector -Next, look for the FTDI pins of the Bone (labeled +J1+ on the Bone), -shown in :ref:`<tips_black_hardware_details_fig>`. -They are next to the P9 header and begin near pin 20. There is a white dot near P9_20. +Next, look for the FTDI pins of the Bone (labeled +J1+ on the Bone), shown in +:ref:`<tips_black_hardware_details_fig>`. They are next to the P9 header +and begin near pin 20. There is a white dot near P9_20. .. _tips_black_hardware_details_fig: @@ -286,7 +305,9 @@ Now, run the following commands on your host computer: beaglebone login: -.. note:: Your screen might initially be blank. Press Enter a couple times to see the login prompt. +.. note:: + Your screen might initially be blank. Press Enter + a couple times to see the login prompt. Discussion ************* @@ -313,7 +334,9 @@ Log in to your Bone and enter the following command: Discussion ************* -:ref:`<basics_latest_os>` shows how to open the _ID.txt_ file to see the OS version. The _/etc/dogtag_ file has the same contents and is easier to find if you already have a command prompt. See :ref:`<basics_install_os>` if you need to update your OS. +:ref:`<basics_latest_os>` shows how to open the _ID.txt_ file to see the OS version. +The _/etc/dogtag_ file has the same contents and is easier to find if you already +have a command prompt. See :ref:`<basics_install_os>` if you need to update your OS. Controlling the Bone Remotely with a VNC ------------------------ @@ -382,9 +405,13 @@ Click Connect to start graphical access to your Bone, as shown in :ref:`<tips_vn The Remmina Remote Desktop Client showing the BeagleBone desktop -.. tip:: You might need to resize the VNC screen on your host to see the bottom menu bar on your Bone. +.. tip:: + You might need to resize the VNC screen on your + host to see the bottom menu bar on your Bone. -.. note:: You need to have X Windows installed and running for the VNC to work. Here's how to install it. This needs some 250M of disk space and 19 minutes to install. +.. note:: + You need to have X Windows installed and running for the VNC to work. + Here's how to install it. This needs some 250M of disk space and 19 minutes to install. .. code-block:: bash @@ -504,14 +531,18 @@ You want to run an editor to change a file. Solution ************* -The Bone comes with a number of editors. The simplest to learn is +nano+. Just enter the following command: +The Bone comes with a number of editors. The simplest to learn is +nano+. +Just enter the following command: .. code-block:: bash -bone$ nano file + bone$ nano file -You are now in nano (:ref:`<tips_nano_fig>`). You can't move around the screen using the mouse, so use the arrow keys. The bottom two lines of the screen list some useful commands. Pressing ˄G (Ctrl-G) will display more useful commands. ˄X (Ctrl-X) exits nano and gives you the option of saving the file. +You are now in nano (:ref:`<tips_nano_fig>`). You can't move around the screen +using the mouse, so use the arrow keys. The bottom two lines of the screen +list some useful commands. Pressing ˄G (Ctrl-G) will display more useful +commands. ˄X (Ctrl-X) exits nano and gives you the option of saving the file. .. _tips_nano_fig: @@ -522,7 +553,9 @@ You are now in nano (:ref:`<tips_nano_fig>`). You can't move around the screen u Editing a file with nano -.. tip:: By default, the file you create will be saved in the directory from which you opened +nano+. +.. tip:: + By default, the file you create will be saved + in the directory from which you opened +nano+. Discussion ************* @@ -557,7 +590,8 @@ and the other end into your home hub/router. The yellow and green link lights on The RJ45 port on the Bone -If your router is already configured to run DHCP (Dynamical Host Configuration Protocol), it will automatically assign an IP address to the Bone. +If your router is already configured to run DHCP (Dynamical Host Configuration Protocol), +it will automatically assign an IP address to the Bone. .. warning:: It might take a minute or two for your router to detect the Bone and assign the IP address. @@ -602,9 +636,12 @@ The +inet+ field shows that my Internet address is +10.0.5.144+ for the RJ45 con On my university campus, you must register your MAC address before any device will work on the network. The +HWaddr+ field gives the MAC address. For +eth0+, it's +c8:a0:30:a6:26:e8+. -The IP address of your Bone can change. If it's been assigned by DHCP, it can change at any time. The MAC address, however, never changes; it is assigned to your ethernet device when it's manufactured. +The IP address of your Bone can change. If it's been assigned by DHCP, it can change at any time. +The MAC address, however, never changes; it is assigned to your ethernet device when it's manufactured. -.. warning:: When a Bone is connected to some networks it becomes visible to the _world_. If you don't secure your Bone, the world will soon find it. See :ref:`<tips_passwords>` and :ref:`<tips_firewall +.. warning:: + When a Bone is connected to some networks it becomes visible to the _world_. If you don't secure your Bone, + the world will soon find it. See :ref:`<tips_passwords>` and :ref:`<tips_firewall On many home networks, you will be behind a firewall and won't be as visible. @@ -658,7 +695,9 @@ To make this recipe, you will need: * USB Wifi adapter (see :ref:`<app_misc>`) * 5 V external power supply (see :ref:`<app_misc>`) -.. warning:: Most adapters need at least 1 A of current to run, and USB supplies only 0.5 A, so be sure to use an external power supply. Otherwise, you will experience erratic behavior and random crashes. +.. warning:: + Most adapters need at least 1 A of current to run, and USB supplies only 0.5 A, so be sure to use an + external power supply. Otherwise, you will experience erratic behavior and random crashes. First, plug in the WiFi adapter and the 5 V external power supply and reboot. @@ -723,7 +762,7 @@ If no name appears, try +ip a+: .. code-block:: bash -bone$ sudo nano /etc/wpa_supplicant/wpa_supplicant-wlan0.conf + bone$ sudo nano /etc/wpa_supplicant/wpa_supplicant-wlan0.conf In the file you'll see: @@ -870,7 +909,8 @@ Web servers typically listen to port +80+. First, look up the IP address of your It's the number following +inet addr:+, which in my case is +137.112.41.35+. -.. tip:: If you are on a wireless network, find the IP address associated with +wlan0+. +.. tip:: + If you are on a wireless network, find the IP address associated with +wlan0+. Then run the following, using your host's IP address: @@ -909,6 +949,7 @@ Check out https://www.howtogeek.com/devops/how-to-secure-your-linux-server-with- I'll summarize the initial setup here. First install and check the status: + .. code-block:: bash bone$ sudo apt install ufw @@ -917,6 +958,7 @@ First install and check the status: Now turn off everything coming in and leave on all outgoing. Note, this won't take effect until +ufw+ is enabled. + .. code-block:: bash bone$ sudo ufw default deny incoming @@ -924,6 +966,7 @@ Now turn off everything coming in and leave on all outgoing. Note, this won't t Don't enable yet, make sure +ssh+ still has access + .. code-block:: bash bone$ sudo ufw allow 22 @@ -973,8 +1016,8 @@ The firewall will remain on, even after a reboot. Disable it now if you don't wa .. code-block:: bash -bone$ sudo ufw disable -Firewall stopped and disabled on system startup + bone$ sudo ufw disable + Firewall stopped and disabled on system startup See the How-To Geek article for more examples. @@ -995,7 +1038,9 @@ You want to do more cool things with your BeagleBone by installing more programs Solution ************* -.. warning:: Your Bone needs to be on the network for this to work. See :ref:`<networking_wired>`, :ref:`<networking_wireless>`, or :ref:`<networking_usb>`. +.. warning:: + Your Bone needs to be on the network for this to work. See :ref:`<networking_wired>`, + :ref:`<networking_wireless>`, or :ref:`<networking_usb>`. The easiest way to install more software is to use +apt+: @@ -1005,7 +1050,10 @@ The easiest way to install more software is to use +apt+: bone$ sudo apt install "name of software" -A +sudo+ is necessary since you aren't running as +root+. The first command downloads package lists from various repositories and updates them to get information on the newest versions of packages and their dependencies. (You need to run it only once a week or so.) The second command fetches the software and installs it and all packages it depends on. +A +sudo+ is necessary since you aren't running as +root+. The first command downloads +package lists from various repositories and updates them to get information on the +newest versions of packages and their dependencies. (You need to run it only once a week or so.) +The second command fetches the software and installs it and all packages it depends on. How do you find out what software you can install? Try running this: @@ -1027,7 +1075,7 @@ Suppose that you would like to install an online dictionary (+dict+). Just run t .. code-block:: bash -bone$ sudo apt install dict + bone$ sudo apt install dict Now you can run +dict+. @@ -1035,12 +1083,10 @@ Now you can run +dict+. Discussion ************* - - .. _tips_apt_remove: Removing Packages Installed with apt ------------------------- +------------------------------------- Problem ************* @@ -1054,18 +1100,18 @@ Solution .. code-block:: bash -bone$ sudo apt remove dict -Reading package lists... Done -Building dependency tree -Reading state information... Done -The following packages were automatically installed and are no longer required: - libmaa3 librecode0 recode -Use 'apt autoremove' to remove them. -The following packages will be REMOVED: - dict -0 upgraded, 0 newly installed, 1 to remove and 27 not upgraded. -After this operation, 164 kB disk space will be freed. -Do you want to continue [Y/n]? y + bone$ sudo apt remove dict + Reading package lists... Done + Building dependency tree + Reading state information... Done + The following packages were automatically installed and are no longer required: + libmaa3 librecode0 recode + Use 'apt autoremove' to remove them. + The following packages will be REMOVED: + dict + 0 upgraded, 0 newly installed, 1 to remove and 27 not upgraded. + After this operation, 164 kB disk space will be freed. + Do you want to continue [Y/n]? y Discussion @@ -1162,7 +1208,9 @@ Removing preinstalled packages You might not need a few things that come preinstalled in the Debian image, including such things as OpenCV, the Chromium web browser, and some documentation. -.. note:: The Chromium web browser is the open source version of Google's Chrome web browser. Unless you are using the Bone as a desktop computer, you can probably remove it. +.. note:: + The Chromium web browser is the open source version of Google's Chrome web browser. + Unless you are using the Bone as a desktop computer, you can probably remove it. Here's how you can remove these: @@ -1288,7 +1336,7 @@ Discussion .. _misc_libsoc: Using C to Interact with the Physical World ------------------------- +-------------------------------------------- Problem ************* diff --git a/bone-cook-book/06iot/code/GPIOserver.js b/beaglebone-cookbook/06iot/code/GPIOserver.js similarity index 100% rename from bone-cook-book/06iot/code/GPIOserver.js rename to beaglebone-cookbook/06iot/code/GPIOserver.js diff --git a/bone-cook-book/06iot/code/analogInContinuous.py b/beaglebone-cookbook/06iot/code/analogInContinuous.py similarity index 100% rename from bone-cook-book/06iot/code/analogInContinuous.py rename to beaglebone-cookbook/06iot/code/analogInContinuous.py diff --git a/bone-cook-book/06iot/code/emailTest.py b/beaglebone-cookbook/06iot/code/emailTest.py similarity index 100% rename from bone-cook-book/06iot/code/emailTest.py rename to beaglebone-cookbook/06iot/code/emailTest.py diff --git a/bone-cook-book/06iot/code/flask/app1.py b/beaglebone-cookbook/06iot/code/flask/app1.py similarity index 100% rename from bone-cook-book/06iot/code/flask/app1.py rename to beaglebone-cookbook/06iot/code/flask/app1.py diff --git a/bone-cook-book/06iot/code/flask/app2.py b/beaglebone-cookbook/06iot/code/flask/app2.py similarity index 100% rename from bone-cook-book/06iot/code/flask/app2.py rename to beaglebone-cookbook/06iot/code/flask/app2.py diff --git a/bone-cook-book/06iot/code/flask/app3.py b/beaglebone-cookbook/06iot/code/flask/app3.py similarity index 100% rename from bone-cook-book/06iot/code/flask/app3.py rename to beaglebone-cookbook/06iot/code/flask/app3.py diff --git a/bone-cook-book/06iot/code/flask/app4.py b/beaglebone-cookbook/06iot/code/flask/app4.py similarity index 100% rename from bone-cook-book/06iot/code/flask/app4.py rename to beaglebone-cookbook/06iot/code/flask/app4.py diff --git a/bone-cook-book/06iot/code/flask/app5.py b/beaglebone-cookbook/06iot/code/flask/app5.py similarity index 100% rename from bone-cook-book/06iot/code/flask/app5.py rename to beaglebone-cookbook/06iot/code/flask/app5.py diff --git a/bone-cook-book/06iot/code/flask/flask.service b/beaglebone-cookbook/06iot/code/flask/flask.service similarity index 100% rename from bone-cook-book/06iot/code/flask/flask.service rename to beaglebone-cookbook/06iot/code/flask/flask.service diff --git a/bone-cook-book/06iot/code/flask/helloWorld.py b/beaglebone-cookbook/06iot/code/flask/helloWorld.py similarity index 100% rename from bone-cook-book/06iot/code/flask/helloWorld.py rename to beaglebone-cookbook/06iot/code/flask/helloWorld.py diff --git a/bone-cook-book/06iot/code/flask/install.sh b/beaglebone-cookbook/06iot/code/flask/install.sh similarity index 100% rename from bone-cook-book/06iot/code/flask/install.sh rename to beaglebone-cookbook/06iot/code/flask/install.sh diff --git a/bone-cook-book/06iot/code/flask/ngrok.yml b/beaglebone-cookbook/06iot/code/flask/ngrok.yml similarity index 100% rename from bone-cook-book/06iot/code/flask/ngrok.yml rename to beaglebone-cookbook/06iot/code/flask/ngrok.yml diff --git a/bone-cook-book/06iot/code/flask/setup.sh b/beaglebone-cookbook/06iot/code/flask/setup.sh similarity index 100% rename from bone-cook-book/06iot/code/flask/setup.sh rename to beaglebone-cookbook/06iot/code/flask/setup.sh diff --git a/bone-cook-book/06iot/code/flask/static/favicon.ico b/beaglebone-cookbook/06iot/code/flask/static/favicon.ico similarity index 100% rename from bone-cook-book/06iot/code/flask/static/favicon.ico rename to beaglebone-cookbook/06iot/code/flask/static/favicon.ico diff --git a/bone-cook-book/06iot/code/flask/static/style.css b/beaglebone-cookbook/06iot/code/flask/static/style.css similarity index 100% rename from bone-cook-book/06iot/code/flask/static/style.css rename to beaglebone-cookbook/06iot/code/flask/static/style.css diff --git a/bone-cook-book/06iot/code/flask/templates/index1.html b/beaglebone-cookbook/06iot/code/flask/templates/index1.html similarity index 100% rename from bone-cook-book/06iot/code/flask/templates/index1.html rename to beaglebone-cookbook/06iot/code/flask/templates/index1.html diff --git a/bone-cook-book/06iot/code/flask/templates/index2.html b/beaglebone-cookbook/06iot/code/flask/templates/index2.html similarity index 100% rename from bone-cook-book/06iot/code/flask/templates/index2.html rename to beaglebone-cookbook/06iot/code/flask/templates/index2.html diff --git a/bone-cook-book/06iot/code/flask/templates/index3.html b/beaglebone-cookbook/06iot/code/flask/templates/index3.html similarity index 100% rename from bone-cook-book/06iot/code/flask/templates/index3.html rename to beaglebone-cookbook/06iot/code/flask/templates/index3.html diff --git a/bone-cook-book/06iot/code/flask/templates/index4.html b/beaglebone-cookbook/06iot/code/flask/templates/index4.html similarity index 100% rename from bone-cook-book/06iot/code/flask/templates/index4.html rename to beaglebone-cookbook/06iot/code/flask/templates/index4.html diff --git a/bone-cook-book/06iot/code/flask/templates/index5.html b/beaglebone-cookbook/06iot/code/flask/templates/index5.html similarity index 100% rename from bone-cook-book/06iot/code/flask/templates/index5.html rename to beaglebone-cookbook/06iot/code/flask/templates/index5.html diff --git a/bone-cook-book/06iot/code/flotDemo.html b/beaglebone-cookbook/06iot/code/flotDemo.html similarity index 100% rename from bone-cook-book/06iot/code/flotDemo.html rename to beaglebone-cookbook/06iot/code/flotDemo.html diff --git a/bone-cook-book/06iot/code/flotDemo.js b/beaglebone-cookbook/06iot/code/flotDemo.js similarity index 100% rename from bone-cook-book/06iot/code/flotDemo.js rename to beaglebone-cookbook/06iot/code/flotDemo.js diff --git a/bone-cook-book/06iot/code/jQueryDemo.js b/beaglebone-cookbook/06iot/code/jQueryDemo.js similarity index 100% rename from bone-cook-book/06iot/code/jQueryDemo.js rename to beaglebone-cookbook/06iot/code/jQueryDemo.js diff --git a/bone-cook-book/06iot/code/jQueryInstall.sh b/beaglebone-cookbook/06iot/code/jQueryInstall.sh similarity index 100% rename from bone-cook-book/06iot/code/jQueryInstall.sh rename to beaglebone-cookbook/06iot/code/jQueryInstall.sh diff --git a/bone-cook-book/06iot/code/launchPad.js b/beaglebone-cookbook/06iot/code/launchPad.js similarity index 100% rename from bone-cook-book/06iot/code/launchPad.js rename to beaglebone-cookbook/06iot/code/launchPad.js diff --git a/bone-cook-book/06iot/code/launchPad/launchPad.ino b/beaglebone-cookbook/06iot/code/launchPad/launchPad.ino similarity index 100% rename from bone-cook-book/06iot/code/launchPad/launchPad.ino rename to beaglebone-cookbook/06iot/code/launchPad/launchPad.ino diff --git a/bone-cook-book/06iot/code/nodemailer-install.sh b/beaglebone-cookbook/06iot/code/nodemailer-install.sh similarity index 100% rename from bone-cook-book/06iot/code/nodemailer-install.sh rename to beaglebone-cookbook/06iot/code/nodemailer-install.sh diff --git a/bone-cook-book/06iot/code/nodemailer-test.js b/beaglebone-cookbook/06iot/code/nodemailer-test.js similarity index 100% rename from bone-cook-book/06iot/code/nodemailer-test.js rename to beaglebone-cookbook/06iot/code/nodemailer-test.js diff --git a/bone-cook-book/06iot/code/noderedExample.json b/beaglebone-cookbook/06iot/code/noderedExample.json similarity index 100% rename from bone-cook-book/06iot/code/noderedExample.json rename to beaglebone-cookbook/06iot/code/noderedExample.json diff --git a/bone-cook-book/06iot/code/processingDemo.js b/beaglebone-cookbook/06iot/code/processingDemo.js similarity index 100% rename from bone-cook-book/06iot/code/processingDemo.js rename to beaglebone-cookbook/06iot/code/processingDemo.js diff --git a/bone-cook-book/06iot/code/server.js b/beaglebone-cookbook/06iot/code/server.js similarity index 100% rename from bone-cook-book/06iot/code/server.js rename to beaglebone-cookbook/06iot/code/server.js diff --git a/bone-cook-book/06iot/code/test.html b/beaglebone-cookbook/06iot/code/test.html similarity index 100% rename from bone-cook-book/06iot/code/test.html rename to beaglebone-cookbook/06iot/code/test.html diff --git a/bone-cook-book/06iot/code/twilio-test.js b/beaglebone-cookbook/06iot/code/twilio-test.js similarity index 100% rename from bone-cook-book/06iot/code/twilio-test.js rename to beaglebone-cookbook/06iot/code/twilio-test.js diff --git a/bone-cook-book/06iot/code/twilioSetup.sh b/beaglebone-cookbook/06iot/code/twilioSetup.sh similarity index 100% rename from bone-cook-book/06iot/code/twilioSetup.sh rename to beaglebone-cookbook/06iot/code/twilioSetup.sh diff --git a/bone-cook-book/06iot/code/twilioTest.py b/beaglebone-cookbook/06iot/code/twilioTest.py similarity index 100% rename from bone-cook-book/06iot/code/twilioTest.py rename to beaglebone-cookbook/06iot/code/twilioTest.py diff --git a/bone-cook-book/06iot/code/twitterInstall.sh b/beaglebone-cookbook/06iot/code/twitterInstall.sh similarity index 100% rename from bone-cook-book/06iot/code/twitterInstall.sh rename to beaglebone-cookbook/06iot/code/twitterInstall.sh diff --git a/bone-cook-book/06iot/code/twitterKeys.sh b/beaglebone-cookbook/06iot/code/twitterKeys.sh similarity index 100% rename from bone-cook-book/06iot/code/twitterKeys.sh rename to beaglebone-cookbook/06iot/code/twitterKeys.sh diff --git a/bone-cook-book/06iot/code/twitterPushbutton.js b/beaglebone-cookbook/06iot/code/twitterPushbutton.js similarity index 100% rename from bone-cook-book/06iot/code/twitterPushbutton.js rename to beaglebone-cookbook/06iot/code/twitterPushbutton.js diff --git a/bone-cook-book/06iot/code/twitterSearch.js b/beaglebone-cookbook/06iot/code/twitterSearch.js similarity index 100% rename from bone-cook-book/06iot/code/twitterSearch.js rename to beaglebone-cookbook/06iot/code/twitterSearch.js diff --git a/bone-cook-book/06iot/code/twitterStream.js b/beaglebone-cookbook/06iot/code/twitterStream.js similarity index 100% rename from bone-cook-book/06iot/code/twitterStream.js rename to beaglebone-cookbook/06iot/code/twitterStream.js diff --git a/bone-cook-book/06iot/code/twitterTimeLine.js b/beaglebone-cookbook/06iot/code/twitterTimeLine.js similarity index 100% rename from bone-cook-book/06iot/code/twitterTimeLine.js rename to beaglebone-cookbook/06iot/code/twitterTimeLine.js diff --git a/bone-cook-book/06iot/code/twitterUpload.js b/beaglebone-cookbook/06iot/code/twitterUpload.js similarity index 100% rename from bone-cook-book/06iot/code/twitterUpload.js rename to beaglebone-cookbook/06iot/code/twitterUpload.js diff --git a/bone-cook-book/06iot/code/twitter_create_tweet.py b/beaglebone-cookbook/06iot/code/twitter_create_tweet.py similarity index 100% rename from bone-cook-book/06iot/code/twitter_create_tweet.py rename to beaglebone-cookbook/06iot/code/twitter_create_tweet.py diff --git a/bone-cook-book/06iot/code/twitter_delete_tweet.py b/beaglebone-cookbook/06iot/code/twitter_delete_tweet.py similarity index 100% rename from bone-cook-book/06iot/code/twitter_delete_tweet.py rename to beaglebone-cookbook/06iot/code/twitter_delete_tweet.py diff --git a/bone-cook-book/06iot/code/twitter_recent_search.py b/beaglebone-cookbook/06iot/code/twitter_recent_search.py similarity index 100% rename from bone-cook-book/06iot/code/twitter_recent_search.py rename to beaglebone-cookbook/06iot/code/twitter_recent_search.py diff --git a/bone-cook-book/06iot/code/twitter_user_tweets.py b/beaglebone-cookbook/06iot/code/twitter_user_tweets.py similarity index 95% rename from bone-cook-book/06iot/code/twitter_user_tweets.py rename to beaglebone-cookbook/06iot/code/twitter_user_tweets.py index 60a04331a49784b792f739164802316e4565addd..19dac531a57a239b87b08705a81178038f50754d 100755 --- a/bone-cook-book/06iot/code/twitter_user_tweets.py +++ b/beaglebone-cookbook/06iot/code/twitter_user_tweets.py @@ -20,7 +20,7 @@ def get_params(): # Options include: # attachments, author_id, context_annotations, # conversation_id, created_at, entities, geo, id, - # in_reply_to_user_id, lang, non_public_metrics, organic_metrics, + # in_reply_to_user_id, lang, non_ai_64_lic_metrics, organic_metrics, # possibly_sensitive, promoted_metrics, public_metrics, referenced_tweets, # source, text, and withheld return {"tweet.fields": "created_at"} diff --git a/bone-cook-book/06iot/code/weather.py b/beaglebone-cookbook/06iot/code/weather.py similarity index 100% rename from bone-cook-book/06iot/code/weather.py rename to beaglebone-cookbook/06iot/code/weather.py diff --git a/bone-cook-book/06iot/code/weatherSetup.sh b/beaglebone-cookbook/06iot/code/weatherSetup.sh similarity index 100% rename from bone-cook-book/06iot/code/weatherSetup.sh rename to beaglebone-cookbook/06iot/code/weatherSetup.sh diff --git a/bone-cook-book/06iot/code/xbee.js b/beaglebone-cookbook/06iot/code/xbee.js similarity index 100% rename from bone-cook-book/06iot/code/xbee.js rename to beaglebone-cookbook/06iot/code/xbee.js diff --git a/bone-cook-book/06iot/figures/GPIOWebPage.png b/beaglebone-cookbook/06iot/figures/GPIOWebPage.png similarity index 100% rename from bone-cook-book/06iot/figures/GPIOWebPage.png rename to beaglebone-cookbook/06iot/figures/GPIOWebPage.png diff --git a/bone-cook-book/06iot/figures/Sine1k.png b/beaglebone-cookbook/06iot/figures/Sine1k.png similarity index 100% rename from bone-cook-book/06iot/figures/Sine1k.png rename to beaglebone-cookbook/06iot/figures/Sine1k.png diff --git a/bone-cook-book/06iot/figures/Tri10k.png b/beaglebone-cookbook/06iot/figures/Tri10k.png similarity index 100% rename from bone-cook-book/06iot/figures/Tri10k.png rename to beaglebone-cookbook/06iot/figures/Tri10k.png diff --git a/bone-cook-book/06iot/figures/apacheIndex.png b/beaglebone-cookbook/06iot/figures/apacheIndex.png similarity index 100% rename from bone-cook-book/06iot/figures/apacheIndex.png rename to beaglebone-cookbook/06iot/figures/apacheIndex.png diff --git a/bone-cook-book/06iot/figures/apacheNoFiles.png b/beaglebone-cookbook/06iot/figures/apacheNoFiles.png similarity index 100% rename from bone-cook-book/06iot/figures/apacheNoFiles.png rename to beaglebone-cookbook/06iot/figures/apacheNoFiles.png diff --git a/bone-cook-book/06iot/figures/apacheTest.png b/beaglebone-cookbook/06iot/figures/apacheTest.png similarity index 100% rename from bone-cook-book/06iot/figures/apacheTest.png rename to beaglebone-cookbook/06iot/figures/apacheTest.png diff --git a/bone-cook-book/06iot/figures/bone-usr3-led.png b/beaglebone-cookbook/06iot/figures/bone-usr3-led.png similarity index 100% rename from bone-cook-book/06iot/figures/bone-usr3-led.png rename to beaglebone-cookbook/06iot/figures/bone-usr3-led.png diff --git a/bone-cook-book/06iot/figures/cape-headers-serial.png b/beaglebone-cookbook/06iot/figures/cape-headers-serial.png similarity index 100% rename from bone-cook-book/06iot/figures/cape-headers-serial.png rename to beaglebone-cookbook/06iot/figures/cape-headers-serial.png diff --git a/bone-cook-book/06iot/figures/flaskServer.png b/beaglebone-cookbook/06iot/figures/flaskServer.png similarity index 100% rename from bone-cook-book/06iot/figures/flaskServer.png rename to beaglebone-cookbook/06iot/figures/flaskServer.png diff --git a/bone-cook-book/06iot/figures/flaskapp1.png b/beaglebone-cookbook/06iot/figures/flaskapp1.png similarity index 100% rename from bone-cook-book/06iot/figures/flaskapp1.png rename to beaglebone-cookbook/06iot/figures/flaskapp1.png diff --git a/bone-cook-book/06iot/figures/flaskapp2.png b/beaglebone-cookbook/06iot/figures/flaskapp2.png similarity index 100% rename from bone-cook-book/06iot/figures/flaskapp2.png rename to beaglebone-cookbook/06iot/figures/flaskapp2.png diff --git a/bone-cook-book/06iot/figures/flaskapp3.png b/beaglebone-cookbook/06iot/figures/flaskapp3.png similarity index 100% rename from bone-cook-book/06iot/figures/flaskapp3.png rename to beaglebone-cookbook/06iot/figures/flaskapp3.png diff --git a/bone-cook-book/06iot/figures/flotDemo.png b/beaglebone-cookbook/06iot/figures/flotDemo.png similarity index 100% rename from bone-cook-book/06iot/figures/flotDemo.png rename to beaglebone-cookbook/06iot/figures/flotDemo.png diff --git a/bone-cook-book/06iot/figures/jQueryDemo.html b/beaglebone-cookbook/06iot/figures/jQueryDemo.html similarity index 100% rename from bone-cook-book/06iot/figures/jQueryDemo.html rename to beaglebone-cookbook/06iot/figures/jQueryDemo.html diff --git a/bone-cook-book/06iot/figures/jQueryFiddle.png b/beaglebone-cookbook/06iot/figures/jQueryFiddle.png similarity index 100% rename from bone-cook-book/06iot/figures/jQueryFiddle.png rename to beaglebone-cookbook/06iot/figures/jQueryFiddle.png diff --git a/bone-cook-book/06iot/figures/jsfiddleFrameworks.png b/beaglebone-cookbook/06iot/figures/jsfiddleFrameworks.png similarity index 100% rename from bone-cook-book/06iot/figures/jsfiddleFrameworks.png rename to beaglebone-cookbook/06iot/figures/jsfiddleFrameworks.png diff --git a/bone-cook-book/06iot/figures/launchPad.fzz b/beaglebone-cookbook/06iot/figures/launchPad.fzz similarity index 100% rename from bone-cook-book/06iot/figures/launchPad.fzz rename to beaglebone-cookbook/06iot/figures/launchPad.fzz diff --git a/bone-cook-book/06iot/figures/launchPad_bb.png b/beaglebone-cookbook/06iot/figures/launchPad_bb.png similarity index 100% rename from bone-cook-book/06iot/figures/launchPad_bb.png rename to beaglebone-cookbook/06iot/figures/launchPad_bb.png diff --git a/bone-cook-book/06iot/figures/nginxTest.png b/beaglebone-cookbook/06iot/figures/nginxTest.png similarity index 100% rename from bone-cook-book/06iot/figures/nginxTest.png rename to beaglebone-cookbook/06iot/figures/nginxTest.png diff --git a/bone-cook-book/06iot/figures/node-disc-out-setup.png b/beaglebone-cookbook/06iot/figures/node-disc-out-setup.png similarity index 100% rename from bone-cook-book/06iot/figures/node-disc-out-setup.png rename to beaglebone-cookbook/06iot/figures/node-disc-out-setup.png diff --git a/bone-cook-book/06iot/figures/node-disc-out.png b/beaglebone-cookbook/06iot/figures/node-disc-out.png similarity index 100% rename from bone-cook-book/06iot/figures/node-disc-out.png rename to beaglebone-cookbook/06iot/figures/node-disc-out.png diff --git a/bone-cook-book/06iot/figures/node-red.png b/beaglebone-cookbook/06iot/figures/node-red.png similarity index 100% rename from bone-cook-book/06iot/figures/node-red.png rename to beaglebone-cookbook/06iot/figures/node-red.png diff --git a/bone-cook-book/06iot/figures/node-twitter-auth.png b/beaglebone-cookbook/06iot/figures/node-twitter-auth.png similarity index 100% rename from bone-cook-book/06iot/figures/node-twitter-auth.png rename to beaglebone-cookbook/06iot/figures/node-twitter-auth.png diff --git a/bone-cook-book/06iot/figures/node-twitter-auth2.png b/beaglebone-cookbook/06iot/figures/node-twitter-auth2.png similarity index 100% rename from bone-cook-book/06iot/figures/node-twitter-auth2.png rename to beaglebone-cookbook/06iot/figures/node-twitter-auth2.png diff --git a/bone-cook-book/06iot/figures/node-twitter-auth3.png b/beaglebone-cookbook/06iot/figures/node-twitter-auth3.png similarity index 100% rename from bone-cook-book/06iot/figures/node-twitter-auth3.png rename to beaglebone-cookbook/06iot/figures/node-twitter-auth3.png diff --git a/bone-cook-book/06iot/figures/node-twitter-beagle.png b/beaglebone-cookbook/06iot/figures/node-twitter-beagle.png similarity index 100% rename from bone-cook-book/06iot/figures/node-twitter-beagle.png rename to beaglebone-cookbook/06iot/figures/node-twitter-beagle.png diff --git a/bone-cook-book/06iot/figures/node-twitter-debug.png b/beaglebone-cookbook/06iot/figures/node-twitter-debug.png similarity index 100% rename from bone-cook-book/06iot/figures/node-twitter-debug.png rename to beaglebone-cookbook/06iot/figures/node-twitter-debug.png diff --git a/bone-cook-book/06iot/figures/node-twitter.png b/beaglebone-cookbook/06iot/figures/node-twitter.png similarity index 100% rename from bone-cook-book/06iot/figures/node-twitter.png rename to beaglebone-cookbook/06iot/figures/node-twitter.png diff --git a/bone-cook-book/06iot/figures/nodeServer.png b/beaglebone-cookbook/06iot/figures/nodeServer.png similarity index 100% rename from bone-cook-book/06iot/figures/nodeServer.png rename to beaglebone-cookbook/06iot/figures/nodeServer.png diff --git a/bone-cook-book/06iot/figures/nodeTest.png b/beaglebone-cookbook/06iot/figures/nodeTest.png similarity index 100% rename from bone-cook-book/06iot/figures/nodeTest.png rename to beaglebone-cookbook/06iot/figures/nodeTest.png diff --git a/bone-cook-book/06iot/figures/processingDemo.html b/beaglebone-cookbook/06iot/figures/processingDemo.html similarity index 100% rename from bone-cook-book/06iot/figures/processingDemo.html rename to beaglebone-cookbook/06iot/figures/processingDemo.html diff --git a/bone-cook-book/06iot/figures/xbee.fzz b/beaglebone-cookbook/06iot/figures/xbee.fzz similarity index 100% rename from bone-cook-book/06iot/figures/xbee.fzz rename to beaglebone-cookbook/06iot/figures/xbee.fzz diff --git a/bone-cook-book/06iot/figures/xbee_bb.png b/beaglebone-cookbook/06iot/figures/xbee_bb.png similarity index 100% rename from bone-cook-book/06iot/figures/xbee_bb.png rename to beaglebone-cookbook/06iot/figures/xbee_bb.png diff --git a/bone-cook-book/06iot/iot.rst b/beaglebone-cookbook/06iot/iot.rst similarity index 94% rename from bone-cook-book/06iot/iot.rst rename to beaglebone-cookbook/06iot/iot.rst index aad754c7b3f1397a5b0ef7671788558f0906b4f7..22df18ea064dab4b23928be7100ee4f893b18c71 100644 --- a/bone-cook-book/06iot/iot.rst +++ b/beaglebone-cookbook/06iot/iot.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-iot: +.. _beaglebone-cookbook-iot: Internet of Things #################### @@ -13,7 +13,7 @@ wirelessly (:ref:`<networking_wireless>`), or through the USB to a host and then Now that you're online, this chapter offers various things to do with your connection. Accessing Your Host Computer's Files on the Bone ---------------------------------------------- +------------------------------------------------- Problem ********* @@ -65,7 +65,7 @@ and then access: Here, we are accessing the files on the Bone as +debian+. We’ve mounted the entire file system, starting with +/+, so you can access any file. Of course, with great power comes great responsibility, so be careful. Discussion -********* +************ The +sshfs+ command gives you easy access from one computer to another. When you are done, you can unmount the files by using the following commands: @@ -107,7 +107,9 @@ A sample web page (test.html) You will see the web page shown in :ref:`<networking_node_page>`. .. _networking_node_page: -.test.html as served by nginx + +test.html as served by nginx + .. figure:: figures/nginxTest.png :align: center :alt: test.html served by nginx @@ -356,7 +358,8 @@ or http://192.168.7.2:8081/ledRed/off -For the above example, *ledRed* is the “deviceName†and *on* or *off* are examples of possible “actionâ€. Those routes will be identified and properly “workedâ€. The main steps are: +For the above example, *ledRed* is the “deviceName†and *on* or *off* are examples of +possible “actionâ€. Those routes will be identified and properly “workedâ€. The main steps are: * Convert the string “ledREDâ€, for example, on its equivalent GPIO pin. @@ -876,6 +879,7 @@ Installing Node-RED To install Node-RED, run the following commands: .. code-block:: bash + bone$ cd # Change to home directory bone$ git clone https://github.com/node-red/node-red.git bone$ cd node-red/ @@ -955,7 +959,8 @@ Node-RED twitter authorization, step 2 .. [start=5] -- Click the "here" link, as shown in :ref:`<networking_node_twitter_auth2_fig>`, and you'll be taken to Twitter to authorize Node-RED. +- Click the "here" link, as shown in :ref:`<networking_node_twitter_auth2_fig>`, and you'll +be taken to Twitter to authorize Node-RED. - Log in to Twitter and click the "Authorize app" button (:ref:`<networking_node_twitter_auth3_fig>`). @@ -969,7 +974,9 @@ Node-RED Twitter site authorization .. [start=7] -- When you're back to Node-RED, click the Add button, add your Twitter credentials, enter the hashtags to respond to (:ref:`<networking_node_twitter_beagle_fig>`), and then click the Ok pass:[<span class="keep-together">button</span>]. +- When you're back to Node-RED, click the Add button, add your Twitter credentials, +enter the hashtags to respond to (:ref:`<networking_node_twitter_beagle_fig>`), and then +click the Ok pass:[<span class="keep-together">button</span>]. .. _networking_node_twitter_beagle_fig: @@ -1034,7 +1041,7 @@ Click Ok and then Deploy. Test again. The LED will toggle every time the hashtag +#BeagleBone+ is tweeted. With a little more exploring, you should be able to have your Bone ringing a bell or spinning a motor in response to tweets. Discussion -********* +*********** Communicating over a Serial Connection to an Arduino or LaunchPad ------------------------------------------------------------------- @@ -1065,9 +1072,8 @@ Wiring a LaunchPad to a Bone via the common serial port :align: center :alt: MSP430 LaunchPad -Add the code (or _sketch_, as it's called in Arduino-speak) in -:ref:`<js_launchPad_code>` to a file called _launchPad.ino_ -and run it on your LaunchPad. +Add the code (or _sketch_, as it's called in Arduino-speak) in :ref:`<js_launchPad_code>` +to a file called _launchPad.ino_ and run it on your LaunchPad. .. _js_launchPad_code: @@ -1128,7 +1134,7 @@ Code for communicating via the UART (launchPad.js) Table of UART outputs Discussion -********* +************ When you run the script in :ref:`<js_launchPadBeagle_code>`, the Bone opens up the serial port and every second sends a new command, either +r+ or +g+. diff --git a/bone-cook-book/07kernel/code/Makefile b/beaglebone-cookbook/07kernel/code/Makefile similarity index 100% rename from bone-cook-book/07kernel/code/Makefile rename to beaglebone-cookbook/07kernel/code/Makefile diff --git a/bone-cook-book/07kernel/code/Makefile.display b/beaglebone-cookbook/07kernel/code/Makefile.display similarity index 100% rename from bone-cook-book/07kernel/code/Makefile.display rename to beaglebone-cookbook/07kernel/code/Makefile.display diff --git a/bone-cook-book/07kernel/code/hello.c b/beaglebone-cookbook/07kernel/code/hello.c similarity index 100% rename from bone-cook-book/07kernel/code/hello.c rename to beaglebone-cookbook/07kernel/code/hello.c diff --git a/bone-cook-book/07kernel/code/hello.patch b/beaglebone-cookbook/07kernel/code/hello.patch similarity index 100% rename from bone-cook-book/07kernel/code/hello.patch rename to beaglebone-cookbook/07kernel/code/hello.patch diff --git a/bone-cook-book/07kernel/code/helloWorld.c b/beaglebone-cookbook/07kernel/code/helloWorld.c similarity index 100% rename from bone-cook-book/07kernel/code/helloWorld.c rename to beaglebone-cookbook/07kernel/code/helloWorld.c diff --git a/bone-cook-book/07kernel/figures/KernelConfig3.16.png b/beaglebone-cookbook/07kernel/figures/KernelConfig3.16.png similarity index 100% rename from bone-cook-book/07kernel/figures/KernelConfig3.16.png rename to beaglebone-cookbook/07kernel/figures/KernelConfig3.16.png diff --git a/bone-cook-book/07kernel/figures/cape-headers-digitalGPIO7.png b/beaglebone-cookbook/07kernel/figures/cape-headers-digitalGPIO7.png similarity index 100% rename from bone-cook-book/07kernel/figures/cape-headers-digitalGPIO7.png rename to beaglebone-cookbook/07kernel/figures/cape-headers-digitalGPIO7.png diff --git a/bone-cook-book/07kernel/kernel.rst b/beaglebone-cookbook/07kernel/kernel.rst similarity index 88% rename from bone-cook-book/07kernel/kernel.rst rename to beaglebone-cookbook/07kernel/kernel.rst index 359f4c5463a6a8e0fe5ca1946b1faa95f6cfd34a..7b0966b78ba324c9bc7d0e1104b96856f8fdcdb3 100644 --- a/bone-cook-book/07kernel/kernel.rst +++ b/beaglebone-cookbook/07kernel/kernel.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-kernel: +.. _beaglebone-cookbook-kernel: The Kernel ########### @@ -16,11 +16,13 @@ can have multiple kernels on your system at the same time and select from among // TODO is this still true? -.. note:: We assume here that you are logged on to your Bone as +root+ and superuser privileges. You also need to be logged in to your Linux host computer as a nonsuperuser. +.. note:: + We assume here that you are logged on to your Bone as +root+ and superuser privileges. + You also need to be logged in to your Linux host computer as a nonsuperuser. Updating the Kernel ---------------------------------- +--------------------- Problem *********** @@ -109,13 +111,16 @@ When compiling on the Bone, all you need to do is load the Kernel Headers for th .. code-block:: bash -bone$ sudo apt install linux-headers-`uname -r` + bone$ sudo apt install linux-headers-`uname -r` -.. note:: ((("` character (backtick)")))((("backtick (`) character")))The quotes around +`uname -r`+ are backtick characters. On a United States keyboard, the backtick key is to the left of the 1 key. +.. note:: + The quotes around +`uname -r`+ are backtick characters. On a United States keyboard, + the backtick key is to the left of the 1 key. -This took a little more than three minutes on my Bone. The +`uname -r`+ part of the command looks up what version of the kernel you are running and loads the headers for it. +This took a little more than three minutes on my Bone. The +`uname -r`+ part of the command +looks up what version of the kernel you are running and loads the headers for it. Next, add the code in :ref:`<kernel_Makefle>` to a file called _Makefile_. @@ -129,7 +134,9 @@ Simple Kernel Module (_Makefile_) -.. note:: Replace the two instances of +<TAB>+ with a tab character (the key left of the Q key on a United States keyboard). The tab characters are very important to makefiles and must appear as shown. +.. note:: + Replace the two instances of +<TAB>+ with a tab character (the key left of the Q key on a United States keyboard). + The tab characters are very important to makefiles and must appear as shown. Now, compile the kernel module by using the +make+ command: @@ -262,6 +269,7 @@ First, you need to map the +P9+ header location to GPIO number using :ref:`<kern .. _kernel_gpio_map_fig: .Mapping P9_42 header position to GPIO 7 + .. figure:: figures/cape-headers-digitalGPIO7.png :align: center :alt: Mapping Header Position to GPIO Numbers @@ -367,7 +375,9 @@ Solution This is easier than it sounds, thanks to some very powerful scripts. -.. warning:: Be sure to run this recipe on your host computer. The Bone has enough computational power to compile a module or two, but compiling the entire kernel takes lots of time and resourses. +.. warning:: + Be sure to run this recipe on your host computer. The Bone has enough computational + power to compile a module or two, but compiling the entire kernel takes lots of time and resourses. Downloading and Compiling the Kernel @@ -456,12 +466,15 @@ Now, while in the _bb-kernel_ directory, run the following command: Are you 100% sure, on selecting [/dev/sdb] (y/n)? y -The script lists the partitions it sees and asks if you have the correct one. If you are sure, press Y, and the script will uncompress and copy the files to the correct locations on your card. When this is finished, eject your card, plug it into the Bone, and boot it up. Run +uname -a+, and you will see that you are running your compiled kernel. +The script lists the partitions it sees and asks if you have the correct one. +If you are sure, press Y, and the script will uncompress and copy the files to +the correct locations on your card. When this is finished, eject your card, plug +it into the Bone, and boot it up. Run +uname -a+, and you +will see that you are running your compiled kernel. Discussion *********** - .. _kernel_using_cross_compiler: Using the Installed Cross Compiler @@ -470,7 +483,8 @@ Using the Installed Cross Compiler Problem *********** -You have followed the instructions in :ref:`<kernel_compiling>` and want to use the cross compiler it has downloaded. +You have followed the instructions in :ref:`<kernel_compiling>` +and want to use the cross compiler it has downloaded. [TIP] @@ -488,7 +502,10 @@ Then skip down to :ref:`<kernel_skip_to_here>`. Solution *********** -:ref:`<kernel_compiling>` installs a cross compiler, but you need to set up a couple of things so that it can be found. :ref:`<kernel_compiling>` installed the kernel and other tools in a directory called _bb-kernel_. Run the following commands to find the path to the cross compiler: +:ref:`<kernel_compiling>` installs a cross compiler, but you need to set up a +couple of things so that it can be found. :ref:`<kernel_compiling>` installed the +kernel and other tools in a directory called _bb-kernel_. Run the +following commands to find the path to the cross compiler: .. code-block:: bash @@ -498,7 +515,8 @@ Solution gcc-linaro-arm-linux-gnueabihf-4.7-2013.04-20130415_linux.tar.xz -Here, the path to the cross compiler contains the version number of the compiler. Yours might be different from mine. +cd+ into it: +Here, the path to the cross compiler contains the version number +of the compiler. Yours might be different from mine. +cd+ into it: .. code-block:: bash @@ -532,7 +550,8 @@ At this point, we are interested in what's in _bin_: arm-linux-gnueabihf-gdb arm-linux-gnueabihf-strip -What you see are all the cross-development tools. You need to add this directory to the +$PATH+ the shell uses to find the commands it runs: +What you see are all the cross-development tools. You need to add this directory +to the +$PATH+ the shell uses to find the commands it runs: .. code-block:: bash @@ -545,7 +564,9 @@ What you see are all the cross-development tools. You need to add this directory /usr/games:/usr/local/games -The first command displays the path to the directory where the cross-development tools are located. The second shows which directories are searched to find commands to be run. Currently, the cross-development tools are not in the +$PATH+. Let's add it: +The first command displays the path to the directory where the cross-development +tools are located. The second shows which directories are searched to find commands +to be run. Currently, the cross-development tools are not in the +$PATH+. Let's add it: .. code-block:: bash @@ -557,7 +578,8 @@ The first command displays the path to the directory where the cross-development /usr/games:/usr/local/games -.. note:: Those are backtick characters (left of the "1" key on your keyboard) around +pwd+.((("` character (backtick)")))((("backtick (`) character"))) +.. note:: + Those are backtick characters (left of the "1" key on your keyboard) around +pwd+. The second line shows the +$PATH+ now contains the directory with the cross-development tools. @@ -658,7 +680,8 @@ Look in the _hello_ directory to see what was created: Discussion *********** -:ref:`<kernel_building_modules>` shows how to build and install a module, and :ref:`<kernel_create_patch>` shows how to create your own patch file. +:ref:`<kernel_building_modules>` shows how to build and install a module, and :ref:`<kernel_create_patch>` +shows how to create your own patch file. .. _kernel_create_patch: @@ -671,6 +694,7 @@ You made a few changes to the kernel, and you want to share them with your frien Solution *********** + Create a patch file that contains just the changes you have made. Before making your changes, check out a new branch: .. code-block:: bash diff --git a/bone-cook-book/08realtime/code/pushLED.c b/beaglebone-cookbook/08realtime/code/pushLED.c similarity index 100% rename from bone-cook-book/08realtime/code/pushLED.c rename to beaglebone-cookbook/08realtime/code/pushLED.c diff --git a/bone-cook-book/08realtime/code/pushLED.js b/beaglebone-cookbook/08realtime/code/pushLED.js similarity index 100% rename from bone-cook-book/08realtime/code/pushLED.js rename to beaglebone-cookbook/08realtime/code/pushLED.js diff --git a/bone-cook-book/08realtime/code/pushLED.py b/beaglebone-cookbook/08realtime/code/pushLED.py similarity index 100% rename from bone-cook-book/08realtime/code/pushLED.py rename to beaglebone-cookbook/08realtime/code/pushLED.py diff --git a/bone-cook-book/08realtime/code/pushLEDmmap.c b/beaglebone-cookbook/08realtime/code/pushLEDmmap.c similarity index 100% rename from bone-cook-book/08realtime/code/pushLEDmmap.c rename to beaglebone-cookbook/08realtime/code/pushLEDmmap.c diff --git a/bone-cook-book/08realtime/code/pushLEDmmap.h b/beaglebone-cookbook/08realtime/code/pushLEDmmap.h similarity index 100% rename from bone-cook-book/08realtime/code/pushLEDmmap.h rename to beaglebone-cookbook/08realtime/code/pushLEDmmap.h diff --git a/bone-cook-book/08realtime/code/rt/cyclictest.png b/beaglebone-cookbook/08realtime/code/rt/cyclictest.png similarity index 100% rename from bone-cook-book/08realtime/code/rt/cyclictest.png rename to beaglebone-cookbook/08realtime/code/rt/cyclictest.png diff --git a/bone-cook-book/08realtime/code/rt/hist.gen b/beaglebone-cookbook/08realtime/code/rt/hist.gen similarity index 100% rename from bone-cook-book/08realtime/code/rt/hist.gen rename to beaglebone-cookbook/08realtime/code/rt/hist.gen diff --git a/bone-cook-book/08realtime/code/rt/hist.plt b/beaglebone-cookbook/08realtime/code/rt/hist.plt similarity index 100% rename from bone-cook-book/08realtime/code/rt/hist.plt rename to beaglebone-cookbook/08realtime/code/rt/hist.plt diff --git a/bone-cook-book/08realtime/code/rt/install.sh b/beaglebone-cookbook/08realtime/code/rt/install.sh similarity index 100% rename from bone-cook-book/08realtime/code/rt/install.sh rename to beaglebone-cookbook/08realtime/code/rt/install.sh diff --git a/bone-cook-book/08realtime/code/rt/setup.sh b/beaglebone-cookbook/08realtime/code/rt/setup.sh similarity index 100% rename from bone-cook-book/08realtime/code/rt/setup.sh rename to beaglebone-cookbook/08realtime/code/rt/setup.sh diff --git a/bone-cook-book/08realtime/figures/kernel_update.png b/beaglebone-cookbook/08realtime/figures/kernel_update.png similarity index 100% rename from bone-cook-book/08realtime/figures/kernel_update.png rename to beaglebone-cookbook/08realtime/figures/kernel_update.png diff --git a/bone-cook-book/08realtime/figures/pushLED.fzz b/beaglebone-cookbook/08realtime/figures/pushLED.fzz similarity index 100% rename from bone-cook-book/08realtime/figures/pushLED.fzz rename to beaglebone-cookbook/08realtime/figures/pushLED.fzz diff --git a/bone-cook-book/08realtime/figures/pushLED_bb.png b/beaglebone-cookbook/08realtime/figures/pushLED_bb.png similarity index 100% rename from bone-cook-book/08realtime/figures/pushLED_bb.png rename to beaglebone-cookbook/08realtime/figures/pushLED_bb.png diff --git a/bone-cook-book/08realtime/figures/pushLEDmmap.fzz b/beaglebone-cookbook/08realtime/figures/pushLEDmmap.fzz similarity index 100% rename from bone-cook-book/08realtime/figures/pushLEDmmap.fzz rename to beaglebone-cookbook/08realtime/figures/pushLEDmmap.fzz diff --git a/bone-cook-book/08realtime/figures/pushLEDmmap_bb.png b/beaglebone-cookbook/08realtime/figures/pushLEDmmap_bb.png similarity index 100% rename from bone-cook-book/08realtime/figures/pushLEDmmap_bb.png rename to beaglebone-cookbook/08realtime/figures/pushLEDmmap_bb.png diff --git a/bone-cook-book/08realtime/figures/pushLEDpru.fzz b/beaglebone-cookbook/08realtime/figures/pushLEDpru.fzz similarity index 100% rename from bone-cook-book/08realtime/figures/pushLEDpru.fzz rename to beaglebone-cookbook/08realtime/figures/pushLEDpru.fzz diff --git a/bone-cook-book/08realtime/figures/pushLEDpru_bb.png b/beaglebone-cookbook/08realtime/figures/pushLEDpru_bb.png similarity index 100% rename from bone-cook-book/08realtime/figures/pushLEDpru_bb.png rename to beaglebone-cookbook/08realtime/figures/pushLEDpru_bb.png diff --git a/bone-cook-book/08realtime/old/pruSpeak.py b/beaglebone-cookbook/08realtime/old/pruSpeak.py similarity index 100% rename from bone-cook-book/08realtime/old/pruSpeak.py rename to beaglebone-cookbook/08realtime/old/pruSpeak.py diff --git a/bone-cook-book/08realtime/old/thumb.patch b/beaglebone-cookbook/08realtime/old/thumb.patch similarity index 100% rename from bone-cook-book/08realtime/old/thumb.patch rename to beaglebone-cookbook/08realtime/old/thumb.patch diff --git a/bone-cook-book/08realtime/old/xenomaiConfig.png b/beaglebone-cookbook/08realtime/old/xenomaiConfig.png similarity index 100% rename from bone-cook-book/08realtime/old/xenomaiConfig.png rename to beaglebone-cookbook/08realtime/old/xenomaiConfig.png diff --git a/bone-cook-book/08realtime/realtime.rst b/beaglebone-cookbook/08realtime/realtime.rst similarity index 89% rename from bone-cook-book/08realtime/realtime.rst rename to beaglebone-cookbook/08realtime/realtime.rst index c7ed6bb72b8e7bc3e38280eea1cb3f39bd604348..a69b133b4a1727a080f67e9e312c4c0aae364d47 100644 --- a/bone-cook-book/08realtime/realtime.rst +++ b/beaglebone-cookbook/08realtime/realtime.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-realtime: +.. _beaglebone-cookbook-realtime: Real-Time I/O ############### @@ -162,7 +162,8 @@ Your C code isn't responding fast enough to the input signal. You want to read t Solution ********** -The solution is to use a simple utility called +devmem2+, with which you can read and write registers from the command line. +The solution is to use a simple utility called +devmem2+, with which +you can read and write registers from the command line. .. warning:: @@ -179,7 +180,8 @@ First, download and install +devmem2+: bone$ <strong>sudo mv devmem2 /usr/bin</strong> This solution will read a pushbutton attached to +P9_42+ and flash an LED attached to +P9_13+. Note that this is a -change from the previous solutions that makes the code used here much simpler. Wire up your Bone as shown in :ref:`<realtime_pushLEDmmap_fig>`. +change from the previous solutions that makes the code used here much simpler. Wire up your Bone as +shown in :ref:`<realtime_pushLEDmmap_fig>`. .. _realtime_pushLEDmmap_fig: @@ -204,7 +206,11 @@ Finding +P9_13+ at GPIO 31, export GPIO 31 and make it an output: The LED will turn on when +1+ is echoed into +value+ and off when +0+ is echoed. -Now that you know the LED is working, look up its memory address. This is where things get very detailed. First, download the http://bit.ly/1B4Cm45[AM335x Technical Reference Manual]. Look up +GPIO0+ in the Memory Map chapter (sensors). Table 2-2 indicates that +GPIO0+ starts at address +0x44E0_7000+. Then go to Section 25.4.1, "GPIO Registers." This shows that +GPIO_DATAIN+ has an offset of +0x138+, +GPIO_CLEARDATAOUT+ has an offset of +0x190+, and +GPIO_SETDATAOUT+ has an offset of +0x194+. +Now that you know the LED is working, look up its memory address. This is where things get very detailed. +First, download the http://bit.ly/1B4Cm45[AM335x Technical Reference Manual]. Look up +GPIO0+ in the +Memory Map chapter (sensors). Table 2-2 indicates that +GPIO0+ starts at address +0x44E0_7000+. Then +go to Section 25.4.1, "GPIO Registers." This shows that +GPIO_DATAIN+ has an offset of +0x138+, +GPIO_CLEARDATAOUT+ +has an offset of +0x190+, and +GPIO_SETDATAOUT+ has an offset of +0x194+. This means you read from address +0x44E0_7000+ + +0x138+ = +0x44E0_7138+ to see the status of the LED: @@ -216,7 +222,9 @@ This means you read from address +0x44E0_7000+ + +0x138+ = +0x44E0_7138+ to see Value at address 0x44E07138 (0xb6f8e138): 0xC000C404 -The returned value +0xC000C404+ (+1100 0000 0000 0000 1100 0100 0000 0100+ in binary) has bit 31 set to +1+, which means the LED is on. Turn the LED off by writing +0x80000000+ (+1000 0000 0000 0000 0000 0000 0000 0000+ binary) to the +GPIO_CLEARDATA+ register at +0x44E0_7000+ + +0x190+ = +0x44E0_7190+: +The returned value +0xC000C404+ (+1100 0000 0000 0000 1100 0100 0000 0100+ in binary) has bit 31 set to +1+, +which means the LED is on. Turn the LED off by writing +0x80000000+ (+1000 0000 0000 0000 0000 0000 0000 0000+ binary) +to the +GPIO_CLEARDATA+ register at +0x44E0_7000+ + +0x190+ = +0x44E0_7190+: .. code-block:: bash @@ -334,7 +342,8 @@ Solution The Kernel can be compiled with PREEMPT_RT enabled which reduces the delay from when a thread is scheduled to when it runs. -Switching to a PREEMPT_RT kernel is rather easy, but be sure to follow the steps in the Discussion to see how much the latencies are reduced. +Switching to a PREEMPT_RT kernel is rather easy, but be sure to follow the steps in the +Discussion to see how much the latencies are reduced. * First see which kernel you are running: @@ -349,6 +358,7 @@ Switching to a PREEMPT_RT kernel is rather easy, but be sure to follow the steps .. _realtime_kernel_update_fig: The regular and RT kernels + .. figure:: figures/kernel_update.pn :align: centerg :alt: The regular and RT kernels @@ -403,7 +413,8 @@ Cyclictest ********** +cyclictest+ is one tool for measuring the latency from when a thread is schduled and when it runs. -The +code/rt+ directory in the git repo has some scripts for gathering latency data and plotting it. Here's how to run the scripts. +The +code/rt+ directory in the git repo has some scripts for gathering latency data and plotting it. +Here's how to run the scripts. * First look in :ref:`<realtime_install_fig>` to see what to install. @@ -420,7 +431,8 @@ The +code/rt+ directory in the git repo has some scripts for gathering latency d bone$ <strong>time sudo ./hist.gen > nort.hist</strong> -:ref:`<realtime_hist_gen_fig>` shows what's being run. It defaults to 100,000 loops, so it takes a while. The data is saved in +nort.hist+, which stands for no RT histogram. +:ref:`<realtime_hist_gen_fig>` shows what's being run. It defaults to 100,000 loops, so it takes a while. +The data is saved in +nort.hist+, which stands for no RT histogram. .. _realtime_hist_gen_fig: @@ -461,7 +473,9 @@ try running ./setup.sh. If that doesn't work try: Congratulations you are running the RT kernel. -.. note:: If the Beagle appears to be running (the LEDs are flashing) but you are having trouble connecting via +ssh 192.168.7.2+, you can try connecting using the approach shown in :ref:`<tips_FTDI>`. +.. note:: + If the Beagle appears to be running (the LEDs are flashing) but you are having trouble connecting + via +ssh 192.168.7.2+, you can try connecting using the approach shown in :ref:`<tips_FTDI>`. Now run the scipt again (note it's being saved in +rt.hist+ this time.) diff --git a/bone-cook-book/09capes/capes.rst b/beaglebone-cookbook/09capes/capes.rst similarity index 94% rename from bone-cook-book/09capes/capes.rst rename to beaglebone-cookbook/09capes/capes.rst index 02cefbb30133dc3acb99d21c4f9ed16473475193..796d056e9b12f3622d70a739bdcdf14805a1016d 100644 --- a/bone-cook-book/09capes/capes.rst +++ b/beaglebone-cookbook/09capes/capes.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-capes: +.. _beaglebone-cookbook-capes: Capes ##### @@ -18,7 +18,8 @@ A cape is simply a board--often a printed circuit board (PCB) that connects to t and +P9+ headers and follows a few standard pin usages. You can stack up to four capes onto the Bone. Capes can range in size from Bone-sized (:ref:`<capes_miniDisplay>`) to much larger than the Bone (:ref:`<capes_7inLCD>`). -This chapter shows how to attach a couple of capes, move your design to a protoboard, then to a PCB, and finally on to mass production. +This chapter shows how to attach a couple of capes, move your design to a protoboard, then to a PCB, +and finally on to mass production. .. _capes_7inLCD: @@ -39,7 +40,10 @@ from `CircuitCo http://circuitco.com/`_ (shown in :ref:`<capes_7inLCD_fig>`) to .. _capes_7inLCD_fig: -.. note:: Seven-inch LCD from CircuitCo, :ref:`<capes_7inLCD_fig>` was originally posted by CircuitCo at http://elinux.org/File:BeagleBone-LCD7-Front.jpg under a `Creative Commons Attribution-ShareAlike 3.0 Unported License <http://creativecommons.org/licenses/by-sa/3.0/>`_. +.. note:: + Seven-inch LCD from CircuitCo, :ref:`<capes_7inLCD_fig>` was originally posted by CircuitCo + at http://elinux.org/File:BeagleBone-LCD7-Front.jpg under a + `Creative Commons Attribution-ShareAlike 3.0 Unported License <http://creativecommons.org/licenses/by-sa/3.0/>`_. .. figure:: figures/LCD.png :align: center @@ -72,7 +76,7 @@ Discussion .. _capes_miniDisplay: Using a 128 x 128-Pixel LCD Cape ------------------------------ +--------------------------------- Problem *********** @@ -82,7 +86,8 @@ You want to use a small LCD to display things other than the desktop. Solution *********** -The http://bit.ly/1xd0r8p[MiniDisplay] is a 128 x 128 full-color LCD cape that just fits on the Bone, as shown in :ref:`<capes_miniDisplay_fig>`. +The http://bit.ly/1xd0r8p[MiniDisplay] is a 128 x 128 full-color LCD cape that just fits on the +Bone, as shown in :ref:`<capes_miniDisplay_fig>`. .. _capes_miniDisplay_fig: @@ -99,6 +104,7 @@ To make this recipe, you will need: Attach to the Bone and apply power. Then run the following commands: .. code-block:: bash + # From http://elinux.org/CircuitCo:MiniDisplay_Cape # Datasheet: # https://www.crystalfontz.com/products/document/3277/ST7735_V2.1_20100505.pdf @@ -125,6 +131,7 @@ The `manufacturer's website <http://bit.ly/1xd0r8p>`_ suggests enabling SPI0 by Hmmm, something isn't working here. Here's how to see what happened: .. code-block:: bash + bone$ <strong>dmesg | tail</strong> [ 625.334497] bone_capemgr.9: part_number 'BB-SPIDEV0', version 'N/A' [ 625.334673] bone_capemgr.9: slot #11: generic override @@ -278,7 +285,7 @@ Discussion .. _capes_soldering: Moving from a Breadboard to a Protoboard ------------------------------ +----------------------------------------- Problem *********** @@ -579,7 +586,6 @@ Looking at the stimulus you'll generate *before* you connect up your hardware wi Discussion *********** - .. _capes_layout: Laying Out Your Cape PCB @@ -621,7 +627,9 @@ through what's called a _via_. One of the goals of PCB design is to minimize the :ref:`<capes_quickRobo_pcb>` wasn't my first try or my last. My approach was to see what was needed to hook where and move the components around to make it easier for the autorouter to carry out its job. -.. note:: There are entire books and websites dedicated to creating PCB layouts. Look around and see what you can find. http://bit.ly/1wXTLki[SparkFun's guide to making PCBs] is particularly useful. +.. note:: + There are entire books and websites dedicated to creating PCB layouts. Look around and see + what you can find. http://bit.ly/1wXTLki[SparkFun's guide to making PCBs] is particularly useful. Customizing the Board Outline ******************************* @@ -657,7 +665,8 @@ Outline SVG for BeagleBone cape (beaglebone_cape_boardoutline.svg) .. </dl> .. ++++ -The measurements are taken from the http://bit.ly/1C5rSa8[BeagleBone Black System Reference Manual], as shown in :ref:`<capes_dimensions_fig>`. +The measurements are taken from the http://bit.ly/1C5rSa8[BeagleBone Black System Reference Manual], +as shown in :ref:`<capes_dimensions_fig>`. .. _capes_dimensions_fig: @@ -775,6 +784,7 @@ Windows, Mac, and Linux, but it has three `limitations <http://bit.ly/1E5Kh3l>`_ You can install Eagle PCB on your Linux host by using the following command: .. code-block:: bash + host$ <strong>sudo apt install eagle</strong> Reading package lists... Done Building dependency tree @@ -840,7 +850,8 @@ The Eagle start screen Ensure that the correct version number appears. -If you are moving a design from Fritzing to Eagle, see :ref:`<capes_schematic_migration>` for tips on converting from one to the other. +If you are moving a design from Fritzing to Eagle, see :ref:`<capes_schematic_migration>` +for tips on converting from one to the other. *DesignSpark PCB* @@ -851,13 +862,16 @@ but it runs only on Windows. Also, it doesn't seem to have the following of Eagl *Upverter* -In addition to free solutions you run on your desktop, you can also work with a browser-based tool called https://upverter.com/[Upverter]. With Upverter, you can collaborate easily, editing your designs from anywhere on the Internet. It also provides many conversion options and a PCB fabrication service. +In addition to free solutions you run on your desktop, you can also work with a browser-based +tool called https://upverter.com/[Upverter]. With Upverter, you can collaborate easily, editing +your designs from anywhere on the Internet. It also provides many conversion options and a PCB fabrication service. -.. note:: Don't confuse Upverter with Upconverter (:ref:`<capes_schematic_migration>`). Though their names differ by only three letters, they differ greatly in what they do. +.. note:: + Don't confuse Upverter with Upconverter (:ref:`<capes_schematic_migration>`). + Though their names differ by only three letters, they differ greatly in what they do. .. _capes_kicad: - *Kicad* Unlike the previously mentioned free (no-cost) solutions, `Kicad <http://bit.ly/1b2bnBg >`_ @@ -885,6 +899,7 @@ to convert the Fritzing file for the diagram shown in :ref:`<capes_quickRobo_fig I found it necessary to install +libfreetype6+ and +freetype-py+ onto my system, but you might not need this first step: .. code-block:: bash + host$ <strong>sudo apt install libfreetype6</strong> Reading package lists... Done Building dependency tree @@ -902,7 +917,9 @@ I found it necessary to install +libfreetype6+ and +freetype-py+ onto my system, Cleaning up... -.. note:: All these commands are being run on the Linux-based host computer, as shown by the +host$+ prompt. Log in as a normal user, not +root+. +.. note:: + All these commands are being run on the Linux-based host computer, as shown by the +host$+ prompt. + Log in as a normal user, not +root+. Now, install the +schematic-file-converter+ tool: @@ -1037,7 +1054,9 @@ To make this recipe, you will need: * Multimeter (see :ref:`app misc<app_misc>`) * Your other components -Upload your design to http://oshpark.com[OSH Park] and order a few boards. :ref:`<capes_oshpark_share>` shows a resulting http://bit.ly/1MtlzAp[shared project page for the quickBot cape] created in :ref:`<capes_layout>`. We'll proceed to break down how this design was uploaded and shared to enable ordering fabricated PCBs. +Upload your design to http://oshpark.com[OSH Park] and order a few boards. :ref:`<capes_oshpark_share>` +shows a resulting http://bit.ly/1MtlzAp[shared project page for the quickBot cape] created in +:ref:`<capes_layout>`. We'll proceed to break down how this design was uploaded and shared to enable ordering fabricated PCBs. .. _capes_oshpark_share: diff --git a/bone-cook-book/09capes/code/quickBot_motor_test.js b/beaglebone-cookbook/09capes/code/quickBot_motor_test.js similarity index 100% rename from bone-cook-book/09capes/code/quickBot_motor_test.js rename to beaglebone-cookbook/09capes/code/quickBot_motor_test.js diff --git a/bone-cook-book/09capes/figures/BBB_GPS_Cape.fzz b/beaglebone-cookbook/09capes/figures/BBB_GPS_Cape.fzz similarity index 100% rename from bone-cook-book/09capes/figures/BBB_GPS_Cape.fzz rename to beaglebone-cookbook/09capes/figures/BBB_GPS_Cape.fzz diff --git a/bone-cook-book/09capes/figures/BeagleBone_Black_Cape__7fab878f3c590bbf730a21__pcb__cc4cbffa099dd2076377e89426324f3b.svg b/beaglebone-cookbook/09capes/figures/BeagleBone_Black_Cape__7fab878f3c590bbf730a21__pcb__cc4cbffa099dd2076377e89426324f3b.svg similarity index 100% rename from bone-cook-book/09capes/figures/BeagleBone_Black_Cape__7fab878f3c590bbf730a21__pcb__cc4cbffa099dd2076377e89426324f3b.svg rename to beaglebone-cookbook/09capes/figures/BeagleBone_Black_Cape__7fab878f3c590bbf730a21__pcb__cc4cbffa099dd2076377e89426324f3b.svg diff --git a/bone-cook-book/09capes/figures/Boris128.jpg b/beaglebone-cookbook/09capes/figures/Boris128.jpg similarity index 100% rename from bone-cook-book/09capes/figures/Boris128.jpg rename to beaglebone-cookbook/09capes/figures/Boris128.jpg diff --git a/bone-cook-book/09capes/figures/EagleLicense.png b/beaglebone-cookbook/09capes/figures/EagleLicense.png similarity index 100% rename from bone-cook-book/09capes/figures/EagleLicense.png rename to beaglebone-cookbook/09capes/figures/EagleLicense.png diff --git a/bone-cook-book/09capes/figures/Fritzing_Inspector.png b/beaglebone-cookbook/09capes/figures/Fritzing_Inspector.png similarity index 100% rename from bone-cook-book/09capes/figures/Fritzing_Inspector.png rename to beaglebone-cookbook/09capes/figures/Fritzing_Inspector.png diff --git a/bone-cook-book/09capes/figures/Fritzing_layers.png b/beaglebone-cookbook/09capes/figures/Fritzing_layers.png similarity index 100% rename from bone-cook-book/09capes/figures/Fritzing_layers.png rename to beaglebone-cookbook/09capes/figures/Fritzing_layers.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz160.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz160.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz160.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz160.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz161.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz161.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz161.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz161.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz162.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz162.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz162.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz162.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz163.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz163.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz163.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz163.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz164.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz164.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz164.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz164.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz166.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz166.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz166.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz166.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz167.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz167.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz167.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz167.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz168.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz168.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz168.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz168.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz169.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz169.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz169.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz169.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz170.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz170.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz170.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz170.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz171.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz171.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz171.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz171.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz172.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz172.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz172.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz172.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz173.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz173.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz173.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz173.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz174.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz174.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz174.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz174.png diff --git a/bone-cook-book/09capes/figures/Google ChromeScreenSnapz175.png b/beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz175.png similarity index 100% rename from bone-cook-book/09capes/figures/Google ChromeScreenSnapz175.png rename to beaglebone-cookbook/09capes/figures/Google ChromeScreenSnapz175.png diff --git a/bone-cook-book/09capes/figures/LCD b/beaglebone-cookbook/09capes/figures/LCD similarity index 100% rename from bone-cook-book/09capes/figures/LCD rename to beaglebone-cookbook/09capes/figures/LCD diff --git a/bone-cook-book/09capes/figures/LCD.png b/beaglebone-cookbook/09capes/figures/LCD.png similarity index 100% rename from bone-cook-book/09capes/figures/LCD.png rename to beaglebone-cookbook/09capes/figures/LCD.png diff --git a/bone-cook-book/09capes/figures/LCD7Desktop.png b/beaglebone-cookbook/09capes/figures/LCD7Desktop.png similarity index 100% rename from bone-cook-book/09capes/figures/LCD7Desktop.png rename to beaglebone-cookbook/09capes/figures/LCD7Desktop.png diff --git a/bone-cook-book/09capes/figures/LCD7back.png b/beaglebone-cookbook/09capes/figures/LCD7back.png similarity index 100% rename from bone-cook-book/09capes/figures/LCD7back.png rename to beaglebone-cookbook/09capes/figures/LCD7back.png diff --git a/bone-cook-book/09capes/figures/MiniDisplay-A1.jpg b/beaglebone-cookbook/09capes/figures/MiniDisplay-A1.jpg similarity index 100% rename from bone-cook-book/09capes/figures/MiniDisplay-A1.jpg rename to beaglebone-cookbook/09capes/figures/MiniDisplay-A1.jpg diff --git a/bone-cook-book/09capes/figures/audioCape.png b/beaglebone-cookbook/09capes/figures/audioCape.png similarity index 100% rename from bone-cook-book/09capes/figures/audioCape.png rename to beaglebone-cookbook/09capes/figures/audioCape.png diff --git a/bone-cook-book/09capes/figures/beaglebone_cape_boardoutline.png b/beaglebone-cookbook/09capes/figures/beaglebone_cape_boardoutline.png similarity index 100% rename from bone-cook-book/09capes/figures/beaglebone_cape_boardoutline.png rename to beaglebone-cookbook/09capes/figures/beaglebone_cape_boardoutline.png diff --git a/bone-cook-book/09capes/figures/beaglebone_cape_boardoutline.svg b/beaglebone-cookbook/09capes/figures/beaglebone_cape_boardoutline.svg similarity index 100% rename from bone-cook-book/09capes/figures/beaglebone_cape_boardoutline.svg rename to beaglebone-cookbook/09capes/figures/beaglebone_cape_boardoutline.svg diff --git a/bone-cook-book/09capes/figures/boneOutline.xml b/beaglebone-cookbook/09capes/figures/boneOutline.xml similarity index 100% rename from bone-cook-book/09capes/figures/boneOutline.xml rename to beaglebone-cookbook/09capes/figures/boneOutline.xml diff --git a/bone-cook-book/09capes/figures/breadboard.png b/beaglebone-cookbook/09capes/figures/breadboard.png similarity index 100% rename from bone-cook-book/09capes/figures/breadboard.png rename to beaglebone-cookbook/09capes/figures/breadboard.png diff --git a/bone-cook-book/09capes/figures/circuithub_part_matching.png b/beaglebone-cookbook/09capes/figures/circuithub_part_matching.png similarity index 100% rename from bone-cook-book/09capes/figures/circuithub_part_matching.png rename to beaglebone-cookbook/09capes/figures/circuithub_part_matching.png diff --git a/bone-cook-book/09capes/figures/circuithub_quote.png b/beaglebone-cookbook/09capes/figures/circuithub_quote.png similarity index 100% rename from bone-cook-book/09capes/figures/circuithub_quote.png rename to beaglebone-cookbook/09capes/figures/circuithub_quote.png diff --git a/bone-cook-book/09capes/figures/convert-fritzing-to-eagle.sh b/beaglebone-cookbook/09capes/figures/convert-fritzing-to-eagle.sh similarity index 100% rename from bone-cook-book/09capes/figures/convert-fritzing-to-eagle.sh rename to beaglebone-cookbook/09capes/figures/convert-fritzing-to-eagle.sh diff --git a/bone-cook-book/09capes/figures/convert-fritzing-to-kicad.sh b/beaglebone-cookbook/09capes/figures/convert-fritzing-to-kicad.sh similarity index 100% rename from bone-cook-book/09capes/figures/convert-fritzing-to-kicad.sh rename to beaglebone-cookbook/09capes/figures/convert-fritzing-to-kicad.sh diff --git a/bone-cook-book/09capes/figures/eagle3.png b/beaglebone-cookbook/09capes/figures/eagle3.png similarity index 100% rename from bone-cook-book/09capes/figures/eagle3.png rename to beaglebone-cookbook/09capes/figures/eagle3.png diff --git a/bone-cook-book/09capes/figures/eagle7.png b/beaglebone-cookbook/09capes/figures/eagle7.png similarity index 100% rename from bone-cook-book/09capes/figures/eagle7.png rename to beaglebone-cookbook/09capes/figures/eagle7.png diff --git a/bone-cook-book/09capes/figures/fritzing1.png b/beaglebone-cookbook/09capes/figures/fritzing1.png similarity index 100% rename from bone-cook-book/09capes/figures/fritzing1.png rename to beaglebone-cookbook/09capes/figures/fritzing1.png diff --git a/bone-cook-book/09capes/figures/fritzing2.png b/beaglebone-cookbook/09capes/figures/fritzing2.png similarity index 100% rename from bone-cook-book/09capes/figures/fritzing2.png rename to beaglebone-cookbook/09capes/figures/fritzing2.png diff --git a/bone-cook-book/09capes/figures/fritzing3.png b/beaglebone-cookbook/09capes/figures/fritzing3.png similarity index 100% rename from bone-cook-book/09capes/figures/fritzing3.png rename to beaglebone-cookbook/09capes/figures/fritzing3.png diff --git a/bone-cook-book/09capes/figures/fritzing4.png b/beaglebone-cookbook/09capes/figures/fritzing4.png similarity index 100% rename from bone-cook-book/09capes/figures/fritzing4.png rename to beaglebone-cookbook/09capes/figures/fritzing4.png diff --git a/bone-cook-book/09capes/figures/miniDisplay_Boris.png b/beaglebone-cookbook/09capes/figures/miniDisplay_Boris.png similarity index 100% rename from bone-cook-book/09capes/figures/miniDisplay_Boris.png rename to beaglebone-cookbook/09capes/figures/miniDisplay_Boris.png diff --git a/bone-cook-book/09capes/figures/miniDisplay_Cape_pcb.png b/beaglebone-cookbook/09capes/figures/miniDisplay_Cape_pcb.png similarity index 100% rename from bone-cook-book/09capes/figures/miniDisplay_Cape_pcb.png rename to beaglebone-cookbook/09capes/figures/miniDisplay_Cape_pcb.png diff --git a/bone-cook-book/09capes/figures/miniDisplay_Cape_schem.png b/beaglebone-cookbook/09capes/figures/miniDisplay_Cape_schem.png similarity index 100% rename from bone-cook-book/09capes/figures/miniDisplay_Cape_schem.png rename to beaglebone-cookbook/09capes/figures/miniDisplay_Cape_schem.png diff --git a/bone-cook-book/09capes/figures/quickBot-cache-cache.lib b/beaglebone-cookbook/09capes/figures/quickBot-cache-cache.lib similarity index 100% rename from bone-cook-book/09capes/figures/quickBot-cache-cache.lib rename to beaglebone-cookbook/09capes/figures/quickBot-cache-cache.lib diff --git a/bone-cook-book/09capes/figures/quickBot-eagle.sch b/beaglebone-cookbook/09capes/figures/quickBot-eagle.sch similarity index 100% rename from bone-cook-book/09capes/figures/quickBot-eagle.sch rename to beaglebone-cookbook/09capes/figures/quickBot-eagle.sch diff --git a/bone-cook-book/09capes/figures/quickBot-eaglexml.sch b/beaglebone-cookbook/09capes/figures/quickBot-eaglexml.sch similarity index 100% rename from bone-cook-book/09capes/figures/quickBot-eaglexml.sch rename to beaglebone-cookbook/09capes/figures/quickBot-eaglexml.sch diff --git a/bone-cook-book/09capes/figures/quickBot-schematic.json b/beaglebone-cookbook/09capes/figures/quickBot-schematic.json similarity index 100% rename from bone-cook-book/09capes/figures/quickBot-schematic.json rename to beaglebone-cookbook/09capes/figures/quickBot-schematic.json diff --git a/bone-cook-book/09capes/figures/quickBot-upverter.PNG b/beaglebone-cookbook/09capes/figures/quickBot-upverter.PNG similarity index 100% rename from bone-cook-book/09capes/figures/quickBot-upverter.PNG rename to beaglebone-cookbook/09capes/figures/quickBot-upverter.PNG diff --git a/bone-cook-book/09capes/figures/quickBot.fzz b/beaglebone-cookbook/09capes/figures/quickBot.fzz similarity index 100% rename from bone-cook-book/09capes/figures/quickBot.fzz rename to beaglebone-cookbook/09capes/figures/quickBot.fzz diff --git a/bone-cook-book/09capes/figures/quickBot.upv b/beaglebone-cookbook/09capes/figures/quickBot.upv similarity index 100% rename from bone-cook-book/09capes/figures/quickBot.upv rename to beaglebone-cookbook/09capes/figures/quickBot.upv diff --git a/bone-cook-book/09capes/figures/quickBot5IR.fzz b/beaglebone-cookbook/09capes/figures/quickBot5IR.fzz similarity index 100% rename from bone-cook-book/09capes/figures/quickBot5IR.fzz rename to beaglebone-cookbook/09capes/figures/quickBot5IR.fzz diff --git a/bone-cook-book/09capes/figures/quickBot_bb.png b/beaglebone-cookbook/09capes/figures/quickBot_bb.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_bb.png rename to beaglebone-cookbook/09capes/figures/quickBot_bb.png diff --git a/bone-cook-book/09capes/figures/quickBot_eaglexml.png b/beaglebone-cookbook/09capes/figures/quickBot_eaglexml.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_eaglexml.png rename to beaglebone-cookbook/09capes/figures/quickBot_eaglexml.png diff --git a/bone-cook-book/09capes/figures/quickBot_fritzing_export.png b/beaglebone-cookbook/09capes/figures/quickBot_fritzing_export.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_fritzing_export.png rename to beaglebone-cookbook/09capes/figures/quickBot_fritzing_export.png diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers.zip b/beaglebone-cookbook/09capes/figures/quickBot_gerbers.zip similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers.zip rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers.zip diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_contour.gm1 b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_contour.gm1 similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_contour.gm1 rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_contour.gm1 diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_copperBottom.gbl b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_copperBottom.gbl similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_copperBottom.gbl rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_copperBottom.gbl diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_copperTop.gtl b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_copperTop.gtl similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_copperTop.gtl rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_copperTop.gtl diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_drill.txt b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_drill.txt similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_drill.txt rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_drill.txt diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_bottom_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_copper_top_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_bottom_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_mask_top_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_bottom_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_paste_mask_top_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_bottom_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top_mirror.pdf b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top_mirror.pdf similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top_mirror.pdf rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_etch_silk_top_mirror.pdf diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_maskBottom.gbs b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_maskBottom.gbs similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_maskBottom.gbs rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_maskBottom.gbs diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_maskTop.gts b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_maskTop.gts similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_maskTop.gts rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_maskTop.gts diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_pnp.txt b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_pnp.txt similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_pnp.txt rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_pnp.txt diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_silkBottom.gbo b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_silkBottom.gbo similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_silkBottom.gbo rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_silkBottom.gbo diff --git a/bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_silkTop.gto b/beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_silkTop.gto similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_gerbers/quickBot_silkTop.gto rename to beaglebone-cookbook/09capes/figures/quickBot_gerbers/quickBot_silkTop.gto diff --git a/bone-cook-book/09capes/figures/quickBot_kicad.sch b/beaglebone-cookbook/09capes/figures/quickBot_kicad.sch similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_kicad.sch rename to beaglebone-cookbook/09capes/figures/quickBot_kicad.sch diff --git a/bone-cook-book/09capes/figures/quickBot_motor_kickback.JPG b/beaglebone-cookbook/09capes/figures/quickBot_motor_kickback.JPG similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_motor_kickback.JPG rename to beaglebone-cookbook/09capes/figures/quickBot_motor_kickback.JPG diff --git a/bone-cook-book/09capes/figures/quickBot_motor_test_cloud9.png b/beaglebone-cookbook/09capes/figures/quickBot_motor_test_cloud9.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_motor_test_cloud9.png rename to beaglebone-cookbook/09capes/figures/quickBot_motor_test_cloud9.png diff --git a/bone-cook-book/09capes/figures/quickBot_motor_test_scope.JPG b/beaglebone-cookbook/09capes/figures/quickBot_motor_test_scope.JPG similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_motor_test_scope.JPG rename to beaglebone-cookbook/09capes/figures/quickBot_motor_test_scope.JPG diff --git a/bone-cook-book/09capes/figures/quickBot_motors.jpg b/beaglebone-cookbook/09capes/figures/quickBot_motors.jpg similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_motors.jpg rename to beaglebone-cookbook/09capes/figures/quickBot_motors.jpg diff --git a/bone-cook-book/09capes/figures/quickBot_on_OSHPark.png b/beaglebone-cookbook/09capes/figures/quickBot_on_OSHPark.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_on_OSHPark.png rename to beaglebone-cookbook/09capes/figures/quickBot_on_OSHPark.png diff --git a/bone-cook-book/09capes/figures/quickBot_oshpark_share.png b/beaglebone-cookbook/09capes/figures/quickBot_oshpark_share.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_oshpark_share.png rename to beaglebone-cookbook/09capes/figures/quickBot_oshpark_share.png diff --git a/bone-cook-book/09capes/figures/quickBot_pcb.JPG b/beaglebone-cookbook/09capes/figures/quickBot_pcb.JPG similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_pcb.JPG rename to beaglebone-cookbook/09capes/figures/quickBot_pcb.JPG diff --git a/bone-cook-book/09capes/figures/quickBot_pcb.png b/beaglebone-cookbook/09capes/figures/quickBot_pcb.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_pcb.png rename to beaglebone-cookbook/09capes/figures/quickBot_pcb.png diff --git a/bone-cook-book/09capes/figures/quickBot_play.fzz b/beaglebone-cookbook/09capes/figures/quickBot_play.fzz similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_play.fzz rename to beaglebone-cookbook/09capes/figures/quickBot_play.fzz diff --git a/bone-cook-book/09capes/figures/quickBot_schem.png b/beaglebone-cookbook/09capes/figures/quickBot_schem.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_schem.png rename to beaglebone-cookbook/09capes/figures/quickBot_schem.png diff --git a/bone-cook-book/09capes/figures/quickBot_schemRaw.png b/beaglebone-cookbook/09capes/figures/quickBot_schemRaw.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_schemRaw.png rename to beaglebone-cookbook/09capes/figures/quickBot_schemRaw.png diff --git a/bone-cook-book/09capes/figures/quickBot_schemZoom.png b/beaglebone-cookbook/09capes/figures/quickBot_schemZoom.png similarity index 100% rename from bone-cook-book/09capes/figures/quickBot_schemZoom.png rename to beaglebone-cookbook/09capes/figures/quickBot_schemZoom.png diff --git a/bone-cook-book/09capes/figures/roboCape.fzz b/beaglebone-cookbook/09capes/figures/roboCape.fzz similarity index 100% rename from bone-cook-book/09capes/figures/roboCape.fzz rename to beaglebone-cookbook/09capes/figures/roboCape.fzz diff --git a/bone-cook-book/09capes/figures/srm_cape_dimensions.png b/beaglebone-cookbook/09capes/figures/srm_cape_dimensions.png similarity index 100% rename from bone-cook-book/09capes/figures/srm_cape_dimensions.png rename to beaglebone-cookbook/09capes/figures/srm_cape_dimensions.png diff --git a/bone-cook-book/09capes/figures/stacking_headers.JPG b/beaglebone-cookbook/09capes/figures/stacking_headers.JPG similarity index 100% rename from bone-cook-book/09capes/figures/stacking_headers.JPG rename to beaglebone-cookbook/09capes/figures/stacking_headers.JPG diff --git a/bone-cook-book/09capes/figures/urls.txt b/beaglebone-cookbook/09capes/figures/urls.txt similarity index 100% rename from bone-cook-book/09capes/figures/urls.txt rename to beaglebone-cookbook/09capes/figures/urls.txt diff --git a/bone-cook-book/10parts/parts.rst b/beaglebone-cookbook/10parts/parts.rst similarity index 98% rename from bone-cook-book/10parts/parts.rst rename to beaglebone-cookbook/10parts/parts.rst index 546860fb45b214c6cc85114272a595ae86475d82..ea216041df7dc647a03890c9f5a25de721047f3c 100644 --- a/bone-cook-book/10parts/parts.rst +++ b/beaglebone-cookbook/10parts/parts.rst @@ -1,4 +1,4 @@ -.. _bone-cook-book-parts: +.. _beaglebone-cookbook-parts: Parts and Suppliers #################### diff --git a/bone-cook-book/index.rst b/beaglebone-cookbook/index.rst similarity index 95% rename from bone-cook-book/index.rst rename to beaglebone-cookbook/index.rst index ebc0977f2a91739024d2bba730277380c1359c32..f8f394aeb8c0637aba811b634770c95607c4ea28 100644 --- a/bone-cook-book/index.rst +++ b/beaglebone-cookbook/index.rst @@ -22,4 +22,3 @@ A cookbook for programming Beagles 07kernel/kernel.rst 08realtime/realtime.rst 09capes/capes.rst - 10parts/parts.rst diff --git a/conf.py b/conf.py index 9d653802b5b7cc0a8fef39fbc171748c2db6c077..7a53167563f8d3fb10c31fe2cb90bf2099e3f5fa 100644 --- a/conf.py +++ b/conf.py @@ -25,11 +25,15 @@ author = 'BeagleBoard.org Foundation' # -- General configuration --------------------------------------------------- extensions = [ - "sphinxcontrib.rsvgconverter", + "sphinxcontrib.rsvgconverter", "sphinx_design" ] templates_path = ['_templates'] +source_suffix = '.rst' +numfig = True +navigation_with_keys = True + # List of patterns, relative to source directory, that match files and # directories to ignore when looking for source files. # This pattern also affects html_static_path and html_extra_path. @@ -43,7 +47,7 @@ html_show_sphinx = False html_theme_path = [sphinx_rtd_theme.get_html_theme_path()] html_theme_options = { "logo_only": True, - "prev_next_buttons_location": None, + 'prev_next_buttons_location': 'bottom', } html_title = "BeagleBoard Documentation" html_logo = str(BBDOCS_BASE / "_static" / "images" / "logo.svg") @@ -117,4 +121,4 @@ latex_documents = [ def setup(app): # theme customizations - app.add_css_file("css/custom.css") \ No newline at end of file + app.add_css_file("css/custom.css") diff --git a/index-tex.rst b/index-tex.rst index 6a93bc866ea5f031ead06473d1f70806dd51d417..da8ac563c3156e28064968075866eb2711aeea2b 100644 --- a/index-tex.rst +++ b/index-tex.rst @@ -17,5 +17,5 @@ BeagleBoard Docs beaglebone-blue/index.rst beagleconnect/index.rst simppru/index.rst - bone-cook-book/index.rst - + beaglebone-cookbook/index.rst + pru-cookbook/index.rst diff --git a/index.rst b/index.rst index 9b7f6f0bf8aaea134c3f0b0eb2f8ed69a1b6ab3e..3d9e4de14601ffdfb1666eef43a835d87f2830a1 100644 --- a/index.rst +++ b/index.rst @@ -39,7 +39,8 @@ Sections :maxdepth: 1 :caption: Books - bone-cook-book/index.rst + beaglebone-cookbook/index.rst + pru-cookbook/index.rst Indices and tables ############################ diff --git a/pru-cookbook/01case/case.rst b/pru-cookbook/01case/case.rst new file mode 100644 index 0000000000000000000000000000000000000000..8082060ee5bcc66e3c66810087ca20a35deb448f --- /dev/null +++ b/pru-cookbook/01case/case.rst @@ -0,0 +1,1122 @@ +.. _pru-cookbook-case: + +Case Studies - Introduction +############################# + +It's an exciting time to be making projects that use embedded processors. +Make:'s `Makers' Guide to Boards <https://makezine.com/comparison/boards/>`_ shows +many of the options that are available and groups them into different types. +*Single board computers* (SBCs) generally run Linux on some sort of `ARM <https://www.arm.com/>`_ +processor. Examples are the BeagleBoard and the Raspberry Pi. Another type +is the *microcontroller*, of which the `Arduino <https://www.arduino.cc/>`_ is +popular. + +The SBCs are used because they have an operating system to manage files, I/O, +and schedule when things are run, all while possibly talking to the Internet. +Microcontrollers shine when things +being interfaced require careful timing and can't afford to have an OS preempt an +operation. + +But what if you have a project that needs the flexibility of an OS and the timing +of a microcontroller? This is where the BeagleBoard excels since it has both +an ARM procssor running Linux and two footnote:[Four if you are on the BeagleBone AI]; +**P**rogrammable **R**eal-Time **U**nits (PRUs). +The PRUs have 32-bit cores which run +independently of the ARM processor, therefore they can +be programmed to respond quickly to inputs and produce very precisely timed +outputs. + +There are many :ref:`Projects <pru-cookbook-projects>` that use the PRU. +They are able to do things that can't be done with just a SBC or just a microcontroller. +Here we present some case studies that give a high-level view of using the PRUs. In later +chapters you will see the details of how they work. + +Here we present: + +.. TODO Switch from LEDscape to FPP + +* `Robotics Control Library <http://strawsondesign.com/docs/roboticscape/>`_ +* `BeagleLogic <https://github.com/abhishek-kakkar/BeagleLogic/wiki>`_ +* `NeoPixels -- 5050 RGB LEDs with Integrated Drivers (Falcon Christmas) <http://falconchristmas.com>`_ +* `RGB LED Matrix (Falcon Christmas) <http://falconchristmas.com>`_ +* `simpPRU -- A python-like language for programming the PRUs`_ <https://github.com/VedantParanjape/simpPRU> +.. * `MachineKit <http://www.machinekit.io/>`_ +.. * `ArduPilot <http://ardupilot.org/>, <http://ardupilot.org/dev/docs/beaglepilot.html>`_ +.. * `BeagleScope <https://github.com/ZeekHuge/BeagleScope>`_ + +The following are resources used in this chapter. + +.. admonition:: Resources + + * `Pocket Beagle System Reference Manual https://github.com/beagleboard/pocketbeagle/wiki/System-Reference-Manual#673_PRUICSS_Pin_Access`_ + * `BeagleBone Black P8 Header Table https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP8HeaderTable.pdf`_ + * `BeagleBone Black P9 Header Table https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP9HeaderTable.pdf`_ + * `BeagleBone AI System Reference Manual https://github.com/beagleboard/beaglebone-ai/wiki/System-Reference-Manual`_ + + +Robotics Control Library +------------------------- + +Robotics is an embedded application that often requires both an SBC to control the +high-level tasks (such as path planning, line following, communicating with the user) +*and* a microcontroller to handle the low-level tasks (such as telling motors how fast +to turn, or how to balance in response to an IMU input). The +`EduMIP <https://www.ucsdrobotics.org/edumip>`_ balancing +`robot <https://www.hackster.io/edumip/edumip-13a29c>`_ +demonstrates that by using the PRU, the Blue can handle both the high +and low -level tasks without an additional microcontroller. The EduMIP is shown +in :ref:`<case_blue>`. + +.. _case_blue: + +Blue balancing +~~~~~~~~~~~~~~~ + +.. figure:: figures/blue.png + :align: center + :alt: Blue balancing + +The `Robotics Control Library <http://strawsondesign.com/docs/roboticscape/>`_ is a +package that is already installed on the Beagle +that contains a C library and example/testing programs. It uses the PRU to extend the +real-time hardware of the Bone by adding eight addional servo channels and one +addition real-time encoder input. + +The following examples show how easy it is to use the PRU for robotics. + +Controlling Eight Servos +************************* + +Problem +~~~~~~~~ + +You need to control eight servos, but the Bone doesn't have enough pulse width +modulation (PWM) channels +and you don't want to add hardware. + +Solution +~~~~~~~~~ + +The Robotics Control Library provides eight additional PWM channels +via the PRU that can be used out of the box. + +.. note:: + The I/O pins on the Beagles have a mutliplexer that lets you select what I/O + appears on a given pin. The Blue has the mux already configured to to run these + examples. Follow the instructions in + :ref:`../03details/details.html#details_configure_servos, Configuring Pins for Controlling Servos` + to configure the pins for the Black and the Pocket. + + +.. * TODO - verify these commands + +Just run: + +.. code-block:: bash + + bone$ sudo rc_test_servos -f 10 -p 1.5 + +The ``-f 10`` says to use a frequency of 10 Hz and the ``-p 1.5`` says to set the position to ``1.5``. The range of positions is +``-1.5`` to ``1.5``. Run ``rc_test_servos -h`` to see all the options. + +.. code-block:: bash + + bone$ rc_test_servos -h + + Options + -c {channel} Specify one channel from 1-8. + Otherwise all channels will be driven equally + -f {hz} Specify pulse frequency, otherwise 50hz is used + -p {position} Drive servo to a position between -1.5 & 1.5 + -w {width_us} Send pulse width in microseconds (us) + -s {limit} Sweep servo back/forth between +- limit + Limit can be between 0 & 1.5 + -r {ch} Use DSM radio channel {ch} to control servo + -h Print this help messege + + sample use to center servo channel 1: + rc_test_servo -c 1 -p 0.0 + +Discussion +~~~~~~~~~~~ + +The BeagleBone Blue sends these eight outputs to it's servo channels. The others use the pins shown in the +:ref:`case__register_to_pin_table`. + +.. _case__register_to_pin_table: + +PRU register to pin table +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-----------+--------+---------+----------+-------+ + |PRU pin |Blue pin|Black pin|Pocket pin|AI pin | + +===========+========+=========+==========+=======+ + |pru1_r30_8 |1 |P8_27 |P2.35 | | + +-----------+--------+---------+----------+-------+ + |pru1_r30_10|2 |P8_28 |P1.35 |P9_42 | + +-----------+--------+---------+----------+-------+ + |pru1_r30_9 |3 |P8_29 |P1.02 |P8_14 | + +-----------+--------+---------+----------+-------+ + |pru1_r30_11|4 |P8_30 |P1.04 |P9_27 | + +-----------+--------+---------+----------+-------+ + |pru1_r30_6 |5 |P8_39 | |P8_19 | + +-----------+--------+---------+----------+-------+ + |pru1_r30_7 |6 |P8_40 | |P8_13 | + +-----------+--------+---------+----------+-------+ + |pru1_r30_4 |7 |P8_41 | | | + +-----------+--------+---------+----------+-------+ + |pru1_r30_5 |8 |P8_42 | |P8_18 | + +-----------+--------+---------+----------+-------+ + + +You can find these details in the +`P8 Header Table <https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP8HeaderTable.pdf>`_, +`P9 Header Table <https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP9HeaderTable.pdf>`_, +`Pocket Beagle System Reference Manual <https://github.com/beagleboard/pocketbeagle/wiki/System-Reference-Manual#673_PRUICSS_Pin_Access>`_ +(Here is a more usable version of the `table <https://docs.google.com/spreadsheets/d/1FRGvYOyW1RiNSEVprvstfJAVeapnASgDXHtxeDOjgqw/edit?usp=sharing>`_.) +and +`BeagleBone AI System Reference Manual <https://github.com/beagleboard/beaglebone-ai/wiki/System-Reference-Manual>`_. +(Here is a more usable version of the `table <https://docs.google.com/spreadsheets/d/1dFSBVem86vAUD7MLXvqdS-N0Efi8_g_O1iTqzql8DAo/edit#gid=0>`_.) + + +Be default the PRUs are already loaded with the code needed to run the +servos. All you have to do is run the command. + +.. [/opt/source/Robotics_Cape_Installer/pru_firmware/src/pru1-servo.asm] + +Controlling Individual Servos +****************************** + +Problem +~~~~~~~~~ + +``rc_test_servos`` is nice, but I need to control the servos individually. + +Solution +~~~~~~~~~ + +You can modify ``rc_test_servos.c``. You'll find it on the bone online at +https://github.com/beagleboard/librobotcontrol/blob/master/examples/src/rc_test_servos.c. + +Just past line 250 you'll find a ``while`` loop that has calls to ``rc_servo_send_pulse_normalized(ch,servo_pos)`` and +``rc_servo_send_pulse_us(ch, width_us)``. The first call sets the pulse width relative to the pulse period; the other +sets the width to an absolute time. Use whichever works for you. + +Controlling More Than Eight Channels +************************************* + +Problem +~~~~~~~~~~ + +I need more than eight PWM channels, or I need less jitter on the off time. + +Solution +~~~~~~~~~~ + +This is a more advanced problem and required reprograming the PRUs. See +:ref:`../05blocks/blocks.html#blocks_pwm, PWM Generator` for an example. + +Reading Hardware Encoders +************************** + +Problem +~~~~~~~~~~ + +I want to use four encoders to measure four motors, but I only see hardware for three. + +Solution +~~~~~~~~~~ + +The forth encoder can be implemented on the PRU. If you run ``rc_test_encoders_eqep`` on the Blue, you will see the output of +encoders E1-E3 which are connected to the eEQP hardware. + +.. code-block:: bash + + bone$ *rc_test_encoders_eqep* + + Raw encoder positions + E1 | E2 | E3 | + 0 | 0 | 0 |^C + +You can also access these hardware encoders on the Black and Pocket using the +pins shown in :ref:`case_pin_mapping`. + +.. _case_pin_mapping: + +eQEP to pin mapping +~~~~~~~~~~~~~~~~~~~~ +.. table:: + + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + |eQEP|Blue pin|Black pin A|Black pin B|AI pin A|AI pin B|Pocket pin A|Pocket pin B | + +====+========+===========+===========+========+========+============+=============+ + |0 |E1 |P9_42B |P9_27 | | |P1.31 |P2.24 | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + |1 |E2 |P8_35 |P8_33 |P8_35 |P8_33 |P2.10 | | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + |2 |E3 |P8_12 |P8_11 |P8_12 |P8_11 |P2.24 |P2.33 | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + |2 | |P8_41 |P8_42 |P9_19 |P9_41 | | | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + | |E4 |P8_16 |P8_15 | | |P2.09 |P2.18 | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + |3 | | | |P8_25 |P8_24 | | | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + |3 | | | |P9_42 |P9_27 | | | + +----+--------+-----------+-----------+--------+--------+------------+-------------+ + +.. note:: + + The I/O pins on the Beagles have a mutliplexer that lets you select what I/O + appears on a given pin. The Blue has the mux already configured to to run these + examples. Follow the instructions in + :ref:`../03details/details.html#details_configure_encoders, Configuring Pins for Controlling Encoders` + to configure the pins for the Black and the Pocket. + + +Reading PRU Encoder +********************* + +Problem +~~~~~~~~ + +I want to access the PRU encoder. + +Solution +~~~~~~~~~ + +The forth encoder is implemented on the PRU and accessed with `sudo rc_test_encoders_pru` +.. note:: + This command needs root permission, so the `sudo` is needed. + +Here's what you will see + +.. code-block:: bash + + bone$ *sudo rc_test_encoders_pru* + [sudo] password for debian: + + Raw encoder position + E4 | + 0 |^C + +.. note:: + + If you aren't running the Blue you will have to configure the pins as shown + in the note above. + + +BeagleLogic -- a 14-channel Logic Analyzer +------------------------------------------- + +Problem +******** + +I need a 100Msps, 14-channel logic analyzer + +Solution +********* + +`BeagleLogic <https://beaglelogic.readthedocs.io/en/latest/>`_ is a 100Msps, +14-channel logic analyzer that runs on the Beagle. + +.. admonition:: information + + BeagleLogic turns your BeagleBone [Black] into a 14-channel, 100Msps Logic + Analyzer. Once loaded, it presents itself as a character device node */dev/beaglelogic*. + The core of the logic analyzer is the 'beaglelogic' kernel module that + reserves memory for and drives the two Programmable Real-Time Units + (PRU) via the remoteproc interface wherein the PRU directly writes logic + samples to the System Memory (DDR RAM) at the configured sample rate + one-shot or continuously without intervention from the ARM core. + + https://github.com/abhishek-kakkar/BeagleLogic/wiki + + +The quickest solution is to get the `no-setup-required image <https://github.com/abhishek-kakkar/BeagleLogic/wiki/BeagleLogic-%22no-setup-required%22-setup:-Introducing-System-Image!>`_. It points to an older image +(beaglelogic-stretch-2017-07-13-4gb.img.xz) but should still work. + +If you want to be running a newer image, there are instructions on the site for `installing BeagleLogic <https://beaglelogic.readthedocs.io/en/latest/install.html>`_, but I had to do the additional steps in :ref:`case_installing_beaglelogic`. + +.. TODO - Recheck + +.. _case_installing_beaglelogic: + +Installing BeagleLogic +~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + + bone$ *git clone https://github.com/abhishek-kakkar/BeagleLogic* + bone$ *cd BeagleLogic/kernel* + bone$ *mv beaglelogic-00A0.dts beaglelogic-00A0.dts.orig* + bone$ *wget https://gist.githubusercontent.com/abhishek-kakkar/0761ef7b10822cff4b3efd194837f49c/raw/eb2cf6cfb59ff5ccb1710dcd7d4a40cc01cfc050/beaglelogic-00A0.dts* + bone$ *make overlay* + bone$ *sudo cp beaglelogic-00A0.dtbo /lib/firmware/* + bone$ *sudo update-initramfs -u -k \`uname -r`* + bone$ *sudo reboot* + +Once the Bone has rebooted, browse to 192.168.7.2:4000 where you'll see +:ref:`case_beaglelogic_capture`. Here you can easily select the sample +rate, number of samples, and which pins to sample. +Then click *Begin Capture* to capture your data, at up to 100 MHz! + +.. _case_beaglelogic_capture: + +BeagleLogic Data Capture +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/beaglelogic_capture.png + :align: center + :alt: BeagleLogic Data Capture + + +Discussion +************ + +BeagleLogic is a complete system that includes firmware for the PRUs, +a kernel module and a web interface that create a powerful 100 MHz +logic analyzer on the Bone with no additional hardware needed. + +.. tip:: + + If you need buffered inputs, consider + http://standalone.beaglelogic.net/en/latest/[BeagleLogic Standalone], + a turnkey Logic Analyzer built on top of BeagleLogic. + + +The kernel interface makes it easy to control the PRUs through the +command line. For example + +.. code-block:: bash + + bone$ *dd if=/dev/beaglelogic of=mydump bs=1M count=1* + +will capture a binary dump from the PRUs. The sample rate and number of +bits per sample can be controlled through ``/sys/``. + +.. code-block:: bash + + bone$ *cd /sys/devices/virtual/misc/beaglelogic* + bone$ *ls* + buffers filltestpattern power state uevent + bufunitsize lasterror samplerate subsystem + dev memalloc sampleunit triggerflags + bone$ *cat samplerate* + 1000 + bone$ *cat sampleunit* + 8bit + +You can set the sample rate by simply writing to ``samplerate``. + +.. code-block:: bash + + bone$ *echo 100000000 > samplerate* + +`sysfs attributes Reference <https://beaglelogic.readthedocs.io/en/latest/sysfs_attributes.html>`_ +has more details on configuring via sysfs. + +If you run ``dmesg -Hw`` in another window you can see when a capture +is started and stopped. + +.. code-block:: bash + + bone$ *dmesg -Hw* + [Jul25 08:46] misc beaglelogic: capture started with sample rate=100000000 Hz, sampleunit=1, triggerflags=0 + [ +0.086261] misc beaglelogic: capture session ended + + +BeagleLogic uses the two PRUs to sample at 100Msps. Getting a PRU running at 200Hz to sample at 100Msps is a slick trick. +`The Embedded Kitchen <http://theembeddedkitchen.net/beaglelogic-building-a-logic-analyzer-with-the-prus-part-1/449>`_ has a nice article +explaining how the PRUs get this type of performance. + + +NeoPixels -- 5050 RGB LEDs with Integrated Drivers (Falcon Christmas) +---------------------------------------------------------------------- + +Problem +********* + +You have an `Adafruit NeoPixel LED string <http://www.adafruit.com/products/1138>`_, +`Adafruit NeoPixel LED matrix <http://www.adafruit.com/products/1487>`_ or +any other type of +`WS2812 LED <https://cdn-shop.adafruit.com/datasheets/WS2812.pdf>`_ +and want to light it up. + +.. TODO Show how to drive ws2812's with FPP. + +Solution +********* + +If you are driving just one string you can write your own code +(See :ref:`../05blocks/blocks.adoc#blocks_ws2812, WS2812 Driver`) +If you plan to drive multiple strings, then consider +Falcon Christmas (`FPP <https://falconchristmas.com/>`_). +FPP can be used to drive both LEDs with an integrated +driver (neopixels) or without an integrated driver. Here we'll show you how to +set up for the integrated drive and in the next section the no driver LEDs will be +show. + +Hardware +********* + +For this setup we'll wire a single string of NeoPixels to the Beagle. +I've attached the black wire on the string to ground on the Beagle +and the red wire to a 3.3V pin on the Beagle. +The yellow data in line is attached to P1.31 (I'm using a PocketBeagle.). + +How did I know to attach to P1.31? The FalconChristmas git repo +(https://github.com/FalconChristmas/fpp) has files that tell which pins +attach to which port. https://github.com/FalconChristmas/fpp/blob/master/capes/pb/strings/F8-B-20.json +has a list of 20 ports and where they are connected. Pin P1.31 appears on +line 27. It's the 20th entry in the list. You could pick any of the others +if you'd rather. + +Software Setup +*************** + +Assuming the PocketBeagle is attached via the USB cable, +on your host computer browse to <http://192.168.7.2/> and you will see +:ref:`case_fpp_program_control2`. + +.. _case_fpp_program_control2: + +Falcon Play Program Control +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_program_control.png + :align: center + :alt: Falcon Play Program Control + +You can test the display by first setting up the Channel Outputs and then +going to *Display Testing*. :ref:`case_channel_outputs_menu2` shows where to +select Channel Outputs and :ref:`case_channel_outputs2` shows which settings to use. + +.. _case_channel_outputs_menu2: + +Selecting Channel Outputs +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_channel_outputs_menu.png + :align: center + :alt: Selecting Channel Outputs + +.. _case_channel_outputs2: + +Channel Outputs Settings +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_channel_outputs_strings.png + :align: center + :alt: Channel Outputs Settings + +Click on the *Pixel Strings* tab. Earlier we noted that *P1.31* is attached +to port 20. Note that at the bottom of the screen, port 20 has a PIXEL COUNT +of 24. We're telling FPP our string has 24 NeoPixels and they are attached +to port 2 which in *P1.31*. + +Be sure to check the *Enable String Cape*. + +Next we need to test the display. Select **Display Testing** shown in +:ref:`case_display_testing_menu2`. + +.. _case_display_testing_menu2: + +Selecting Display Testing +~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_display_testing_menu2.png + :align: center + :alt: Selecting Display Testing + +Set the *End Channel* to *72*. (72 is 3*24) +Click *Enable Test Mode* and your matrix should light up. Try the different +testing patterns shown in :ref:`case_display_testing2`. + +.. note:: + + Clicking on the *-3* will subtract three from the End Channel, which should + then display three fewer LEDs which is one NeoPixel. The last of your NeoPixels + should go black. This is an easy way to make sure you have the correct pixel + count. + +.. _case_display_testing2: + +Display Testing Options +~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_display_testing2.png + :align: center + :alt: Display Testing Options + +You can control the LED string using the E1.31 protocol. +(https://www.doityourselfchristmas.com/wiki/index.php?title=E1.31_(Streaming-ACN)_Protocol) +First configure the input channels by going to Channel Inputs as shown in +:ref:`case_channel_inputs`. + +.. _case_channel_inputs: + +Going to Channel Inputs +~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_channel_inputs.png + :align: center + :alt: Going to Channel Inputs + +Tell it you have 72 LEDs and enable the input as shown in :ref:`case_set_inputs`. + +.. _case_set_inputs: + +Setting Channel Inputs +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_inputs_setup2.png + :align: center + :alt: Setting Channel Inputs + +Finally go to the Status Page as shown in :ref:`case_status`. + +.. _case_status: + +Watching the status +~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_status.png + :align: center + :alt: Watching Status + +Now run a program on another computer that generated E1.31 packets. +:ref:`case_1.31_example` is an example python program. + +.. _cse_e1.31_example: + +.e1.31-test.py +~~~~~~~~~~~~~~ + +:downlod:`e1.31-test.py <code/e1.31-test.py>`- Example of generating packets to control the NeoPixels + +.. TODO document the code + +.. _case_rgb_matrix: + +RGB LED Matrix -- No Integrated Drivers (Falcon Christmas) +----------------------------------------------------------- + +Problem +************************* + +You want to use a RGB LED Matrix display that doesn't have integrated +drivers such as the +`64x32 RGB LED Matrix <https://www.adafruit.com/product/2277>`_ by Adafuit +shown in :ref:`case_adfruit_matrix`. + +.. _case_adfruit_matrix: + +Adafruit LED Matrix +~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/ledmatrix.jpg + :align: center + :alt: Adafruit LED Matrix + +Solution +************************* + +`Falcon Christmas <http://falconchristmas.com>`_ makes a software package +called +`Falcon Player <http://falconchristmas.com/forum/index.php/board,8.0.html>`_ (FPP) which can drive +such displays. + +.. admonition:: information: + + The Falcon Player (FPP) is a lightweight, optimized, feature-rich sequence player + designed to run on low-cost SBC's (Single Board Computers). + FPP is a software solution that you download and install on hardware which can be + purchased from numerous sources around the internet. + FPP aims to be controller agnostic, it can talk E1.31, DMX, Pixelnet, and + Renard to hardware from multiple hardware vendors, including controller + hardware from Falcon Christmas available via COOPs or in the store on FalconChristmas.com. + + http://www.falconchristmas.com/wiki/FPP:FAQ#What_is_FPP.3F + +Hardware +~~~~~~~~~ + +The Beagle hardware can be either a BeagleBone Black with the +`Octoscroller Cape <https://oshpark.com/shared_projects/7mSHNZcD>`_, or a +PocketBeagle with the +`PocketScroller LED Panel Cape <https://www.hackster.io/daniel-kulp/pocketscroller-led-panel-cape-for-pocketbeagle-fe12a6>`_. +(See `to purchase <https://kulplights.com/product/pocketscroller/>`_.) +`Building and Octoscroller Matrix Display <https://www.diychristmas.org/wiki/index.php?title=Building_an_Octoscroller_Matrix_Display>`_ +gives details for using the BeagleBone Black. + +:ref:`case_pocket` shows how to attach the PocketBeagle to the P5 LED matrix +and where to attach the 5V power. If you are going to turn on all the LEDs +to full white at the same time you will need at least a 4A supply. + +.. _case_pocket: + +Pocket Beagle Driving a P5 RGB LED Matrix via the PocketScroller Cape +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pocketscroller.jpg + :align: center + :alt: Pocket Beagle Driving a P5 RGB LED Matrix via the PocketScroller Cape + + +Software +~~~~~~~~~~ + +The FPP software is most easily installed by downloading the +`current FPP release <https://github.com/FalconChristmas/fpp/releases/>`_, flashing an SD card and +booting from it. + +.. tip:: + + The really brave can install it on a already running image. See details at + https://github.com/FalconChristmas/fpp/blob/master/SD/FPP_Install.sh + + +Assuming the PocketBeagle is attached via the USB cable, +on your host computer browse to http://192.168.7.2/ and you will see +:ref:`case_fpp_program_control`. + +.. _case_fpp_program_control: + +Falcon Play Program Control +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_program_control.png + :align: center + :alt: Falcon Play Program Control + +You can test the display by first setting up the Channel Outputs and then +going to *Display Testing*. :ref:`case_channel_outputs_menu` shows where to +select Channel Outputs and :ref:`case_channel_outputs` shows which settings to use. + +.. _case_channel_outputs_menu: + +Selecting Channel Outputs +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_channel_outputs_menu.png + :align: center + :alt: Selecting Channel Outputs + +.. _case_channel_outputs: + +Channel Outputs Settings +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_channel_outputs.png + :align: center + :alt: Channel Outputs Settings + +Click on the **LED Panels** tab and then the only changes I made was +to select the **Single Panel Size** to be +*64x32* and to check the **Enable LED Panel Output**. + +Next we need to test the display. Select *Display Testing* shown in +:ref:`case_display_testing_menu`. + +.. _case_display_testing_menu: + +Selecting Display Testing +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_display_testing_menu.png + :align: center + :alt: Selecting Display Testing + +Set the **End Channel** to **6144**. (6144 is 3*64*32) +Click **Enable Test Mode** and your matrix should light up. Try the different +testing patterns shown in :ref:`case_display_testing`. + +.. _case_display_testing: + +Display Testing Options +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_display_testing.png + :align: center + :alt: Display Testing Options + +xLights - Creating Content for the Display +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Once you are sure your LED Matrix is working correctly you can program it +with a sequence. + + +.. admonition:: information: + + xLights is a free and open source program that enables you to design, create and play + amazing lighting displays through the use of DMX controllers, E1.31 Ethernet controllers and more. + + With it you can layout your display visually then assign effects to the various items + throughout your sequence. This can be in time to music (with beat-tracking built into xLights) + or just however you like. xLights runs on Windows, OSX and Linux + + https://xlights.org/ + + +xLights can be installed on your host computer (not the Beagle) by +following instructions at https://xlights.org/releases/. + +Run xLights and you'll see :ref:`case_xlights_setup`. + +.. code-block:: bash + + host$ *chmod +x xLights-2021.18-x86_64.AppImage* + host$ *./xLights-2021.18-x86_64.AppImage* + +.. TODO update the figures. + +.. _case_xlights_setup: + +xLights Setup +~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_setup.png + :align: center + :alt: xLights Setup + +We'll walk you through a simple setup to get an animation to display on the +RGB Matrix. xLights can use a protocol called E1.31 to send information to +the display. Setup xLights by clicking on *Add Ethernet* and entering the values +shown in :ref:`case_xlights_setup_e1_31`. + +.. _case_xlights_setup_e1_31: + +Setting Up E1.31 +~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_setup_e1_31.png + :align: center + :alt: Setting Up E1.31 + +The **IP Address** is the Bone's address as seen from the host computer. +Each LED is one channel, so one RGB LED is three channels. The P5 board +has 3*64*32 or 6144 channels. These are grouped into universes of 512 +channels each. This gives 6144/512 = 12 universes. See the +`E.13 documentation <https://www.doityourselfchristmas.com/wiki/index.php?title=E1.31_(Streaming-ACN)_Protocol#Configuring_Sequencing_Software_to_use_E1.31_Output>`_ +for more details. + +Your setup should look like :ref:`case_xlights_setup_done`. Click the +*Save Setup* button to save. + +.. _case_xlights_setup_done: + +xLights setup for P5 display +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_setup_done.png + :align: center + :alt: xLights setup for P5 display + +Next click on the **Layout** tab. Click on the *Matrix* button as shown in +:ref:`case_xlights_matrix`, then click on the black area where you want your +matrix to appear. + +.. _case_xlights_matrix: + +Setting up the Matrix Layout +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_layout.png + :align: center + :alt: Setting up the Matrix Layout + +:ref:`case_xlights_layout_details` shows the setting to use for the P5 matrix. + +.. _case_xlights_layout_details: + +Layout details for P5 matrix +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_layout_details.png + :align: center + :alt: Layout details for P5 matrix + +All I changed was **# Strings**, **Nodes/String**, **Starting Location** and most +importantly, expand **String Properties** and select at **String Type** of +**RGB Nodes**. Above the setting you should see that **Start Chan** is 1 and +the **End Chan** is 6144, which is the total number of individual LEDs (3*63*32). +xLights now knows we are working with a P5 matrix, now on to the sequencer. + +Now click on the *Sequencer* tab and then click on the **New Sequence** button +(:ref:`case_seq_new`). + +.. _case_seq_new: + +Starting a new sequence +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_seq_new.png + :align: center + :alt: Starting a new sequence + +Then click on **Animation**, **20fps (50ms)**, and **Quick Start**. Learning how to +do sequences is beyond the scope of this cookbook, however I'll shown you how +do simple sequence just to be sure xLights is talking to the Bone. + +Setting Up E1.31 on the Bone +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +First we need to setup FPP to take input from xLights. Do this by going to +the *Input/Output Setup* menu and selecting *Channel Inputs*. Then +enter *12* for *Universe Count* and click *set* and you will see +:ref:`case_inputs_setup`. + +.. _case_inputs_setup: + +E1.31 Inputs +~~~~~~~~~~~~~ + +.. figure:: figures/fpp_inputs_setup.png + :align: center + :alt: .E1.31 Inputs + +Click on the **Save** button above the table. + +Then go to the **Status/Control** menu and select **Status Page**. + +.. TODO update this + +.. _case_mode_bridge: + +Bridge Mode +~~~~~~~~~~~~~ + +.. figure:: figures/fpp_mode_bridge.png + :align: center + :alt: Bridge Mode + +Testing the xLights Connection +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +The Bone is now listening for commands from xLights via the E1.31 protocol. +A quick way to verify everything is t o return to xLights and go to the +*Tools* menu and select **Test** (:ref:`case_xlights_test`). + +.. _case_xlights_test: + +xLights test page +~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_test.png + :align: center + :alt: xLights test page + +Click the box under **Select channels...**, click **Output to lights** and +select **Twinkle 50%**. You matrix should have a colorful twinkle pattern +(:ref:`case_xlights_twinkle`). + +.. _case_xlights_twinkle: + +xLights Twinkle test pattern +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_twinkle.jpg + :align: center + :alt: xLights Twinkle test pattern + +A Simple xLights Sequence +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Now that the xLights to FPP link is tested you can generate a sequence to +play. Close the Test window and click on the **Sequencer** tab. Then drag +an effect from the **Effects** box to the timeline that below it. Drop it to +the right of the **Matrix** label (:ref:`case_seq_drag`). The click +*Output To Lights* which is the yellow lightbulb to the right on the top +toolbar. Your matrix should now be displaying your effect. + +.. _case_seq_drag: + +Drag an effect to the timeline +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/xlights_seq_drag.png + :align: center + :alt: Drag an effect to the timeline + +The setup requires the host computer to send the animation data to the Bone. +The next section shows how to save the sequence and play it on the Bone +standalone. + +Saving a Sequence and Playing it Standalone +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +In xLights save your sequence by hitting Ctrl-S and giving it a name. I called +mine *fire* since I used a fire effect. Now, switch back to FPP and select +the *Content Setup* menu and select *File Manager*. Click the black +*Select Files* button and select your sequence file that ends in *.fseq* +(:ref:`case_file_manager`). + +.. _case_file_manager: + +FPP file manager +~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_file_manager.png + :align: center + :alt: FPP file manager + +Once your sequence is uploaded, got to **Content Steup** and select **Playlists**. +Enter you playlist name (I used **fire**) and click **Add**. Then click +**Add a Sequence/Entry** and select **Sequence Only** +(:ref:`case_playlist`), then click **Add**. + +.. _case_playlist: + +Adding a new playlist to FPP +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_playlist.png + :align: center + :alt: Adding a new playlist to FPP + +Be sure to click **Save Playlist** on the right. Now return to +**Status/Control** and **Status Page** and make sure **FPPD Mode:** is set +to **Standalone**. You should see your playlist. Click the **Play** +button and your sequence will play. + +.. _case_playlist_status: + +Adding a new playlist to FPP +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/fpp_playlist_status.png + :align: center + :alt: Playing a playlist + +The beauty of the PRU is that the Beagle can play a detailed sequence at +20 frames per second and the ARM procossor is only 15% used. The PRUs +are doing all the work. + +simpPRU -- A python-like language for programming the PRUs +----------------------------------------------------------- + +`simpPRU <https://github.com/VedantParanjape/simpPRU>`_ is a simple, python-like +programming languge designed to make programming the PRUs easy. +It has detailed `documentation <https://simppru.readthedocs.io/en/latest/>`_ and +many `examples <https://simppru.readthedocs.io/en/latest/examples/digital_read/>`_. + +.. admonition:: information + + simpPRU is a procedural programming language that is statically typed. + Variables and functions must be assigned data types during compilation. + It is typesafe, and data types of variables are decided during compilation. + simPRU codes have a +.sim+ extension. + simpPRU provides a console app to use Remoteproc functionality. + + https://simppru.readthedocs.io/en/latest/ + +You can `build simpPRU <https://simppru.readthedocs.io/en/latest/install/build/>`_ from +source, more easily just `install it <https://simppru.readthedocs.io/en/latest/install/install/>`_. +On the Beagle run: + +.. code-block:: bash + + bone$ wget https://github.com/VedantParanjape/simpPRU/releases/download/1.4/simppru-1.4-armhf.deb + bone$ sudo dpkg -i simppru-1.4-armhf.deb + bone$ sudo apt update + bone$ sudo apt install gcc-pru + +Now, suppose you wanted to run the +`LED blink <https://simppru.readthedocs.io/en/latest/examples/led_blink/>`_ +example which is reproduced here. + +LED Blink (blink.sim) +~~~~~~~~~~~~~~~~~~~~~ + +:download:`blink.sim <code/blink.sim>` + +Just run simppru + +.. code-block:: bash + + bone$ simppru blink.sim --load + Detected TI AM335x PocketBeagle + inside while + [4] : setting P1_31 as output + + Current mode for P1_31 is: pruout + +Detected TI AM335x PocketBeagle +-------------------------------- + +The +--load+ flag caused the compiled code to be copied to +/lib/firmware+. +To start just do: + +.. code-block:: bash + + bone$ cd /dev/remoteproc/pruss-core0/ + bone$ ls + device firmware name power state subsystem uevent + bone$ echo start > state + bone$ cat state + running + +Your LED should now be blinking. + +Check out the many examples (https://simppru.readthedocs.io/en/latest/examples/led_blink/). + +simpPRU Examples +~~~~~~~~~~~~~~~~ + +.. figure:: figures/LEDblink.png + :align: center + :alt: simpPRU Examples + + +MachineKit +----------- + +`MachineKit <http://www.machinekit.io/>`_ is a platform for machine control +applications. It can control machine tools, robots, or other automated devices. It can control servo +motors, stepper motors, relays, and other devices related to machine tools. + +.. admonition:: information + + Machinekit is portable across a wide range of hardware platforms and real-time environments, + and delivers excellent performance at low cost. It is based on the HAL component architecture, + an intuitive and easy to use circuit model that includes over 150 building blocks for digital logic, + motion, control loops, signal processing, and hardware drivers. Machinekit supports local and + networked UI options, including ubiquitous platforms like phones or tablets. + + http://www.machinekit.io/about/ + +ArduPilot +---------- +`ArduPilot <http://ardupilot.org/>`_ is a open source autopilot system supporting +multi-copters, traditional helicopters, fixed wing aircraft and rovers. ArduPilot runs on a many +`hardware platforms <http://ardupilot.org/copter/docs/common-autopilots.html>`_ including the +`BeagleBone Black <http://ardupilot.org/dev/docs/building-for-beaglebone-black-on-linux.html#building-for-beaglebone-black-on-linux>`_ and the +`BeagleBone Blue <http://ardupilot.org/copter/docs/common-beagle-bone-blue.html>`_. + + + +.. admonition:: information + + Ardupilot is the most advanced, full-featured and reliable open source autopilot software available. + It has been developed over 5+ years by a team of diverse professional engineers and computer scientists. + It is the only autopilot software capable of controlling any vehicle system imaginable, from conventional + airplanes, multirotors, and helicopters, to boats and even submarines. And now being expanded to feature + support for new emerging vehicle types such as quad-planes and compound helicopters. + + Installed in over 1,000,000 vehicles world-wide, and with its advanced data-logging, analysis + and simulation tools, Ardupilot is the most tested and proven autopilot software. The open-source + code base means that it is rapidly evolving, always at the cutting edge of technology development. + With many peripheral suppliers creating interfaces, users benefit from a broad ecosystem of sensors, + companion computers and communication systems. Finally, since the source code is open, it can be + audited to ensure compliance with security and secrecy requirements. + + The software suite is installed in aircraft from many OEM UAV companies, such as 3DR, jDrones, + PrecisionHawk, AgEagle and Kespry. It is also used for testing and development by several large + institutions and corporations such as NASA, Intel and Insitu/Boeing, as well as countless + colleges and universities around the world. + + http://www.machinekit.io/about/ diff --git a/pru-cookbook/01case/code/blink.sim b/pru-cookbook/01case/code/blink.sim new file mode 100644 index 0000000000000000000000000000000000000000..212a21689e76170ad44112b926a0c4300e0c03a4 --- /dev/null +++ b/pru-cookbook/01case/code/blink.sim @@ -0,0 +1,7 @@ +/* From: https://simppru.readthedocs.io/en/latest/examples/led_blink/ */ +while : 1 == 1 { + digital_write(P1_31, true); + delay(250); /* Delay 250 ms */ + digital_write(P1_31, false); + delay(250); +} diff --git a/pru-cookbook/01case/code/circle.py b/pru-cookbook/01case/code/circle.py new file mode 100755 index 0000000000000000000000000000000000000000..1feb41cc089af343f1f2c490deef92853965bee4 --- /dev/null +++ b/pru-cookbook/01case/code/circle.py @@ -0,0 +1,43 @@ +#!/usr/bin/env python3 + +"""A demo client for Open Pixel Control +http://github.com/zestyping/openpixelcontrol + +Runs an LED around in a circle + +""" + +import time +import opc + +ADDRESS = 'localhost:7890' + +# Create a client object +client = opc.Client(ADDRESS) + +# Test if it can connect +if client.can_connect(): + print('connected to %s' % ADDRESS) +else: + # We could exit here, but instead let's just print a warning + # and then keep trying to send pixels in case the server + # appears later + print('WARNING: could not connect to %s' % ADDRESS) + +# Send pixels forever +STR_LEN=16 +for i in range(STR_LEN): + leds = [(0, 0, 0)] * STR_LEN +leds[0] = (0, 127, 0) + +while True: + tmp = leds[0] + for i in range(STR_LEN-1): + leds[i] = leds[i+1] + leds[-1] = tmp + if client.put_pixels(leds, channel=0): + print('sent') + else: + print('not connected') + time.sleep(0.1) + diff --git a/pru-cookbook/01case/code/e1.31-test.py b/pru-cookbook/01case/code/e1.31-test.py new file mode 100755 index 0000000000000000000000000000000000000000..6016c9eb9e879a6d21f8ccf8fd20ed32bd4c45b3 --- /dev/null +++ b/pru-cookbook/01case/code/e1.31-test.py @@ -0,0 +1,70 @@ +#!/usr/bin/env python3 +# Controls a NeoPixel (WS2812) string via E1.31 and FPP +# https://pypi.org/project/sacn/ +# https://github.com/FalconChristmas/fpp/releases +import sacn +import time + +# provide an IP-Address to bind to if you are using Windows and want to use multicast +sender = sacn.sACNsender("192.168.7.1") +sender.start() # start the sending thread +sender.activate_output(1) # start sending out data in the 1st universe +sender[1].multicast = False # set multicast to True +sender[1].destination = "192.168.7.2" # or provide unicast information. +sender.manual_flush = True # turning off the automatic sending of packets +# Keep in mind that if multicast is on, unicast is not used +LEDcount = 24 +# Have green fade is as it goes +data = [] +for i in range(LEDcount): + data.append(0) # Red + data.append(i) # Green + data.append(0) # Blue +sender[1].dmx_data = data +sender.flush() +time.sleep(0.5) + +# Turn off all LEDs +data=[] +for i in range(3*LEDcount): + data.append(0) +sender.flush() +sender[1].dmx_data = data +time.sleep(0.5) + +# Have red fade in +data = [] +for i in range(LEDcount): + data.append(i) + data.append(0) + data.append(0) +sender[1].dmx_data = data +sender.flush() +time.sleep(0.25) + +# Make LED circle 5 times +for j in range(15): + for i in range(LEDcount-1): + data[3*i+0] = 0 + data[3*i+1] = 0 + data[3*i+2] = 0 + data[3*i+3] = 0 + data[3*i+4] = 64 + data[3*i+5] = 0 + sender[1].dmx_data = data + sender.flush() + time.sleep(0.02) +# Wrap around + i = LEDcount-1 + data[0] = 0 + data[1] = 64 + data[2] = 0 + data[3*i+0] = 0 + data[3*i+1] = 0 + data[3*i+2] = 0 + sender[1].dmx_data = data + sender.flush() + time.sleep(0.02) + +time.sleep(2) # send the data for 10 seconds +sender.stop() # do not forget to stop the sender diff --git a/pru-cookbook/01case/code/encoder_setup.sh b/pru-cookbook/01case/code/encoder_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..ffc4ad68842fb6314173b523425295893cd05fc1 --- /dev/null +++ b/pru-cookbook/01case/code/encoder_setup.sh @@ -0,0 +1,49 @@ +#!/bin/bash +# Configure the pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine + +# Configure eQEP pins +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_92 P9_27 P8_35 P8_33 P8_12 P8_11 P8_41 P8_42" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_31 P2_34 P2_10 P2_24 P2_33" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin qep + config-pin -q $pin +done + +########################################## +# Configure PRU pins +if [ $machine = "Black" ]; then + echo " Found" + pins="P8_16 P8_15" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_09 P2_18" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruin + config-pin -q $pin +done diff --git a/pru-cookbook/01case/code/fire.fseq b/pru-cookbook/01case/code/fire.fseq new file mode 100644 index 0000000000000000000000000000000000000000..4bad5ae76476a87bb7bb71e853e3412b1fc63f29 Binary files /dev/null and b/pru-cookbook/01case/code/fire.fseq differ diff --git a/pru-cookbook/01case/code/logic_install.sh b/pru-cookbook/01case/code/logic_install.sh new file mode 100644 index 0000000000000000000000000000000000000000..7fb76122b55b55da9e50ea2a084b1242b8e30b78 --- /dev/null +++ b/pru-cookbook/01case/code/logic_install.sh @@ -0,0 +1,34 @@ +# Instructions for installing BeagleLogic +# https://beaglelogic.readthedocs.io/en/latest/index.html +# From: https://github.com/abhishek-kakkar/BeagleLogic/wiki + +# beaglelogic is installed on the 4.9 kernel, but not the 4.14, so +# if you are running 4.14, switch to 4.9 +sudo /opt/scripts/tools/update_kernel.sh --lts-4_9 +sudo reboot + +git clone https://github.com/abhishek-kakkar/BeagleLogic +cd BeagleLogic +sudo ./install.sh +sudo reboot + +# Now the kernel driver headers +sudo apt install linux-headers-`uname -r` + +modinfo beaglelogic +modprobe beaglelogic + +# From: https://beaglelogic.readthedocs.io/en/latest + +# Here's what works from abhishek +git clone https://github.com/abhishek-kakkar/BeagleLogic +cd BeagleLogic +# 1. Get the dts file from this gist (https://gist.github.com/abhishek-kakkar/0761ef7b10822cff4b3efd194837f49c) + +# 2. cd to 'BeagleLogic/kernel' directory. Put the dts file there. (edited) +# 3. Run 'make overlay' +# 4. Run 'sudo cp -v beaglelogic-00A0.dtbo /lib/firmware/' (edited) +# 5. Run 'sudo update-initramfs -u -k ``uname -r``' (edited) +# (single backticks only, apparently Slack treats single backtick as code) (edited) +# 6. Reboot +# 7. Browse to bone:4000 diff --git a/pru-cookbook/01case/code/main_pru1.c b/pru-cookbook/01case/code/main_pru1.c new file mode 100644 index 0000000000000000000000000000000000000000..8d751b3e65e06a8dd4e47c9ef6f28f370d60ca24 --- /dev/null +++ b/pru-cookbook/01case/code/main_pru1.c @@ -0,0 +1,53 @@ +/* + * Source Modified by Zubeen Tolani < ZeekHuge - zeekhuge@gmail.com > + * Based on the examples distributed by TI + * + * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_pru1.h" + +// The function is defined in pru1_asm_blinky.asm in same dir +// We just need to add a declaration here, the defination can be +// seperately linked +extern void start(void); + +void main(void) +{ + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + start(); +} + diff --git a/pru-cookbook/01case/code/my-config.json b/pru-cookbook/01case/code/my-config.json new file mode 100644 index 0000000000000000000000000000000000000000..d1534cfff15f46bbabb10068c01542bb721583c4 --- /dev/null +++ b/pru-cookbook/01case/code/my-config.json @@ -0,0 +1,19 @@ +{ + "outputMode": "ws281x", + "outputMapping": "original-ledscape", + "demoMode": "fade", + "ledsPerStrip": 16, + "usedStripCount": 1, + "colorChannelOrder": "BRG", + "opcTcpPort": 7890, + "opcUdpPort": 7890, + "enableInterpolation": false, + "enableDithering": false, + "enableLookupTable": true, + "lumCurvePower": 2.0000, + "whitePoint": { + "red": 0.9000, + "green": 1.0000, + "blue": 1.0000 + } +} diff --git a/pru-cookbook/01case/code/opc.py b/pru-cookbook/01case/code/opc.py new file mode 100755 index 0000000000000000000000000000000000000000..36a3bbd6b76e7ff8e6660ea7b163da2271bbe696 --- /dev/null +++ b/pru-cookbook/01case/code/opc.py @@ -0,0 +1,177 @@ +#!/usr/bin/env python + +"""Python Client library for Open Pixel Control +http://github.com/zestyping/openpixelcontrol + +Sends pixel values to an Open Pixel Control server to be displayed. +http://openpixelcontrol.org/ + +Recommended use: + + import opc + + # Create a client object + client = opc.Client('localhost:7890') + + # Test if it can connect (optional) + if client.can_connect(): + print('connected to %s' % ADDRESS) + else: + # We could exit here, but instead let's just print a warning + # and then keep trying to send pixels in case the server + # appears later + print('WARNING: could not connect to %s' % ADDRESS) + + # Send pixels forever at 30 frames per second + while True: + my_pixels = [(255, 0, 0), (0, 255, 0), (0, 0, 255)] + if client.put_pixels(my_pixels, channel=0): + print('...') + else: + print('not connected') + time.sleep(1/30.0) + +""" + +import socket +import struct +import sys + +SET_PIXEL_COLOURS = 0 # "Set pixel colours" command (see openpixelcontrol.org) + + +class Client(object): + def __init__(self, server_ip_port, long_connection=True, verbose=False): + """Create an OPC client object which sends pixels to an OPC server. + + server_ip_port should be an ip:port or hostname:port as a single string. + For example: '127.0.0.1:7890' or 'localhost:7890' + + There are two connection modes: + * In long connection mode, we try to maintain a single long-lived + connection to the server. If that connection is lost we will try to + create a new one whenever put_pixels is called. This mode is best + when there's high latency or very high framerates. + * In short connection mode, we open a connection when it's needed and + close it immediately after. This means creating a connection for each + call to put_pixels. Keeping the connection usually closed makes it + possible for others to also connect to the server. + + A connection is not established during __init__. To check if a + connection will succeed, use can_connect(). + + If verbose is True, the client will print debugging info to the console. + + """ + self.verbose = verbose + + self._long_connection = long_connection + + self._ip, self._port = server_ip_port.split(':') + self._port = int(self._port) + + self._socket = None # will be None when we're not connected + + def _debug(self, m): + if self.verbose: + print(' %s' % str(m)) + + def _ensure_connected(self): + """Set up a connection if one doesn't already exist. + + Return True on success or False on failure. + + """ + if self._socket: + self._debug('_ensure_connected: already connected, doing nothing') + return True + + try: + self._debug('_ensure_connected: trying to connect...') + self._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) + self._socket.settimeout(1) + self._socket.connect((self._ip, self._port)) + self._debug('_ensure_connected: ...success') + return True + except socket.error: + self._debug('_ensure_connected: ...failure') + self._socket = None + return False + + def disconnect(self): + """Drop the connection to the server, if there is one.""" + self._debug('disconnecting') + if self._socket: + self._socket.close() + self._socket = None + + def can_connect(self): + """Try to connect to the server. + + Return True on success or False on failure. + + If in long connection mode, this connection will be kept and re-used for + subsequent put_pixels calls. + + """ + success = self._ensure_connected() + if not self._long_connection: + self.disconnect() + return success + + def put_pixels(self, pixels, channel=0): + """Send the list of pixel colors to the OPC server on the given channel. + + channel: Which strand of lights to send the pixel colors to. + Must be an int in the range 0-255 inclusive. + 0 is a special value which means "all channels". + + pixels: A list of 3-tuples representing rgb colors. + Each value in the tuple should be in the range 0-255 inclusive. + For example: [(255, 255, 255), (0, 0, 0), (127, 0, 0)] + Floats will be rounded down to integers. + Values outside the legal range will be clamped. + + Will establish a connection to the server as needed. + + On successful transmission of pixels, return True. + On failure (bad connection), return False. + + The list of pixel colors will be applied to the LED string starting + with the first LED. It's not possible to send a color just to one + LED at a time (unless it's the first one). + + """ + self._debug('put_pixels: connecting') + is_connected = self._ensure_connected() + if not is_connected: + self._debug('put_pixels: not connected. ignoring these pixels.') + return False + + # build OPC message + command = SET_PIXEL_COLOURS + header = struct.pack('>BBH', channel, SET_PIXEL_COLOURS, len(pixels)*3) + pieces = [struct.pack( + 'BBB', + min(255, max(0, int(r))), + min(255, max(0, int(g))), + min(255, max(0, int(b))) + ) for r, g, b in pixels] + if bytes is str: + message = header + ''.join(pieces) + else: + message = header + b''.join(pieces) + + self._debug('put_pixels: sending pixels to server') + try: + self._socket.send(message) + except socket.error: + self._debug('put_pixels: connection lost. could not send pixels.') + self._socket = None + return False + + if not self._long_connection: + self._debug('put_pixels: disconnecting') + self.disconnect() + + return True diff --git a/pru-cookbook/01case/code/pru1-servo.asm b/pru-cookbook/01case/code/pru1-servo.asm new file mode 100644 index 0000000000000000000000000000000000000000..c8e66e5234214137be6c7f7a781083c21061190e --- /dev/null +++ b/pru-cookbook/01case/code/pru1-servo.asm @@ -0,0 +1,162 @@ +;* +;* Copyright (C) 2016 Zubeen Tolani <ZeekHuge - zeekhuge@gmail.com> +;* +;* This file is as an example to show how to develope +;* and compile inline assembly code for PRUs +;* +;* This program is free software; you can redistribute it and/or modify +;* it under the terms of the GNU General Public License version 2 as +;* published by the Free Software Foundation. + + + .cdecls "main_pru1.c" + +DELAY .macro time, reg + LDI32 reg, time + QBEQ $E?, reg, 0 +$M?: SUB reg, reg, 1 + QBNE $M?, reg, 0 +$E?: + .endm + + + .clink + .global start +start: + LDI R30, 0xFFFF + DELAY 10000000, r11 + LDI R30, 0x0000 + DELAY 10000000, r11 +; JMP start + +; HALT + + +; these pin definitions are specific to SD-101C Robotics Cape + .asg r30.t8, CH1BIT ; P8_27 + .asg r30.t10, CH2BIT ; P8_28 + .asg r30.t9, CH3BIT ; P8_29 + .asg r30.t11, CH4BIT ; P8_30 + .asg r30.t6, CH5BIT ; P8_39 + .asg r30.t7, CH6BIT ; P8_40 + .asg r30.t4, CH7BIT ; P8_41 + .asg r30.t5, CH8BIT ; P8_42 + + .asg C4, CONST_SYSCFG + .asg C28, CONST_PRUSHAREDRAM + + .asg 0x22000, PRU0_CTRL + .asg 0x24000, PRU1_CTRL ; page 19 + .asg 0x28, CTPPR0 ; page 75 + + .asg 0x000, OWN_RAM + .asg 0x020, OTHER_RAM + .asg 0x100, SHARED_RAM ; This is so prudebug can find it. + + LBCO &r0, CONST_SYSCFG, 4, 4 ; Enable OCP master port + CLR r0, r0, 4 ; Clear SYSCFG[STANDBY_INIT] to enable OCP master port + SBCO &r0, CONST_SYSCFG, 4, 4 + +; Configure the programmable pointer register for PRU0 by setting c28_pointer[15:0] + LDI r0, SHARED_RAM ; Set C28 to point to shared RAM + LDI32 r1, PRU1_CTRL + CTPPR0 ; Note we use beginning of shared ram unlike example which + SBBO &r0, r1, 0, 4 ; page 25 + + LDI r9, 0x0 ; erase r9 to use to use later + + LDI r0, 0x0 ; clear internal counters + LDI r1, 0x0 + LDI r2, 0x0 + LDI r3, 0x0 + LDI r4, 0x0 + LDI r5, 0x0 + LDI r6, 0x0 + LDI32 r7, 0x0 + LDI r30, 0x0 ; turn off GPIO outputs + + +; Beginning of loop, should always take 48 instructions to complete +CH1: + QBEQ CLR1, r0, 0 ; If timer is 0, jump to clear channel + SET r30, CH1BIT ; If non-zero turn on the corresponding channel + SUB r0, r0, 1 ; Subtract one from timer + CLR r9, r9.t1 ; waste a cycle for timing + SBCO &r9, CONST_PRUSHAREDRAM, 0, 4 ; write 0 to shared memory +CH2: + QBEQ CLR2, r1, 0 + SET r30, CH2BIT + SUB r1, r1, 1 + CLR r9, r9.t1 + SBCO &r9, CONST_PRUSHAREDRAM, 4, 4 +CH3: + QBEQ CLR3, r2, 0 + SET r30, CH3BIT + SUB r2, r2, 1 + CLR r9, r9.t1 + SBCO &r9, CONST_PRUSHAREDRAM, 8, 4 +CH4: + QBEQ CLR4, r3, 0 + SET r30, CH4BIT + SUB r3, r3, 1 + CLR r9, r9.t1 + SBCO &r9, CONST_PRUSHAREDRAM, 12, 4 +CH5: + QBEQ CLR5, r4, 0 + SET r30, CH5BIT + SUB r4, r4, 1 + CLR r9, r9.t1 + SBCO &r9, CONST_PRUSHAREDRAM, 16, 4 +CH6: + QBEQ CLR6, r5, 0 + SET r30, CH6BIT + SUB r5, r5, 1 + CLR r9, r9.t1 + SBCO &r9, CONST_PRUSHAREDRAM, 20, 4 +CH7: + QBEQ CLR7, r6, 0 + SET r30, CH7BIT + SUB r6, r6, 1 + CLR r9, r9.t1 + SBCO &r9, CONST_PRUSHAREDRAM, 24, 4 +CH8: + QBEQ CLR8, r7, 0 + SET r30, CH8BIT + SUB r7, r7, 1 + SBCO &r9, CONST_PRUSHAREDRAM, 28, 4 + + QBA CH1 ; return to beginning of loop + ; no need to waste a cycle for timing here because of the QBA above + + +CLR1: + CLR r30, CH1BIT ; turn off the corresponding channel + LBCO &r0, CONST_PRUSHAREDRAM, 0, 4 ; Load new timer register + QBA CH2 +CLR2: + CLR r30, CH2BIT + LBCO &r1, CONST_PRUSHAREDRAM, 4, 4 + QBA CH3 +CLR3: + CLR r30, CH3BIT + LBCO &r2, CONST_PRUSHAREDRAM, 8, 4 + QBA CH4 +CLR4: + CLR r30, CH4BIT + LBCO &r3, CONST_PRUSHAREDRAM, 12, 4 + QBA CH5 +CLR5: + CLR r30, CH5BIT + LBCO &r4, CONST_PRUSHAREDRAM, 16, 4 + QBA CH6 +CLR6: + CLR r30, CH6BIT + LBCO &r5, CONST_PRUSHAREDRAM, 20, 4 + QBA CH7 +CLR7: + CLR r30, CH7BIT + LBCO &r6, CONST_PRUSHAREDRAM, 24, 4 + QBA CH8 +CLR8: + CLR r30, CH8BIT + LBCO &r7, CONST_PRUSHAREDRAM, 28, 4 + QBA CH1 ; return to beginning of loop diff --git a/pru-cookbook/01case/code/servo-test.c b/pru-cookbook/01case/code/servo-test.c new file mode 100644 index 0000000000000000000000000000000000000000..1f94e274e573223dcf84b4bd111f9a3df16ac867 --- /dev/null +++ b/pru-cookbook/01case/code/servo-test.c @@ -0,0 +1,84 @@ +/* + * + * servo tester + * (c) Copyright 2016 + * Mark A. Yoder, 20-July-2016 + * + */ + +#include <stdio.h> +#include <fcntl.h> +#include <sys/mman.h> +#include "robotics_cape_defs.h" + +#define PRU_ADDR 0x4A300000 // Start of PRU memory Page 184 am335x TRM +#define PRU_LEN 0x80000 // Length of PRU memory +#define PRU_SHAREDMEM 0x10000 // Offset to shared memory + +unsigned int *prusharedMem_32int_ptr; // Points to the start of the shared memory + +/******************************************************************************* +* int send_servo_pulse_us(int ch, int us) +* +* Sends a single pulse of duration us (microseconds) to a single channel (ch) +*******************************************************************************/ +int send_servo_pulse_us(int ch, int us) { + // Sanity Checks + if(ch<1 || ch>SERVO_CHANNELS){ + printf("ERROR: Servo Channel must be between 1&%d\n", SERVO_CHANNELS); + return -1; + } if(prusharedMem_32int_ptr == NULL){ + printf("ERROR: PRU servo Controller not initialized\n"); + return -1; + } + // PRU runs at 200Mhz. find #loops needed + unsigned int num_loops = ((us*200.0)/PRU_SERVO_LOOP_INSTRUCTIONS); + // printf("num_loops: %d\n", num_loops); + // write to PRU shared memory + prusharedMem_32int_ptr[ch-1] = num_loops; + return 0; +} + +int main(int argc, char *argv[]) +{ + unsigned int *pru; // Points to start of PRU memory. + int fd; + printf("Servo tester\n"); + + fd = open ("/dev/mem", O_RDWR | O_SYNC); + if (fd == -1) { + printf ("ERROR: could not open /dev/mem.\n\n"); + return 1; + } + pru = mmap (0, PRU_LEN, PROT_READ | PROT_WRITE, MAP_SHARED, fd, PRU_ADDR); + if (pru == MAP_FAILED) { + printf ("ERROR: could not map memory.\n\n"); + return 1; + } + close(fd); + printf ("Using /dev/mem.\n"); + + prusharedMem_32int_ptr = pru + PRU_SHAREDMEM/4; // Points to start of shared memory + + // while(1) { + // printf("value to store: "); + // scanf("%d", &value); + // printf("Storing: %d in %lx\n", value, addr); + // pru[addr/4] = value; + // } + + int i; + while(1) { + for(i=1; i<=SERVO_CHANNELS; i++) { + send_servo_pulse_us(i, 10*i); + } + usleep(200); + } + + if(munmap(pru, PRU_LEN)) { + printf("munmap failed\n"); + } else { + printf("munmap succeeded\n"); + } +} + diff --git a/pru-cookbook/01case/code/servos_setup.sh b/pru-cookbook/01case/code/servos_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..330e8f34e30d7fd97d9368861ded65817222e903 --- /dev/null +++ b/pru-cookbook/01case/code/servos_setup.sh @@ -0,0 +1,24 @@ +#!/bin/bash +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P8_27 P8_28 P8_29 P8_30 P8_39 P8_40 P8_41 P8_42" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_35 P1_35 P1_02 P1_04" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done \ No newline at end of file diff --git a/pru-cookbook/01case/figures/LEDblink.png b/pru-cookbook/01case/figures/LEDblink.png new file mode 100644 index 0000000000000000000000000000000000000000..793795c761196f2d92ee22b590f81693433b5646 Binary files /dev/null and b/pru-cookbook/01case/figures/LEDblink.png differ diff --git a/pru-cookbook/01case/figures/beaglelogic_capture.png b/pru-cookbook/01case/figures/beaglelogic_capture.png new file mode 100644 index 0000000000000000000000000000000000000000..afa771caec1d7b66ba59f0bc6d649ba010fad9e8 Binary files /dev/null and b/pru-cookbook/01case/figures/beaglelogic_capture.png differ diff --git a/pru-cookbook/01case/figures/blue.png b/pru-cookbook/01case/figures/blue.png new file mode 100644 index 0000000000000000000000000000000000000000..035b5d3b9b46fc97e145c28505181de6d5f46091 Binary files /dev/null and b/pru-cookbook/01case/figures/blue.png differ diff --git a/pru-cookbook/01case/figures/fpp_channel_inputs.png b/pru-cookbook/01case/figures/fpp_channel_inputs.png new file mode 100644 index 0000000000000000000000000000000000000000..ef0c191bcc35eb63ccc868b84a7993555cc581aa Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_channel_inputs.png differ diff --git a/pru-cookbook/01case/figures/fpp_channel_outputs.png b/pru-cookbook/01case/figures/fpp_channel_outputs.png new file mode 100644 index 0000000000000000000000000000000000000000..8b2658b766080bac7457ebfd4db479f06e54a1f4 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_channel_outputs.png differ diff --git a/pru-cookbook/01case/figures/fpp_channel_outputs_menu.png b/pru-cookbook/01case/figures/fpp_channel_outputs_menu.png new file mode 100644 index 0000000000000000000000000000000000000000..7f3984c868401454ee98e4dccfba2244bff5971a Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_channel_outputs_menu.png differ diff --git a/pru-cookbook/01case/figures/fpp_channel_outputs_strings.png b/pru-cookbook/01case/figures/fpp_channel_outputs_strings.png new file mode 100644 index 0000000000000000000000000000000000000000..4f6ab170207cee07848cd1a3aa80a89b70d9d94f Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_channel_outputs_strings.png differ diff --git a/pru-cookbook/01case/figures/fpp_display_testing.png b/pru-cookbook/01case/figures/fpp_display_testing.png new file mode 100644 index 0000000000000000000000000000000000000000..515781eaf4dcc076ab6ab633c99629ccd42ad259 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_display_testing.png differ diff --git a/pru-cookbook/01case/figures/fpp_display_testing2.png b/pru-cookbook/01case/figures/fpp_display_testing2.png new file mode 100644 index 0000000000000000000000000000000000000000..78a6ba07115dfe5dec7fcc9321192ba8bac63ed0 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_display_testing2.png differ diff --git a/pru-cookbook/01case/figures/fpp_display_testing_menu.png b/pru-cookbook/01case/figures/fpp_display_testing_menu.png new file mode 100644 index 0000000000000000000000000000000000000000..6cb7b134f663162ddac1c6f1bbd8986eab06336b Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_display_testing_menu.png differ diff --git a/pru-cookbook/01case/figures/fpp_display_testing_menu2.png b/pru-cookbook/01case/figures/fpp_display_testing_menu2.png new file mode 100644 index 0000000000000000000000000000000000000000..c5a5b33b6a003a9b10d5f3c96e84fe105679ab7a Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_display_testing_menu2.png differ diff --git a/pru-cookbook/01case/figures/fpp_file_manager.png b/pru-cookbook/01case/figures/fpp_file_manager.png new file mode 100644 index 0000000000000000000000000000000000000000..958bfab9125a87a7d546f63be35b7b98cb92be88 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_file_manager.png differ diff --git a/pru-cookbook/01case/figures/fpp_inputs_setup.png b/pru-cookbook/01case/figures/fpp_inputs_setup.png new file mode 100644 index 0000000000000000000000000000000000000000..32046a8ff9bc41d9469ab48c8a91ffcbe010fd15 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_inputs_setup.png differ diff --git a/pru-cookbook/01case/figures/fpp_inputs_setup2.png b/pru-cookbook/01case/figures/fpp_inputs_setup2.png new file mode 100644 index 0000000000000000000000000000000000000000..f721e6c5e15ed54e8e84db7861b3e5301fc76422 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_inputs_setup2.png differ diff --git a/pru-cookbook/01case/figures/fpp_mode_bridge.png b/pru-cookbook/01case/figures/fpp_mode_bridge.png new file mode 100644 index 0000000000000000000000000000000000000000..9b3366565ff44237767210c9883cbce463837aaa Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_mode_bridge.png differ diff --git a/pru-cookbook/01case/figures/fpp_playlist.png b/pru-cookbook/01case/figures/fpp_playlist.png new file mode 100644 index 0000000000000000000000000000000000000000..5c8fa072db13cf23e6254ea5da9dad63eb1031d4 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_playlist.png differ diff --git a/pru-cookbook/01case/figures/fpp_playlist_status.png b/pru-cookbook/01case/figures/fpp_playlist_status.png new file mode 100644 index 0000000000000000000000000000000000000000..1994a186874a94817094f036af3a5b8ad84ecf80 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_playlist_status.png differ diff --git a/pru-cookbook/01case/figures/fpp_program_control.png b/pru-cookbook/01case/figures/fpp_program_control.png new file mode 100644 index 0000000000000000000000000000000000000000..b3b3662d029234f9884c08e5e71bf86792d70c6a Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_program_control.png differ diff --git a/pru-cookbook/01case/figures/fpp_status.png b/pru-cookbook/01case/figures/fpp_status.png new file mode 100644 index 0000000000000000000000000000000000000000..48748134234085f6edb3dc98693e8124307211d8 Binary files /dev/null and b/pru-cookbook/01case/figures/fpp_status.png differ diff --git a/pru-cookbook/01case/figures/ledmatrix.jpg b/pru-cookbook/01case/figures/ledmatrix.jpg new file mode 100644 index 0000000000000000000000000000000000000000..20f52dc3c56b5ab982b17a808416f5c2d2a3957c Binary files /dev/null and b/pru-cookbook/01case/figures/ledmatrix.jpg differ diff --git a/pru-cookbook/01case/figures/pocketscroller.jpg b/pru-cookbook/01case/figures/pocketscroller.jpg new file mode 100644 index 0000000000000000000000000000000000000000..972e803dde8a7f6af7f6c979b6a5ff3d31796fab Binary files /dev/null and b/pru-cookbook/01case/figures/pocketscroller.jpg differ diff --git a/pru-cookbook/01case/figures/xlights_layout.png b/pru-cookbook/01case/figures/xlights_layout.png new file mode 100644 index 0000000000000000000000000000000000000000..7a3e6930b704d791935aac4dfa5400ce3ee30638 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_layout.png differ diff --git a/pru-cookbook/01case/figures/xlights_layout_details.png b/pru-cookbook/01case/figures/xlights_layout_details.png new file mode 100644 index 0000000000000000000000000000000000000000..393a0ead4923f018163816ad3b0819b37c1b8745 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_layout_details.png differ diff --git a/pru-cookbook/01case/figures/xlights_seq_drag.png b/pru-cookbook/01case/figures/xlights_seq_drag.png new file mode 100644 index 0000000000000000000000000000000000000000..5dbbd14a9adce10c5b1c2f48da985659cc878520 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_seq_drag.png differ diff --git a/pru-cookbook/01case/figures/xlights_seq_new.png b/pru-cookbook/01case/figures/xlights_seq_new.png new file mode 100644 index 0000000000000000000000000000000000000000..840d8d0021a2f695eb7190c151c69adf3aedbf62 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_seq_new.png differ diff --git a/pru-cookbook/01case/figures/xlights_setup.png b/pru-cookbook/01case/figures/xlights_setup.png new file mode 100644 index 0000000000000000000000000000000000000000..4fd1d9a67db0c2854a5618066b7068d0c6b3cfb1 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_setup.png differ diff --git a/pru-cookbook/01case/figures/xlights_setup_done.png b/pru-cookbook/01case/figures/xlights_setup_done.png new file mode 100644 index 0000000000000000000000000000000000000000..1a6036f09ffcfdd0616e9cbcdd2deb9ebe860abf Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_setup_done.png differ diff --git a/pru-cookbook/01case/figures/xlights_setup_e1_31.png b/pru-cookbook/01case/figures/xlights_setup_e1_31.png new file mode 100644 index 0000000000000000000000000000000000000000..66ab80a082f756ae04dcfb9c35e57706cd7a2041 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_setup_e1_31.png differ diff --git a/pru-cookbook/01case/figures/xlights_setup_e1_31.png.orig b/pru-cookbook/01case/figures/xlights_setup_e1_31.png.orig new file mode 100644 index 0000000000000000000000000000000000000000..40d6f0789b4b1e8eeaa7b6ba3956769612eb96ac Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_setup_e1_31.png.orig differ diff --git a/pru-cookbook/01case/figures/xlights_test.png b/pru-cookbook/01case/figures/xlights_test.png new file mode 100644 index 0000000000000000000000000000000000000000..4023a4ceb00b045c71a872252a763dbb6623eef3 Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_test.png differ diff --git a/pru-cookbook/01case/figures/xlights_twinkle.jpg b/pru-cookbook/01case/figures/xlights_twinkle.jpg new file mode 100644 index 0000000000000000000000000000000000000000..d98d1fd7991ce1d1dea2e60c769854127c28ad8e Binary files /dev/null and b/pru-cookbook/01case/figures/xlights_twinkle.jpg differ diff --git a/pru-cookbook/02start/code/Makefile b/pru-cookbook/02start/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/02start/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/02start/code/ai.notes b/pru-cookbook/02start/code/ai.notes new file mode 100644 index 0000000000000000000000000000000000000000..d11ac1d4288af4ac3f1cd5a5aa1ac6cec14cb97f --- /dev/null +++ b/pru-cookbook/02start/code/ai.notes @@ -0,0 +1,59 @@ +Here are some differences between the AI and the Black for the PRU. + +Black +/sys/devices/platform/ocp/ + 4a326004.pruss-soc-bus/ + 4a300000.pruss/ + 4a334000.pru/remoteproc/remoteproc1 + 4a338000.pru/remoteproc/remoteproc2 + + +AI +/sys/devices/platform/44000000.ocp/ + 4b226004.pruss-soc-bus/ + 4b200000.pruss/ + 4b234000.pru/remoteproc/remoteproc0 + 4b238000.pru/remoteproc/remoteproc1 + + + 4b2a6004.pruss-soc-bus/ + 4b280000.pruss/ + 4b2b4000.pru/remoteproc/remoteproc2 + 4b2b8000.pru/remoteproc/remoteproc3 + +LED addresses +https://github.com/beagleboard/BeagleBoard-DeviceTrees/blob/v4.19.x-ti/src/arm/am5729-beagleboneai.dts#L134-L171 + led0 { + label = "beaglebone:green:usr0"; + gpios = <&gpio3 17 GPIO_ACTIVE_HIGH>; + linux,default-trigger = "heartbeat"; + default-state = "off"; + }; + + led1 { + label = "beaglebone:green:usr1"; + gpios = <&gpio5 5 GPIO_ACTIVE_HIGH>; + linux,default-trigger = "mmc0"; + default-state = "off"; + }; + + led2 { + label = "beaglebone:green:usr2"; + gpios = <&gpio3 15 GPIO_ACTIVE_HIGH>; + linux,default-trigger = "cpu"; + default-state = "off"; + }; + + led3 { + label = "beaglebone:green:usr3"; + gpios = <&gpio3 14 GPIO_ACTIVE_HIGH>; + linux,default-trigger = "mmc1"; + default-state = "off"; + }; + + led4 { + label = "beaglebone:green:usr4"; + gpios = <&gpio3 7 GPIO_ACTIVE_HIGH>; + linux,default-trigger = "phy0assoc"; + +GPIO Addresses: Page 404 for TRM \ No newline at end of file diff --git a/pru-cookbook/02start/code/hello.pru0.c b/pru-cookbook/02start/code/hello.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..ab61e79eaafed3a7198532358d9ae3415559048f --- /dev/null +++ b/pru-cookbook/02start/code/hello.pru0.c @@ -0,0 +1,35 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + uint32_t *gpio1 = (uint32_t *)GPIO1; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<10; i++) { + gpio1[GPIO_SETDATAOUT] = USR3; // The the USR3 LED on + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio1[GPIO_CLEARDATAOUT] = USR3; + + __delay_cycles(500000000/5); + + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr3/trigger\0none\0" \ + "\0\0"; diff --git a/pru-cookbook/02start/code/hello.pru1_1.c b/pru-cookbook/02start/code/hello.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..8cb8146f74c88ee890a5a251108c19642cdb0c75 --- /dev/null +++ b/pru-cookbook/02start/code/hello.pru1_1.c @@ -0,0 +1,35 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + uint32_t *gpio3 = (uint32_t *)GPIO3; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<10; i++) { + gpio3[GPIO_SETDATAOUT] = USR3; // The the USR3 LED on + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio3[GPIO_CLEARDATAOUT] = USR3; + + __delay_cycles(500000000/5); + + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr3/trigger\0none\0" \ + "\0\0"; diff --git a/pru-cookbook/02start/code/hello2.pru0.c b/pru-cookbook/02start/code/hello2.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..d268d33b637cb9d51a9928589a69ce158101a4e2 --- /dev/null +++ b/pru-cookbook/02start/code/hello2.pru0.c @@ -0,0 +1,42 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + uint32_t *gpio1 = (uint32_t *)GPIO1; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<10; i++) { + gpio1[GPIO_SETDATAOUT] = USR1; // The the USR3 LED on + gpio1[GPIO_CLEARDATAOUT] = USR2; + + // __R30 |= gpio; // Set the GPIO pin to 1 + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio1[GPIO_CLEARDATAOUT] = USR1; + gpio1[GPIO_SETDATAOUT] = USR2; + + // __R30 &= ~gpio; // Clearn the GPIO pin + + __delay_cycles(500000000/5); + + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr1/trigger\0none\0" \ + "/sys/class/leds/beaglebone:green:usr2/trigger\0none\0" \ + "\0\0"; diff --git a/pru-cookbook/02start/code/hello2.pru1.c b/pru-cookbook/02start/code/hello2.pru1.c new file mode 100644 index 0000000000000000000000000000000000000000..15df54495ecf681b3707a42dfaa6dfae65990559 --- /dev/null +++ b/pru-cookbook/02start/code/hello2.pru1.c @@ -0,0 +1,43 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + uint32_t *gpio1 = (uint32_t *)GPIO1; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<10; i++) { + gpio1[GPIO_SETDATAOUT] = USR1; // The the USR3 LED on + gpio1[GPIO_CLEARDATAOUT] = USR2; + + // __R30 |= gpio; // Set the GPIO pin to 1 + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio1[GPIO_CLEARDATAOUT] = USR1; + gpio1[GPIO_SETDATAOUT] = USR2; + + // __R30 &= ~gpio; // Clearn the GPIO pin + + __delay_cycles(500000000/5); + + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr1/trigger\0none\0" \ + "/sys/class/leds/beaglebone:green:usr2/trigger\0none\0" \ + "\0\0"; + diff --git a/pru-cookbook/02start/code/hello2.pru1_0.c b/pru-cookbook/02start/code/hello2.pru1_0.c new file mode 100644 index 0000000000000000000000000000000000000000..564e1bc9b8c0fe7df8c21df6594f519d92470690 --- /dev/null +++ b/pru-cookbook/02start/code/hello2.pru1_0.c @@ -0,0 +1,63 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + // uint32_t *gpio1 = (uint32_t *)GPIO1; + // uint32_t *gpio2 = (uint32_t *)GPIO2; + uint32_t *gpio3 = (uint32_t *)GPIO3; + // uint32_t *gpio4 = (uint32_t *)GPIO4; + uint32_t *gpio5 = (uint32_t *)GPIO5; + uint32_t *gpio6 = (uint32_t *)GPIO6; + // uint32_t *gpio7 = (uint32_t *)GPIO7; + uint32_t *gpio8 = (uint32_t *)GPIO8; + + // Select which pins to toggle. These are all on pru1_1 + uint32_t gpio = P9_16 | P8_15 | P8_16 | P8_26; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<100; i++) { + gpio5[GPIO_SETDATAOUT] = USR1; // Turn the USR1 LED on + gpio3[GPIO_CLEARDATAOUT] = USR2; + gpio8[GPIO_SETDATAOUT] = P8_17; + gpio6[GPIO_SETDATAOUT] = P9_25; + + __R30 |= gpio; // Set the GPIO pin to 1 + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio5[GPIO_CLEARDATAOUT] = USR1; + gpio3[GPIO_SETDATAOUT] = USR2; + gpio8[GPIO_CLEARDATAOUT] = P8_17; + gpio6[GPIO_CLEARDATAOUT] = P9_25; + + __R30 &= ~gpio; // Clear the GPIO pin + + __delay_cycles(500000000/5); + + if((__R31&P8_19) == P8_19) { + gpio3[GPIO_CLEARDATAOUT] = USR3; // Turn on LED + } else + gpio3[GPIO_SETDATAOUT] = USR3; // Turn off LED + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr1/trigger\0none\0" \ + "/sys/class/leds/beaglebone:green:usr2/trigger\0none\0" \ + "/sys/class/gpio/export\0 177\0" \ + "/sys/class/gpio/gpio177/direction\0out\0" \ + "\0\0"; + diff --git a/pru-cookbook/02start/code/hello2.pru1_1.c b/pru-cookbook/02start/code/hello2.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..564e1bc9b8c0fe7df8c21df6594f519d92470690 --- /dev/null +++ b/pru-cookbook/02start/code/hello2.pru1_1.c @@ -0,0 +1,63 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + // uint32_t *gpio1 = (uint32_t *)GPIO1; + // uint32_t *gpio2 = (uint32_t *)GPIO2; + uint32_t *gpio3 = (uint32_t *)GPIO3; + // uint32_t *gpio4 = (uint32_t *)GPIO4; + uint32_t *gpio5 = (uint32_t *)GPIO5; + uint32_t *gpio6 = (uint32_t *)GPIO6; + // uint32_t *gpio7 = (uint32_t *)GPIO7; + uint32_t *gpio8 = (uint32_t *)GPIO8; + + // Select which pins to toggle. These are all on pru1_1 + uint32_t gpio = P9_16 | P8_15 | P8_16 | P8_26; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<100; i++) { + gpio5[GPIO_SETDATAOUT] = USR1; // Turn the USR1 LED on + gpio3[GPIO_CLEARDATAOUT] = USR2; + gpio8[GPIO_SETDATAOUT] = P8_17; + gpio6[GPIO_SETDATAOUT] = P9_25; + + __R30 |= gpio; // Set the GPIO pin to 1 + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio5[GPIO_CLEARDATAOUT] = USR1; + gpio3[GPIO_SETDATAOUT] = USR2; + gpio8[GPIO_CLEARDATAOUT] = P8_17; + gpio6[GPIO_CLEARDATAOUT] = P9_25; + + __R30 &= ~gpio; // Clear the GPIO pin + + __delay_cycles(500000000/5); + + if((__R31&P8_19) == P8_19) { + gpio3[GPIO_CLEARDATAOUT] = USR3; // Turn on LED + } else + gpio3[GPIO_SETDATAOUT] = USR3; // Turn off LED + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr1/trigger\0none\0" \ + "/sys/class/leds/beaglebone:green:usr2/trigger\0none\0" \ + "/sys/class/gpio/export\0 177\0" \ + "/sys/class/gpio/gpio177/direction\0out\0" \ + "\0\0"; + diff --git a/pru-cookbook/02start/code/hello2.pru2_0.c b/pru-cookbook/02start/code/hello2.pru2_0.c new file mode 100644 index 0000000000000000000000000000000000000000..564e1bc9b8c0fe7df8c21df6594f519d92470690 --- /dev/null +++ b/pru-cookbook/02start/code/hello2.pru2_0.c @@ -0,0 +1,63 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + // uint32_t *gpio1 = (uint32_t *)GPIO1; + // uint32_t *gpio2 = (uint32_t *)GPIO2; + uint32_t *gpio3 = (uint32_t *)GPIO3; + // uint32_t *gpio4 = (uint32_t *)GPIO4; + uint32_t *gpio5 = (uint32_t *)GPIO5; + uint32_t *gpio6 = (uint32_t *)GPIO6; + // uint32_t *gpio7 = (uint32_t *)GPIO7; + uint32_t *gpio8 = (uint32_t *)GPIO8; + + // Select which pins to toggle. These are all on pru1_1 + uint32_t gpio = P9_16 | P8_15 | P8_16 | P8_26; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<100; i++) { + gpio5[GPIO_SETDATAOUT] = USR1; // Turn the USR1 LED on + gpio3[GPIO_CLEARDATAOUT] = USR2; + gpio8[GPIO_SETDATAOUT] = P8_17; + gpio6[GPIO_SETDATAOUT] = P9_25; + + __R30 |= gpio; // Set the GPIO pin to 1 + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio5[GPIO_CLEARDATAOUT] = USR1; + gpio3[GPIO_SETDATAOUT] = USR2; + gpio8[GPIO_CLEARDATAOUT] = P8_17; + gpio6[GPIO_CLEARDATAOUT] = P9_25; + + __R30 &= ~gpio; // Clear the GPIO pin + + __delay_cycles(500000000/5); + + if((__R31&P8_19) == P8_19) { + gpio3[GPIO_CLEARDATAOUT] = USR3; // Turn on LED + } else + gpio3[GPIO_SETDATAOUT] = USR3; // Turn off LED + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr1/trigger\0none\0" \ + "/sys/class/leds/beaglebone:green:usr2/trigger\0none\0" \ + "/sys/class/gpio/export\0 177\0" \ + "/sys/class/gpio/gpio177/direction\0out\0" \ + "\0\0"; + diff --git a/pru-cookbook/02start/code/hello2.pru2_1.c b/pru-cookbook/02start/code/hello2.pru2_1.c new file mode 100644 index 0000000000000000000000000000000000000000..564e1bc9b8c0fe7df8c21df6594f519d92470690 --- /dev/null +++ b/pru-cookbook/02start/code/hello2.pru2_1.c @@ -0,0 +1,63 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register unsigned int __R30; +volatile register unsigned int __R31; + +void main(void) { + int i; + + // uint32_t *gpio1 = (uint32_t *)GPIO1; + // uint32_t *gpio2 = (uint32_t *)GPIO2; + uint32_t *gpio3 = (uint32_t *)GPIO3; + // uint32_t *gpio4 = (uint32_t *)GPIO4; + uint32_t *gpio5 = (uint32_t *)GPIO5; + uint32_t *gpio6 = (uint32_t *)GPIO6; + // uint32_t *gpio7 = (uint32_t *)GPIO7; + uint32_t *gpio8 = (uint32_t *)GPIO8; + + // Select which pins to toggle. These are all on pru1_1 + uint32_t gpio = P9_16 | P8_15 | P8_16 | P8_26; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + for(i=0; i<100; i++) { + gpio5[GPIO_SETDATAOUT] = USR1; // Turn the USR1 LED on + gpio3[GPIO_CLEARDATAOUT] = USR2; + gpio8[GPIO_SETDATAOUT] = P8_17; + gpio6[GPIO_SETDATAOUT] = P9_25; + + __R30 |= gpio; // Set the GPIO pin to 1 + + __delay_cycles(500000000/5); // Wait 1/2 second + + gpio5[GPIO_CLEARDATAOUT] = USR1; + gpio3[GPIO_SETDATAOUT] = USR2; + gpio8[GPIO_CLEARDATAOUT] = P8_17; + gpio6[GPIO_CLEARDATAOUT] = P9_25; + + __R30 &= ~gpio; // Clear the GPIO pin + + __delay_cycles(500000000/5); + + if((__R31&P8_19) == P8_19) { + gpio3[GPIO_CLEARDATAOUT] = USR3; // Turn on LED + } else + gpio3[GPIO_SETDATAOUT] = USR3; // Turn off LED + } + __halt(); +} + +// Turns off triggers +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr1/trigger\0none\0" \ + "/sys/class/leds/beaglebone:green:usr2/trigger\0none\0" \ + "/sys/class/gpio/export\0 177\0" \ + "/sys/class/gpio/gpio177/direction\0out\0" \ + "\0\0"; + diff --git a/pru-cookbook/02start/code/setup.sh b/pru-cookbook/02start/code/setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..eaf9943a2c375bc9f16b158994492b1198997dda --- /dev/null +++ b/pru-cookbook/02start/code/setup.sh @@ -0,0 +1,5 @@ +#!/bin/bash + +export TARGET=hello.pru0 + +echo TARGET=$TARGET diff --git a/pru-cookbook/02start/code/setup2.sh b/pru-cookbook/02start/code/setup2.sh new file mode 100755 index 0000000000000000000000000000000000000000..76dd6713f4f55d364403bf6a187b68e2e66172b0 --- /dev/null +++ b/pru-cookbook/02start/code/setup2.sh @@ -0,0 +1,5 @@ +#!/bin/bash + +export TARGET=hello2.pru1 + +echo TARGET=$TARGET diff --git a/pru-cookbook/02start/figures/BB_AI_BeautyAngle_800px.jpg b/pru-cookbook/02start/figures/BB_AI_BeautyAngle_800px.jpg new file mode 100644 index 0000000000000000000000000000000000000000..0070dd54ebd10285f016e792cdfbf862b273cf1c Binary files /dev/null and b/pru-cookbook/02start/figures/BB_AI_BeautyAngle_800px.jpg differ diff --git a/pru-cookbook/02start/figures/PocketBeagle-size-compare-small.jpg b/pru-cookbook/02start/figures/PocketBeagle-size-compare-small.jpg new file mode 100644 index 0000000000000000000000000000000000000000..f7d440bdab452fd3251a136bcc652018cacf8357 Binary files /dev/null and b/pru-cookbook/02start/figures/PocketBeagle-size-compare-small.jpg differ diff --git a/pru-cookbook/02start/figures/beagle-blue.png b/pru-cookbook/02start/figures/beagle-blue.png new file mode 100644 index 0000000000000000000000000000000000000000..8777a254b074347d31b151f1cad690bdf71be39c Binary files /dev/null and b/pru-cookbook/02start/figures/beagle-blue.png differ diff --git a/pru-cookbook/02start/figures/c9.png b/pru-cookbook/02start/figures/c9.png new file mode 100644 index 0000000000000000000000000000000000000000..edb7a272bf57253b007ceaf931e8a6e0696a198d Binary files /dev/null and b/pru-cookbook/02start/figures/c9.png differ diff --git a/pru-cookbook/02start/figures/c9ShowHome.png b/pru-cookbook/02start/figures/c9ShowHome.png new file mode 100644 index 0000000000000000000000000000000000000000..91f268137135041853691ba981c6cfd0664108f3 Binary files /dev/null and b/pru-cookbook/02start/figures/c9ShowHome.png differ diff --git a/pru-cookbook/02start/figures/edumip.png b/pru-cookbook/02start/figures/edumip.png new file mode 100644 index 0000000000000000000000000000000000000000..e0faa0174e27f3daac479940334babd3a37c1bd6 Binary files /dev/null and b/pru-cookbook/02start/figures/edumip.png differ diff --git a/pru-cookbook/02start/figures/etcher.png b/pru-cookbook/02start/figures/etcher.png new file mode 100644 index 0000000000000000000000000000000000000000..6c7c92fc6ea019c2025ef93194d28b88ee841b1f Binary files /dev/null and b/pru-cookbook/02start/figures/etcher.png differ diff --git a/pru-cookbook/02start/figures/latest-images.png b/pru-cookbook/02start/figures/latest-images.png new file mode 100644 index 0000000000000000000000000000000000000000..c6351706e16cea91b8084121e2c393686d329344 Binary files /dev/null and b/pru-cookbook/02start/figures/latest-images.png differ diff --git a/pru-cookbook/02start/figures/product_detail_black_sm.jpg b/pru-cookbook/02start/figures/product_detail_black_sm.jpg new file mode 100644 index 0000000000000000000000000000000000000000..e26e0c27fb7527d0c3a7ed75c818c8ae0c8dec90 Binary files /dev/null and b/pru-cookbook/02start/figures/product_detail_black_sm.jpg differ diff --git a/pru-cookbook/02start/start.rst b/pru-cookbook/02start/start.rst new file mode 100644 index 0000000000000000000000000000000000000000..dee7273df5bb6ed688a21156b65cf02433d76667 --- /dev/null +++ b/pru-cookbook/02start/start.rst @@ -0,0 +1,396 @@ +.. _pru-cookbook-start: + +Getting Started +################ + +We assume you have some experience with the Beagle and are here to learn about +the PRU. This chapter discusses what Beagles are out there, how to load the +latest software image on your beagle, how to run the Cloud9 IDE and how to +blink an LED. + +If you already have your Beagle and know your way around it, you can find the +code (and the whole book) on the PRU Cookbook github site: +https://github.com/MarkAYoder/PRUCookbook. + +Selecting a Beagle +==================== + +Problem +---------- + +Which Beagle should you use? + +Solution +---------- + +http://beagleboard.org/boards lists the many Beagles from which to choose. +Here we'll give examples for the venerable `BeagleBone Black <http://beagleboard.org/black>`_, +the robotics `BeagleBone Blue <http://beagleboard.org/blue>`_, +tiny `PockeBeagle <http://beagleboard.org/pocket>`_ and the powerful `AI <http://beagleboard.org/ai>`_. +All the examples should also run on the other Beagles too. + +Discussion +------------ + +BeagleBone Black +~~~~~~~~~~~~~~~~~ + +If you aren't sure which Beagle to use, it's hard to go wrong with the +`BeagleBone Black <http://beagleboard.org/black>`_. It's the most popular +member of the open hardware Beagle family. + +.. _start_black: + +BeagleBone Black +~~~~~~~~~~~~~~~~~ + +.. figure:: figures/product_detail_black_sm.jpg + :align: center + :alt: BeableBone Black + +The Black has: + +* AM335x 1GHz ARM® Cortex-A8 processor +* 512MB DDR3 RAM +* 4GB 8-bit eMMC on-board flash storage +* 3D graphics accelerator +* NEON floating-point accelerator +* 2x PRU 32-bit microcontrollers +* USB client for power & communications +* USB host +* Ethernet +* HDMI +* 2x 46 pin headers + +See http://beagleboard.org/black for more details. + +BeagleBone Blue +~~~~~~~~~~~~~~~~~ + +The `Blue <http://beagleboard.org/blue>`_ is a good choice if you are doing robotics. + +.. _start_blue: + +BeagleBone Blue +~~~~~~~~~~~~~~~~ + +.. figure:: figures/beagle-blue.png + :align: center + :alt: BeagleBone Blue + +The Blue has everything the Black has except it has no Ethernet or HDMI. +But it also has: + +* Wireless: 802.11bgn, Bluetooth 4.1 and BLE +* Battery support: 2-cell LiPo with balancing, LED state-of-charge monitor +* Charger input: 9-18V +* Motor control: 8 6V servo out, 4 bidirectional DC motor out, 4 quadrature encoder in +* Sensors: 9 axis IMU (accels, gyros, magnetometer), barometer, thermometer +* User interface: 11 user programmable LEDs, 2 user programmable buttons + +In addition you can mount the Blue on the +https://www.renaissancerobotics.com/eduMIP.html[EduMIP kit] as shown in +:ref:`start_edumip` to get a balancing robot. + +.. _start_edumip: + +BeagleBone Blue EduMIP Kit +~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/edumip.png + :align: center + :alt: BeagleBone Blue EduMIP Kit + +https://www.hackster.io/53815/controlling-edumip-with-ni-labview-2005f8 +shows how to assemble the robot and control it from +`LabVIEW <http://www.ni.com/en-us/shop/labview.html>`_. + +PocketBeagle +~~~~~~~~~~~~~~ + +The `PocketBeagle <http://beagleboard.org/pocket>`_ is the smallest member of the +Beagle family. It is an ultra-tiny-yet-complete Beagle that is software +compatible with the other Beagles. + +.. _start_pocket: + +PocketBeagle +~~~~~~~~~~~~~ + +.. figure:: figures/PocketBeagle-size-compare-small.jpg + :align: center + :alt: PocketBeagle + +The Pocket is based on the same processor as the Black and Blue and has: + +* 8 analog inputs +* 44 digital I/Os and +* numerous digital interface peripherals + +See http://beagleboard.org/pocket for more details. + +BeagleBone AI +~~~~~~~~~~~~~~~~~ + +If you want to do deep learning, try the `BeagleBone AI <http://beagleboard.org/ai>`_. + +.. _start_ai: + +BeagleBone AI +~~~~~~~~~~~~~~ + +.. figure:: figures/BB_AI_BeautyAngle_800px.jpg + :align: center + :alt: BeableBone AI + +The AI has: + +* Dual Arm® Cortex®-A15 microprocessor subsystem +* 2 C66x floating-point VLIW DSPs +* 2.5MB of on-chip L3 RAM +* 2x dual Arm® Cortex®-M4 co-processors +* 4x Embedded Vision Engines (EVEs) +* 2x dual-core Programmable Real-Time Unit and Industrial Communication SubSystem (PRU-ICSS) +* 2D-graphics accelerator (BB2D) subsystem +* Dual-core PowerVR® SGX544â„¢ 3D GPU +* IVA-HD subsystem (4K @ 15fps encode and decode support for H.264, 1080p60 for others) +* BeagleBone Black mechanical and header compatibility +* 1GB RAM and 16GB on-board eMMC flash with high-speed interface +* USB type-C for power and superspeed dual-role controller; and USB type-A host +* Gigabit Ethernet, 2.4/5GHz WiFi, and Bluetooth +* microHDMI +* Zero-download out-of-box software experience with Debian GNU/Linux + + +Installing the Latest OS on Your Bone +====================================== + +Problem +--------- + +You want to find the lastest version of Debian that is available for your Bone. + +Solution +--------- + +On your host computer open a browser and go to +http://beagleboard.org/latest-images. + +.. TODO Update links + +This shows you two current choices of recent Debian images, +one for the BeagleBone AI +(`AM5729 Debian 10.3 2020-04-06 8GB SD IoT TIDL <https://debian.beagleboard.org/images/am57xx-debian-10.3-iot-tidl-armhf-2020-04-06-6gb.img.xz>`_) and +one for all the other Beagles ( +`AM3358 Debian 10.3 2020-04-06 4GB SD IoT <https://debian.beagleboard.org/images/bone-debian-10.3-iot-armhf-2020-04-06-4gb.img.xz>`_). +Download the one for your Beagle. + +Latest Debian images +~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/latest-images.png + :align: center + :alt: Latest Debian images + +It contains all the packages we'll need. + +Flashing a Micro SD Card +========================= + +Problem +--------- + +I've downloaded the image and need to flash my micro SD card. + +Solution +--------- + +Get a micro SD card that has at least 4GB and preferibly 8GB. + +There are many ways to flash the card, but the best seems to be Etcher by +https://www.balena.io/. Go to https://www.balena.io/etcher/ and download the version for your host +computer. Fire up Etcher, select the image you just downloaded (no need to +uncompress it, Etcher does it for you), select the SD card and hit the *Flash* +button and wait for it to finish. + +.. _start_etcher: + +Etcher +~~~~~~~~ + +.. figure:: figures/etcher.png + :align: center + :alt: Ether + +Once the SD is flashed, insert it in the Beagle and power it up. + +Cloud9 IDE +=========== + +Problem +------------ + +How do I manage and edit my files? + +Solution +------------ + +The image you downloaded includes `Cloud9 <https://aws.amazon.com/cloud9/>`_, +a web-based intergrated development environment (IDE) as shown in +:ref:`start_c9`. + +.. _start_c9: + +Cloud9 IDE +~~~~~~~~~~~~ + +.. figure:: figures/c9.png + :align: center + :alt: The Cloud9 IDE + +Just point the browswer on your host computer to http://192.168.7.2 +and start exploring. If you want the files in your home directory to appear +in the tree structure click the settings gear and select *Show Home in Favorites* +as shown in :ref:`start_c9_show_home`. + +.. _start_c9_show_home: + +Cloud9 Showing Home files +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/c9ShowHome.png + :align: center + :alt: Cloud9 showing home files + + +If you want to edit files beyond your home directory you can link to the root file system by: + +.. code-block:: bash + + bone$ *cd* + bone$ *ln -s / root* + bone$ *cd root* + bone$ *ls* + bbb-uEnv.txt boot etc ID.txt lost+found mnt opt root sbin sys usr + bin dev home lib media nfs-uEnv.txt proc run srv tmp var + +Now you can reach all the files from Cloud9. + +Getting Example Code +==================== + +Problem +--------- + +You are ready to start playing with the examples and need to find the code. + +Solution +--------- + +You can find the code (and the whole book) on the PRU Cookbook github site: +<https://github.com/MarkAYoder/PRUCookbook/tree/master/docs>. Just clone +it on your Beagle and then look in the *docs* directory. + +.. code-block::bash + + bone$ *git clone https://github.com/MarkAYoder/PRUCookbook.git* + bone$ *cd PRUCookbook/docs/* + bone$ *ls -F* + 01case/ 05blocks/ book.adoc copyright.adoc index.html projects.adoc + 02start/ 06io/ book.html hack.sh* Makefile projects.html + 03details/ 07more/ book.pdf header.adoc notes.adoc style.adoc + 04debug/ 08ai/ common/ index.adoc notes.html style.html + + +Each chapter has its own directory and within that directory is a **code** directory that has all of the code. + +.. code-block::bash + + bone$ *cd 02start/code/* + bone$ *ls* + hello.pru0.c hello.pru1_1.c Makefile setup.sh + +Go and explore. + +Blinking an LED +================= + +Problem +--------- + +You want to make sure everything is set up by blinking an LED. + +Solution +---------- + +The 'hello, world' of the embedded world is to flash an LED. :ref:`start_hello` +is some code that blinks the ``USR3`` LED ten times using the PRU. + +.. TODO The *'s and _'s in the code are messing with the formatting. + +.. _start_hello: + +hello.pru0.c +~~~~~~~~~~~~~ + +:download:`hello.pru0.c <code/hello.pru0.c>` + +Later chapters will go into details of how this code works, but if you want +to run it right now do the following. + +.. code-block:: bash + + bone$ *git clone https://github.com/MarkAYoder/PRUCookbook.git* + bone$ *cd PRUCookbook/docs/02start/code* + +.. tip:: + + If the following doesn't work see + `Compiling with clpru and lnkpru <../03details/details.html#_compiling_with_clpru_and_lnkpru>`_ + for instillation instructions. + +.. _start_running_code: + +Running Code on the Black or Pocket +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + + bone$ *make TARGET=hello.pru0* + /var/lib/cloud9/common/Makefile:28: MODEL=TI_AM335x_BeagleBone_Black,TARGET=hello.pru0,COMMON=/var/lib/cloud9/common + /var/lib/cloud9/common/Makefile:147: GEN_DIR=/tmp/cloud9-examples,CHIP=am335x,PROC=pru,PRUN=0,PRU_DIR=/sys/class/remoteproc/remoteproc1,EXE=.out + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/hello.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + writing "none" to "/sys/class/leds/beaglebone:green:usr3/trigger" + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + + +Running Code on the AI +~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + + bone$ *make TARGET=hello.pru1_1* + /var/lib/cloud9/common/Makefile:28: MODEL=BeagleBoard.org_BeagleBone_AI,TARGET=hello.pru1_1 + - Stopping PRU 1_1 + CC hello.pru1_1.c + "/var/lib/cloud9/common/prugpio.h", line 4: warning #1181-D: #warning directive: "Found AI" + LD /tmp/cloud9-examples/hello.pru1_1.o + - copying firmware file /tmp/cloud9-examples/hello.pru1_1.out to /lib/firmware/am57xx-pru1_1-fw + write_init_pins.sh + writing "none" to "/sys/class/leds/beaglebone:green:usr3/trigger" + - Starting PRU 1_1 + MODEL = BeagleBoard.org_BeagleBone_AI + PROC = pru + PRUN = 1_1 + PRU_DIR = /dev/remoteproc/pruss1-core1 + rm /tmp/cloud9-examples/hello.pru1_1.o + +Look quickly and you will see the ``USR3`` LED blinking. + +Later sections give more details on how all this works. diff --git a/pru-cookbook/03details/code/Makefile b/pru-cookbook/03details/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/03details/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/03details/code/encoder_setup.sh b/pru-cookbook/03details/code/encoder_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..ffc4ad68842fb6314173b523425295893cd05fc1 --- /dev/null +++ b/pru-cookbook/03details/code/encoder_setup.sh @@ -0,0 +1,49 @@ +#!/bin/bash +# Configure the pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine + +# Configure eQEP pins +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_92 P9_27 P8_35 P8_33 P8_12 P8_11 P8_41 P8_42" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_31 P2_34 P2_10 P2_24 P2_33" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin qep + config-pin -q $pin +done + +########################################## +# Configure PRU pins +if [ $machine = "Black" ]; then + echo " Found" + pins="P8_16 P8_15" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_09 P2_18" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruin + config-pin -q $pin +done diff --git a/pru-cookbook/03details/code/gpio_setup.sh b/pru-cookbook/03details/code/gpio_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..e676ed7abe2a6beaf62d3ce6c56f5aa151d86700 --- /dev/null +++ b/pru-cookbook/03details/code/gpio_setup.sh @@ -0,0 +1,28 @@ +#!/bin/bash + +export TARGET=gpio.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_11" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_05" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin gpio + config-pin -q $pin +done diff --git a/pru-cookbook/03details/code/servos_setup.sh b/pru-cookbook/03details/code/servos_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..330e8f34e30d7fd97d9368861ded65817222e903 --- /dev/null +++ b/pru-cookbook/03details/code/servos_setup.sh @@ -0,0 +1,24 @@ +#!/bin/bash +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P8_27 P8_28 P8_29 P8_30 P8_39 P8_40 P8_41 P8_42" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_35 P1_35 P1_02 P1_04" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done \ No newline at end of file diff --git a/pru-cookbook/03details/details.rst b/pru-cookbook/03details/details.rst new file mode 100644 index 0000000000000000000000000000000000000000..78ece37394c81558138fce1a74b8041d46590b31 --- /dev/null +++ b/pru-cookbook/03details/details.rst @@ -0,0 +1,490 @@ +.. _pru-cookbook-details: + +Running a Program; Configuring Pins +#################################### + +There are a lot of details in compiling and running PRU code. +Fortunately those details are captured in a common `Makefile` that is +used througout this book. This chapter shows how to use the `Makefile` to +compile code and also start and stop the PRUs. + +The following are resources used in this chapter. + +Resources +~~~~~~~~~~ + +* `PRU Code Generation Tools - Compiler <http://software-dl.ti.com/codegen/esd/cgt_ai_64_lic_sw/PRU/2.1.5/ti_cgt_pru_2.1.5_armlinuxa8hf_busybox_installer.sh>`_ +* `PRU Software Support Package <http://git.ti.com/pru-software-support-package>`_ +* `PRU Optimizing C/C++ Compiler <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ +* `PRU Assembly Language Tools <http://www.ti.com/lit/ug/spruhv6b/spruhv6b.pdf>`_ +* `AM572x Technical Reference Manual <http://www.ti.com/lit/pdf/spruhz6l>`_ (AI) +* `AM335x Technical Reference Manual <http://www.ti.com/lit/pdf/spruh73>`_ (All others) + +Getting Example Code +===================== + +Problem +--------- + +I want to get the files used in this book. + +Solution +--------- + +It's all on a GitHub repository. + +.. code-block:: bash + + bone$ git clone https://github.com/MarkAYoder/PRUCookbook.git + +Compiling with clpru and lnkpru +================================ + + +Problem +--------- + +You need details on the c compiler, linker and other tools for the PRU. + +Solution +--------- + +The PRU compiler and linker are already installed on many images. +They are called ``clpru`` and ``lnkpru``. Do the following to see if ``clpru`` is installed. + +.. code-block:: bash + + bone$ which clpru + /usr/bin/clpru + +.. tip:: + If ``clpru`` isn't installed, follow the instructions at + https://elinux.org/Beagleboard:BeagleBoneBlack_Debian#TI_PRU_Code_Generation_Tools + to install it. + +.. code-block:: bash + + bone$ sudo apt update + bone$ sudo apt install ti-pru-cgt-installer + + +Details on each can be found here: + +* `PRU Optimizing C/C++ Compiler <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ +* `PRU Assembly Language Tools <http://www.ti.com/lit/ug/spruhv6b/spruhv6b.pdf>`_ + +In fact there are PRU versions of many of the standard code generation tools. + +code tools +~~~~~~~~~~~ +.. code-block:: bash + + bone$ ls /usr/bin/*pru + /usr/bin/abspru /usr/bin/clistpru /usr/bin/hexpru /usr/bin/ofdpru + /usr/bin/acpiapru /usr/bin/clpru /usr/bin/ilkpru /usr/bin/optpru + /usr/bin/arpru /usr/bin/dempru /usr/bin/libinfopru /usr/bin/rc_test_encoders_pru + /usr/bin/asmpru /usr/bin/dispru /usr/bin/lnkpru /usr/bin/strippru + /usr/bin/cgpru /usr/bin/embedpru /usr/bin/nmpru /usr/bin/xrefpru + +See the `PRU Assembly Language Tools <http://www.ti.com/lit/ug/spruhv6b/spruhv6b.pdf>`_ for more details. + +Making sure the PRUs are configured +==================================== + +Problem +--------- + +When running the Makefile for the PRU you get and error about ``/dev/remoteproc`` is missing. + +Solution +--------- + +Edit ``/boot/uEnv.txt`` and enble pru_rproc by doing the following. + +.. code-block:: bash + + bone$ *sudo vi /boot/uEnv.txt* + +Around line 40 you will see: + +.. code-block:: bash + + ###pru_rproc (4.19.x-ti kernel) + uboot_overlay_pru=AM335X-PRU-RPROC-4-19-TI-00A0.dtbo + +Uncomment the ``uboot_overlay`` line as shown and then reboot. +``/dev/remoteproc`` should now be there. + +.. code-block:: bash + + bone$ sudo reboot + bone$ ls -ls /dev/remoteproc/ + total 0 + 0 lrwxrwxrwx 1 root root 33 Jul 29 16:12 pruss-core0 -> /sys/class/remoteproc/remoteproc1 + 0 lrwxrwxrwx 1 root root 33 Jul 29 16:12 pruss-core1 -> /sys/class/remoteproc/remoteproc2 + +Compiling and Running +====================== + +Problem +--------- + +I want to compile and run an example. + +Solution +--------- + +Change to the directory of the code you want to run. + +.. code-block:: bash + + bone$ cd PRUCookbook/docs/06io/code + bone$ ls + gpio.pru0.c Makefile setup.sh + +Source the setup file. + +.. code-block:: bash + + bone$ source setup.sh + TARGET=gpio.pru0 + PocketBeagle Found + P2_05 + Current mode for P2_05 is: gpio + Current mode for P2_05 is: gpio + +Now you are ready to compile and run. This is automated for you in the Makefile + +.. code-block:: bash + + bone$ make + /var/lib/cloud9/common/Makefile:28: MODEL=TI_AM335x_BeagleBone_Black,TARGET=gpio.pru0,COMMON=/var/lib/cloud9/common + /var/lib/cloud9/common/Makefile:147: GEN_DIR=/tmp/cloud9-examples,CHIP=am335x,PROC=pru,PRUN=0,PRU_DIR=/sys/class/remoteproc/remoteproc1,EXE=.out + - Stopping PRU 0 + /bin/sh: 1: echo: echo: I/O error + Cannot stop 0 + CC gpio.pru0.c + "/var/lib/cloud9/common/prugpio.h", line 53: warning #1181-D: #warning directive: "Found am335x" + LD /tmp/cloud9-examples/gpio.pru0.o + - copying firmware file /tmp/cloud9-examples/gpio.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + writing "out" to "/sys/class/gpio/gpio30/direction" + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + rm /tmp/cloud9-examples/gpio.pru0.o + +Congratulations, your are now running a PRU. If you have an LED attached to +``P9_11`` on the Black, or ``P2_05`` on the Pocket, it should be blinking. + +Discussion +------------ + +The ``setup.sh`` file sets the ``TARGET`` to the file you want to compile. +Set it to the filename, without the ``.c`` extension (``gpio.pru0``). +The file extension ``.pru0`` specifies the number of the PRU you are using +(either ``1_0``, ``1_1``, ``2_0``, ``2_1`` on the AI or ``0`` or ``1`` on the others) + +You can override the ``TARGET`` on the command line. + +.. code-block:: bash + + bone$ cp gpio.pru0.c gpio.pru1.c + bone$ export TARGET=gpio.pru1 + +Notice the ``TARGET`` doesn't have the ``.c`` on the end. + +You can also specify them when running ``make``. +.. code-block:: bash + + bone$ cp gpio.pru0.c gpio.pru1.c + bone$ make TARGET=gpio.pru1 + +The setup file also contains instructions to figure out which Beagle you are running +and then configure the pins acordingly. + +setup.sh +~~~~~~~~~ + +:download:`gpio_setup.sh <code/gpio_setup.sh>` + +.. table:: + + +-----+---------------------------------------------------+ + |Line | Explanation | + +=====+===================================================+ + |2-5 | Set which PRU to use and which file to compile. | + +-----+---------------------------------------------------+ + |7 | Figure out which type of Beagle we have. | + +-----+---------------------------------------------------+ + |9-21 | Based on the type, set the `pins`. | + +-----+---------------------------------------------------+ + |23-28| Configure (set the pin mux) for each of the pins. | + +-----+---------------------------------------------------+ + +.. tip:: + + The BeagleBone AI has it's pins preconfigured at boot time, so there's no + need to use ``config-pin``. + + +The ``Makefile`` stops the PRU, compiles the file and moves it where it will +be loaded, and then restarts the PRU. + +Stopping and Starting the PRU +============================== + +Problem +--------- + +I want to stop and start the PRU. + +Solution +--------- + +It's easy, if you already have ``TARGET`` set up: + +.. code-block:: bash + + bone$ make stop + - Stopping PRU 0 + stop + bone$ make start + - Starting PRU 0 + start + +See :ref:`../04debug/debug.html#_dmesg_hw,dmesg -Hw` to see how to tell if the PRU +is stopped. + +This assumes ``TARGET`` is set to the PRU you are using. +If you want to control the other PRU use: + +.. code-block:: bash + + bone$ cp gpio.pru0.c gpio.pru1.c + bone$ make TARGET=gpio.pru1 + bone$ make TARGET=gpio.pru1 stop + bone$ make TARGET=gpio.pru1 start + + +.. _details_makefile: + +The Standard Makefile +===================== + +Problem +--------- + +There are all sorts of options that need to be set when compiling +a program. How can I be sure to get them all right? + +Solution +--------- + +The surest way to make sure everything is right is to use our +standard ``Makefile``. + +Discussion +----------- + +It's assumed you already know how Makefiles work. If not, there are +many resources online that can bring you up to speed. +Here is the local ``Makefile`` used throughout this book. + +Local Makefile +~~~~~~~~~~~~~~~ + +:download:`Makefile <code/Makefile>` + +Each of the local Makefiles refer to the same standard Makefile. The details +of how the Makefile works is beyond the scope of this cookbook. + +Fortunately you shouldn't have to modify the `Makefile`. + +.. _detail_linker: + +The Linker Command File - am335x_pru.cmd +========================================= + + +Problem +--------- + +The linker needs to be told where in memory to place the code and variables. + +Solution +--------- + +``am335x_pru.cmd`` is the standard linker command file that tells the linker +where to put what for the BeagleBone Black and Blue, and the Pocket. +The ``am57xx_pru.cmd`` does the same for the AI. +Both files can be found in ``/var/lib/cloud9/common``. + +am335x_pru.cmd +~~~~~~~~~~~~~~~~ + +:download:`am335x_pru.cmd <code/am335x_pru.cmd>` + +.. TODO does this need updating? + +The cmd file for the AI is about the same, with appropriate addresses for the AI. + +Discussion +----------- + + +The important things to notice in the file are given in the following table. + +AM335x_PRU.cmd important things +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +.. table:: + + +-----+-----------------------------------------------------------------------------------------+ + |Line | Explanation | + +=====+=========================================================================================+ + |16 | This is where the instructions are stored. See page 206 of the | + | | `AM335x Technical Reference Manual <https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf>`_ | + | | Or see page 417 of | + | | `AM572x Technical Reference Manual <http://www.ti.com/lit/pdf/spruhz6l>`_ for the AI. | + +-----+-----------------------------------------------------------------------------------------+ + |22 | This is where PRU 0's DMEM 0 is mapped. It's also where PRU 1's | + | | DMEM 1 is mapped. | + +-----+-----------------------------------------------------------------------------------------+ + |23 | The reverse to above. PRU 0's DMEM 1 appears here and PRU 1's DMEM 0 | + | | is here. | + +-----+-----------------------------------------------------------------------------------------+ + |26 | The shared memory for both PRU's appears here. | + +-----+-----------------------------------------------------------------------------------------+ + |72 | The `.text` section is where the code goes. It's mapped to `IMEM` | + +-----+-----------------------------------------------------------------------------------------+ + |73 | The ((stack)) is then mapped to DMEM 0. Notice that DMEM 0 is one bank | + +-----+-----------------------------------------------------------------------------------------+ + | | of memory for PRU 0 and another for PRU1, so they both get their own stacks. | + +-----+-----------------------------------------------------------------------------------------+ + |74 | The `.bss` section is where the **heap** goes. | + +-----+-----------------------------------------------------------------------------------------+ + +Why is it important to understand this file? If you are going to store things +in DMEM, you need to be sure to start at address 0x0200 since the **stack** and +the **heap** are in the locations below 0x0200. + +Loading Firmware +================== + +Problem +--------- + +I have my PRU code all compiled and need to load it on the PRU. + +Solution +--------- + +It's a simple three step process. + +* Stop the PRU +* Write the ``.out`` file to the right place in ``/lib/firmware`` +* Start the PRU. + +This is all handled in the :ref:`details_makefile`. + +Discussion +----------- + + +The PRUs appear in the Linux file space at ``/dev/remoteproc/``. + +Finding the PRUs +~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + + bone$ cd /dev/remoteproc/ + bone$ ls + pruss-core0 pruss-core1 + +Or if you are on the AI: + +.. code-block:: bash + + bone$ cd /dev/remoteproc/ + bone$ ls + dsp1 dsp2 ipu1 ipu2 pruss1-core0 pruss1-core1 pruss2-core0 pruss2-core1 + +You see there that the AI has two pairs of PRUs, plus a couple of DSPs and other goodies. + +Here we see PRU 0 and PRU 1 in the path. Let's follow PRU 0. + +.. code-block:: bash + + bone$ cd pruss-core0 + bone$ ls + device firmware name power state subsystem uevent + +Here we see the files that control PRU 0. ``firmware`` tells where in ``/lib/firmware`` +to look for the code to run on the PRU. + +.. code-block:: bash + + bone$ cat firmware + am335x-pru0-fw + +Therefore you copy your ``.out`` file to ``/lib/firmware/am335x-pru0-fw``. + +.. _details_configure_servos: + +Configuring Pins for Controlling Servos +======================================== + +Problem +--------- + +You want to **configure** the pins so the PRU outputs are accessable. + +Solution +--------- + +It depends on which Beagle you are running on. If you are on the AI or Blue, +everything is already configured for you. +If you are on the Black or Pocket you'll need to run the following script. + +servos_setup.sh +~~~~~~~~~~~~~~~~ + +:download:`servos_setup.sh <code/servos_setup.sh>` + +Discussion +----------- + +The first part of the code looks in ``/proc/device-tree/model`` to see which Beagle is running. Based on that it +assigns ``pins`` a list of pins to configure. Then the last part of the script loops through each of the pins and configures it. + + +.. _details_configure_encoders: + +Configuring Pins for Controlling Encoders +========================================== + +Problem +--------- + +You want to **configure** the pins so the PRU inputs are accessable. + +Solution +--------- + +It depends on which Beagle you are running on. If you are on the AI or Blue, +everything is already configured for you. +If you are on the Black or Pocket you'll need to run the following script. + +.encoder_setup.sh + +:download:`encoder_setup.sh <code/encoder_setup.sh>` + +Discussion +----------- + +This works like the servo setup except some of the pins are configured as +to the hardware eQEPs and other to the PRU inputs. + diff --git a/pru-cookbook/04debug/code/Makefile b/pru-cookbook/04debug/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/04debug/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/04debug/code/copyright.c b/pru-cookbook/04debug/code/copyright.c new file mode 100644 index 0000000000000000000000000000000000000000..a34918faa44a789c3624c0e5551704e630017dae --- /dev/null +++ b/pru-cookbook/04debug/code/copyright.c @@ -0,0 +1,32 @@ +/* + * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ diff --git a/pru-cookbook/04debug/code/resource_table_empty.h b/pru-cookbook/04debug/code/resource_table_empty.h new file mode 100644 index 0000000000000000000000000000000000000000..c14bd2b222644e05a84a4a328792a4c8513ff1e8 --- /dev/null +++ b/pru-cookbook/04debug/code/resource_table_empty.h @@ -0,0 +1,72 @@ +/* + * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +/* + * ======== resource_table_empty.h ======== + * + * Define the resource table entries for all PRU cores. This will be + * incorporated into corresponding base images, and used by the remoteproc + * on the host-side to allocated/reserve resources. Note the remoteproc + * driver requires that all PRU firmware be built with a resource table. + * + * This file contains an empty resource table. It can be used either as: + * + * 1) A template, or + * 2) As-is if a PRU application does not need to configure PRU_INTC + * or interact with the rpmsg driver + * + */ + +#ifndef _RSC_TABLE_PRU_H_ +#define _RSC_TABLE_PRU_H_ + +#include <stddef.h> +#include <rsc_types.h> + +struct my_resource_table { + struct resource_table base; + + uint32_t offset[1]; /* Should match 'num' in actual definition */ +}; + +#pragma DATA_SECTION(pru_remoteproc_ResourceTable, ".resource_table") +#pragma RETAIN(pru_remoteproc_ResourceTable) +struct my_resource_table pru_remoteproc_ResourceTable = { + 1, /* we're the first version that implements this */ + 0, /* number of entries in the table */ + 0, 0, /* reserved, must be zero */ + 0, /* offset[0] */ +}; + +#endif /* _RSC_TABLE_PRU_H_ */ + diff --git a/pru-cookbook/04debug/code/uart1.pru0.c b/pru-cookbook/04debug/code/uart1.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..e9b40d5c8c6fe6f1bda4492bb84160a1d4c08b3e --- /dev/null +++ b/pru-cookbook/04debug/code/uart1.pru0.c @@ -0,0 +1,88 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/trees/master/examples/am335x/PRU_Hardware_UART + +#include <stdint.h> +#include <pru_uart.h> +#include "resource_table_empty.h" + +/* The FIFO size on the PRU UART is 16 bytes; however, we are (arbitrarily) + * only going to send 8 at a time */ +#define FIFO_SIZE 16 +#define MAX_CHARS 8 + +void main(void) +{ + uint8_t tx; + uint8_t rx; + uint8_t cnt; + + /* hostBuffer points to the string to be printed */ + char* hostBuffer; + + /*** INITIALIZATION ***/ + + /* Set up UART to function at 115200 baud - DLL divisor is 104 at 16x oversample + * 192MHz / 104 / 16 = ~115200 */ + CT_UART.DLL = 104; + CT_UART.DLH = 0; + CT_UART.MDR = 0x0; + + /* Enable Interrupts in UART module. This allows the main thread to poll for + * Receive Data Available and Transmit Holding Register Empty */ + CT_UART.IER = 0x7; + + /* If FIFOs are to be used, select desired trigger level and enable + * FIFOs by writing to FCR. FIFOEN bit in FCR must be set first before + * other bits are configured */ + /* Enable FIFOs for now at 1-byte, and flush them */ + CT_UART.FCR = (0x8) | (0x4) | (0x2) | (0x1); + //CT_UART.FCR = (0x80) | (0x4) | (0x2) | (0x01); // 8-byte RX FIFO trigger + + /* Choose desired protocol settings by writing to LCR */ + /* 8-bit word, 1 stop bit, no parity, no break control and no divisor latch */ + CT_UART.LCR = 3; + + /* Enable loopback for test */ + CT_UART.MCR = 0x00; + + /* Choose desired response to emulation suspend events by configuring + * FREE bit and enable UART by setting UTRST and URRST in PWREMU_MGMT */ + /* Allow UART to run free, enable UART TX/RX */ + CT_UART.PWREMU_MGMT = 0x6001; + + /*** END INITIALIZATION ***/ + + /* Priming the 'hostbuffer' with a message */ + hostBuffer = "Hello! This is a long string\r\n"; + + /*** SEND SOME DATA ***/ + + /* Let's send/receive some dummy data */ + while(1) { + cnt = 0; + while(1) { + /* Load character, ensure it is not string termination */ + if ((tx = hostBuffer[cnt]) == '\0') + break; + cnt++; + CT_UART.THR = tx; + + /* Because we are doing loopback, wait until LSR.DR == 1 + * indicating there is data in the RX FIFO */ + while ((CT_UART.LSR & 0x1) == 0x0); + + /* Read the value from RBR */ + rx = CT_UART.RBR; + + /* Wait for TX FIFO to be empty */ + while (!((CT_UART.FCR & 0x2) == 0x2)); + } + } + + /*** DONE SENDING DATA ***/ + + /* Disable UART before halting */ + CT_UART.PWREMU_MGMT = 0x0; + + /* Halt PRU core */ + __halt(); +} diff --git a/pru-cookbook/04debug/code/uart1.pru1_0.c b/pru-cookbook/04debug/code/uart1.pru1_0.c new file mode 100644 index 0000000000000000000000000000000000000000..4c558bb96e6e21b9c0b05f9acee21f7d7ad3c225 --- /dev/null +++ b/pru-cookbook/04debug/code/uart1.pru1_0.c @@ -0,0 +1,90 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/trees/master/examples/am335x/PRU_Hardware_UART +// This example was converted to the am5729 by changing the names in pru_uart.h +// for the am335x to the more descriptive names for the am5729. +// For example DLL convertes to DIVISOR_REGISTER_LSB_ +#include <stdint.h> +#include <pru_uart.h> +#include "resource_table_empty.h" + +/* The FIFO size on the PRU UART is 16 bytes; however, we are (arbitrarily) + * only going to send 8 at a time */ +#define FIFO_SIZE 16 +#define MAX_CHARS 8 + +void main(void) +{ + uint8_t tx; + uint8_t rx; + uint8_t cnt; + + /* hostBuffer points to the string to be printed */ + char* hostBuffer; + + /*** INITIALIZATION ***/ + + /* Set up UART to function at 115200 baud - DLL divisor is 104 at 16x oversample + * 192MHz / 104 / 16 = ~115200 */ + CT_UART.DIVISOR_REGISTER_LSB_ = 104; + CT_UART.DIVISOR_REGISTER_MSB_ = 0; + CT_UART.MODE_DEFINITION_REGISTER = 0x0; + + /* Enable Interrupts in UART module. This allows the main thread to poll for + * Receive Data Available and Transmit Holding Register Empty */ + CT_UART.INTERRUPT_ENABLE_REGISTER = 0x7; + + /* If FIFOs are to be used, select desired trigger level and enable + * FIFOs by writing to FCR. FIFOEN bit in FCR must be set first before + * other bits are configured */ + /* Enable FIFOs for now at 1-byte, and flush them */ + CT_UART.INTERRUPT_IDENTIFICATION_REGISTER_FIFO_CONTROL_REGISTER = (0x8) | (0x4) | (0x2) | (0x1); + //CT_UART.FCR = (0x80) | (0x4) | (0x2) | (0x01); // 8-byte RX FIFO trigger + + /* Choose desired protocol settings by writing to LCR */ + /* 8-bit word, 1 stop bit, no parity, no break control and no divisor latch */ + CT_UART.LINE_CONTROL_REGISTER = 3; + + /* Enable loopback for test */ + CT_UART.MODEM_CONTROL_REGISTER = 0x00; + + /* Choose desired response to emulation suspend events by configuring + * FREE bit and enable UART by setting UTRST and URRST in PWREMU_MGMT */ + /* Allow UART to run free, enable UART TX/RX */ + CT_UART.POWERMANAGEMENT_AND_EMULATION_REGISTER = 0x6001; + + /*** END INITIALIZATION ***/ + + /* Priming the 'hostbuffer' with a message */ + hostBuffer = "Hello! This is a long string\r\n"; + + /*** SEND SOME DATA ***/ + + /* Let's send/receive some dummy data */ + while(1) { + cnt = 0; + while(1) { + /* Load character, ensure it is not string termination */ + if ((tx = hostBuffer[cnt]) == '\0') + break; + cnt++; + CT_UART.RBR_THR_REGISTERS = tx; + + /* Because we are doing loopback, wait until LSR.DR == 1 + * indicating there is data in the RX FIFO */ + while ((CT_UART.LINE_STATUS_REGISTER & 0x1) == 0x0); + + /* Read the value from RBR */ + rx = CT_UART.RBR_THR_REGISTERS; + + /* Wait for TX FIFO to be empty */ + while (!((CT_UART.INTERRUPT_IDENTIFICATION_REGISTER_FIFO_CONTROL_REGISTER & 0x2) == 0x2)); + } + } + + /*** DONE SENDING DATA ***/ + + /* Disable UART before halting */ + CT_UART.POWERMANAGEMENT_AND_EMULATION_REGISTER = 0x0; + + /* Halt PRU core */ + __halt(); +} diff --git a/pru-cookbook/04debug/code/uart2.pru0.c b/pru-cookbook/04debug/code/uart2.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..56ca985d5a45d1daeb2efd22b9576047791519bf --- /dev/null +++ b/pru-cookbook/04debug/code/uart2.pru0.c @@ -0,0 +1,126 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/trees/master/pru_cape/pru_fw/PRU_Hardware_UART + +#include <stdint.h> +#include <pru_uart.h> +#include "resource_table_empty.h" + +/* The FIFO size on the PRU UART is 16 bytes; however, we are (arbitrarily) + * only going to send 8 at a time */ +#define FIFO_SIZE 16 +#define MAX_CHARS 8 +#define BUFFER 40 + +//****************************************************************************** +// Print Message Out +// This function take in a string literal of any size and then fill the +// TX FIFO when it's empty and waits until there is info in the RX FIFO +// before returning. +//****************************************************************************** +void PrintMessageOut(volatile char* Message) +{ + uint8_t cnt, index = 0; + + while (1) { + cnt = 0; + + /* Wait until the TX FIFO and the TX SR are completely empty */ + while (!CT_UART.LSR_bit.TEMT); + + while (Message[index] != NULL && cnt < MAX_CHARS) { + CT_UART.THR = Message[index]; + index++; + cnt++; + } + if (Message[index] == NULL) + break; + } + + /* Wait until the TX FIFO and the TX SR are completely empty */ + while (!CT_UART.LSR_bit.TEMT); + +} + +//****************************************************************************** +// IEP Timer Config +// This function waits until there is info in the RX FIFO and then returns +// the first character entered. +//****************************************************************************** +char ReadMessageIn(void) +{ + while (!CT_UART.LSR_bit.DR); + + return CT_UART.RBR_bit.DATA; +} + +void main(void) +{ + uint32_t i; + volatile uint32_t not_done = 1; + + char rxBuffer[BUFFER]; + rxBuffer[BUFFER-1] = NULL; // null terminate the string + + /*** INITIALIZATION ***/ + + /* Set up UART to function at 115200 baud - DLL divisor is 104 at 16x oversample + * 192MHz / 104 / 16 = ~115200 */ + CT_UART.DLL = 104; + CT_UART.DLH = 0; + CT_UART.MDR_bit.OSM_SEL = 0x0; + + /* Enable Interrupts in UART module. This allows the main thread to poll for + * Receive Data Available and Transmit Holding Register Empty */ + CT_UART.IER = 0x7; + + /* If FIFOs are to be used, select desired trigger level and enable + * FIFOs by writing to FCR. FIFOEN bit in FCR must be set first before + * other bits are configured */ + /* Enable FIFOs for now at 1-byte, and flush them */ + CT_UART.FCR = (0x80) | (0x8) | (0x4) | (0x2) | (0x01); // 8-byte RX FIFO trigger + + /* Choose desired protocol settings by writing to LCR */ + /* 8-bit word, 1 stop bit, no parity, no break control and no divisor latch */ + CT_UART.LCR = 3; + + /* If flow control is desired write appropriate values to MCR. */ + /* No flow control for now, but enable loopback for test */ + CT_UART.MCR = 0x00; + + /* Choose desired response to emulation suspend events by configuring + * FREE bit and enable UART by setting UTRST and URRST in PWREMU_MGMT */ + /* Allow UART to run free, enable UART TX/RX */ + CT_UART.PWREMU_MGMT_bit.FREE = 0x1; + CT_UART.PWREMU_MGMT_bit.URRST = 0x1; + CT_UART.PWREMU_MGMT_bit.UTRST = 0x1; + + /* Turn off RTS and CTS functionality */ + CT_UART.MCR_bit.AFE = 0x0; + CT_UART.MCR_bit.RTS = 0x0; + + /*** END INITIALIZATION ***/ + + while(1) { + /* Print out greeting message */ + PrintMessageOut("Hello you are in the PRU UART demo test please enter some characters\r\n"); + + /* Read in characters from user, then echo them back out */ + for (i = 0; i < BUFFER-1 ; i++) { + rxBuffer[i] = ReadMessageIn(); + if(rxBuffer[i] == '\r') { // Quit early if ENTER is hit. + rxBuffer[i+1] = NULL; + break; + } + } + + PrintMessageOut("you typed:\r\n"); + PrintMessageOut(rxBuffer); + PrintMessageOut("\r\n"); + } + + /*** DONE SENDING DATA ***/ + /* Disable UART before halting */ + CT_UART.PWREMU_MGMT = 0x0; + + /* Halt PRU core */ + __halt(); +} diff --git a/pru-cookbook/04debug/code/uart2.pru1_0.c b/pru-cookbook/04debug/code/uart2.pru1_0.c new file mode 100644 index 0000000000000000000000000000000000000000..c934ef7d0bd645734aa2c181b5ab68646713755e --- /dev/null +++ b/pru-cookbook/04debug/code/uart2.pru1_0.c @@ -0,0 +1,126 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/trees/master/pru_cape/pru_fw/PRU_Hardware_UART + +#include <stdint.h> +#include <pru_uart.h> +#include "resource_table_empty.h" + +/* The FIFO size on the PRU UART is 16 bytes; however, we are (arbitrarily) + * only going to send 8 at a time */ +#define FIFO_SIZE 16 +#define MAX_CHARS 8 +#define BUFFER 40 + +//****************************************************************************** +// Print Message Out +// This function take in a string literal of any size and then fill the +// TX FIFO when it's empty and waits until there is info in the RX FIFO +// before returning. +//****************************************************************************** +void PrintMessageOut(volatile char* Message) +{ + uint8_t cnt, index = 0; + + while (1) { + cnt = 0; + + /* Wait until the TX FIFO and the TX SR are completely empty */ + while (!CT_UART.LINE_STATUS_REGISTER_bit.TEMT); + + while (Message[index] != NULL && cnt < MAX_CHARS) { + CT_UART.RBR_THR_REGISTERS = Message[index]; + index++; + cnt++; + } + if (Message[index] == NULL) + break; + } + + /* Wait until the TX FIFO and the TX SR are completely empty */ + while (!CT_UART.LINE_STATUS_REGISTER_bit.TEMT); + +} + +//****************************************************************************** +// IEP Timer Config +// This function waits until there is info in the RX FIFO and then returns +// the first character entered. +//****************************************************************************** +char ReadMessageIn(void) +{ + while (!CT_UART.LINE_STATUS_REGISTER_bit.DR); + + return CT_UART.RBR_THR_REGISTERS_bit.DATA; +} + +void main(void) +{ + uint32_t i; + volatile uint32_t not_done = 1; + + char rxBuffer[BUFFER]; + rxBuffer[BUFFER-1] = NULL; // null terminate the string + + /*** INITIALIZATION ***/ + + /* Set up UART to function at 115200 baud - DLL divisor is 104 at 16x oversample + * 192MHz / 104 / 16 = ~115200 */ + CT_UART.DIVISOR_REGISTER_LSB_ = 104; + CT_UART.DIVISOR_REGISTER_MSB_ = 0; + CT_UART.MODE_DEFINITION_REGISTER_bit.OSM_SEL = 0x0; + + /* Enable Interrupts in UART module. This allows the main thread to poll for + * Receive Data Available and Transmit Holding Register Empty */ + CT_UART.INTERRUPT_ENABLE_REGISTER = 0x7; + + /* If FIFOs are to be used, select desired trigger level and enable + * FIFOs by writing to FCR. FIFOEN bit in FCR must be set first before + * other bits are configured */ + /* Enable FIFOs for now at 1-byte, and flush them */ + CT_UART.INTERRUPT_IDENTIFICATION_REGISTER_FIFO_CONTROL_REGISTER = (0x80) | (0x8) | (0x4) | (0x2) | (0x01); // 8-byte RX FIFO trigger + + /* Choose desired protocol settings by writing to LCR */ + /* 8-bit word, 1 stop bit, no parity, no break control and no divisor latch */ + CT_UART.LINE_CONTROL_REGISTER = 3; + + /* If flow control is desired write appropriate values to MCR. */ + /* No flow control for now, but enable loopback for test */ + CT_UART.MODEM_CONTROL_REGISTER = 0x00; + + /* Choose desired response to emulation suspend events by configuring + * FREE bit and enable UART by setting UTRST and URRST in PWREMU_MGMT */ + /* Allow UART to run free, enable UART TX/RX */ + CT_UART.POWERMANAGEMENT_AND_EMULATION_REGISTER_bit.FREE = 0x1; + CT_UART.POWERMANAGEMENT_AND_EMULATION_REGISTER_bit.URRST = 0x1; + CT_UART.POWERMANAGEMENT_AND_EMULATION_REGISTER_bit.UTRST = 0x1; + + /* Turn off RTS and CTS functionality */ + CT_UART.MODEM_CONTROL_REGISTER_bit.AFE = 0x0; + CT_UART.MODEM_CONTROL_REGISTER_bit.RTS = 0x0; + + /*** END INITIALIZATION ***/ + + while(1) { + /* Print out greeting message */ + PrintMessageOut("Hello you are in the PRU UART demo test please enter some characters\r\n"); + + /* Read in characters from user, then echo them back out */ + for (i = 0; i < BUFFER-1 ; i++) { + rxBuffer[i] = ReadMessageIn(); + if(rxBuffer[i] == '\r') { // Quit early if ENTER is hit. + rxBuffer[i+1] = NULL; + break; + } + } + + PrintMessageOut("you typed:\r\n"); + PrintMessageOut(rxBuffer); + PrintMessageOut("\r\n"); + } + + /*** DONE SENDING DATA ***/ + /* Disable UART before halting */ + CT_UART.POWERMANAGEMENT_AND_EMULATION_REGISTER = 0x0; + + /* Halt PRU core */ + __halt(); +} diff --git a/pru-cookbook/04debug/code/uart_setup.sh b/pru-cookbook/04debug/code/uart_setup.sh new file mode 100644 index 0000000000000000000000000000000000000000..db11d1f4fb5226a1f66482d5262809085dfc0ea9 --- /dev/null +++ b/pru-cookbook/04debug/code/uart_setup.sh @@ -0,0 +1,28 @@ +export TARGET=uart1.pru0 +echo TARGET=$TARGET + +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + # Configure tx + config-pin P9_24 pru_uart + # Configure rx + config-pin P9_26 pru_uart +elif [ $machine = "AI" ]; then + echo " Found" + echo "See AI chapter for configuring via device tree" +elif [ $machine = "Blue" ]; then + echo " Found" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + # Configure tx + config-pin P1_06 pru_uart + config-pin P2_09 pru_uart + # Configure rx + config-pin P1_12 pru_uart + config-pin P2_11 pru_uart +else + echo " Not Found" + pins="" +fi diff --git a/pru-cookbook/04debug/debug.rst b/pru-cookbook/04debug/debug.rst new file mode 100644 index 0000000000000000000000000000000000000000..a9f0edfbd8f0a884beca0d263300cb12b1e89d6e --- /dev/null +++ b/pru-cookbook/04debug/debug.rst @@ -0,0 +1,532 @@ +.. _pru-cookbook-debug: + +Debugging and Benchmarking +########################### + +One of the challenges is getting debug +information out of the PRUs since they don't have a traditional ``printf()``. +In this chapter four different methods are presented that I've found useful in +debugging. The first is simply attaching an LED. The second is using +``dmesg`` to watch the kernel messages. ``prudebug``, a simple debugger that +allows you to inspect registers and memory of the PRUs, is then presented. +Finally, using one of the UARTS to send debugging information out a serial port +is shown. + +Debugging via an LED +===================== + +Problem +--------- + +I need a simple way to see if my program is running without slowing the real-time execution. + +Solution +--------- + +One of the simplest ways to do this is to attach an LED to the output pin and watch it +flash. :ref:`debug_LED` shows an LED attached to pin P9_29 of the BeagleBone Black. + +.. _debug_LED: + +LED used for debugging P9_29 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/LED_bb.png + :align: center + :alt: LED used for debugging P9_29 + +Make sure you have the LED in the correct way, or it won't work. + +Discussion +--------- + +If your output is changing more than a few times a second, the LED will be +blinking too fast and you'll need an oscilloscope or a logic analyzer to +see what's happening. + +Another useful tool that let's you see the contents of the registers and +RAM is discussed in :ref:`debug_prudebug`. + +dmesg Hw +========= + +Problem +--------- +I'm getting an error message +(``/sys/devices/platform/ocp/4a326000.pruss-soc-bus/4a300000.pruss/4a334000.pru0/remoteproc/remoteproc1/state: Invalid argument``) +when I load my code, but don't know what's causing it. + +Solution +--------- +The command ``dmesg`` outputs useful information when dealing with the kernel. +Simplying running ``dmesg -Hw`` can tell you a lot. The ``-H`` flag puts the +dates in the human readable form, the ``-w`` tells it to wait for more information. +Often I'll have a window open running ``dmesg -Hw``. +. + +Here's what ``dmesg`` said for the example above. + +dmesg -Hw +~~~~~~~~~~ + +.. code-block:: bash + + [ +0.000018] remoteproc remoteproc1: header-less resource table + [ +0.011879] remoteproc remoteproc1: Failed to find resource table + [ +0.008770] remoteproc remoteproc1: Boot failed: -22 + +It quickly told me I needed to add the line ``#include "resource_table_empty.h"`` +to my code. + +.. _debug_prudebug: + +prudebug - A Simple Debugger for the PRU +========================================= + +Problem +--------- + +You need to examine registers and memory on the PRUs. + +Solution +--------- + +``prudebug`` is a simple debugger for the PRUs that lets you start and stop the PRUs and +examine the registers and memory. It can be found on GitHub +https://github.com/RRvW/prudebug-rl. I have a version I updated to use byte +addressing rather than word addressing. This makes it easier to work with the +assembler output. You can find it in my GitHub BeagleBoard repo +https://github.com/MarkAYoder/BeagleBoard-exercises/tree/master/pru/prudebug. + +Just download the files and type ``make``. + +Discussion +------------ + +Once ``prudebug`` is installed is rather easy to use. + +.. note:: + + ``prudebug`` has now been ported to the AI. + +.. TODO Isn't working on Pocket at this time. + +.. code-block:: bash + + bone$ *sudo prudebug* + PRU Debugger v0.25 + (C) Copyright 2011, 2013 by Arctica Technologies. All rights reserved. + Written by Steven Anderson + + Using /dev/mem device. + Processor type AM335x + PRUSS memory address 0x4a300000 + PRUSS memory length 0x00080000 + + offsets below are in 32-bit byte addresses (not ARM byte addresses) + PRU Instruction Data Ctrl + 0 0x00034000 0x00000000 0x00022000 + 1 0x00038000 0x00002000 0x00024000 + +You get help by entering ``help``. You cal also enter ``hb`` to get a brief help. + +.. code-block:: bash + + PRU0> *hb* + Command help + + BR [breakpoint_number [address]] - View or set an instruction breakpoint + D memory_location_ba [length] - Raw dump of PRU data memory (32-bit byte offset from beginning of full PRU memory block - all PRUs) + DD memory_location_ba [length] - Dump data memory (32-bit byte offset from beginning of PRU data memory) + DI memory_location_ba [length] - Dump instruction memory (32-bit byte offset from beginning of PRU instruction memory) + DIS memory_location_ba [length] - Disassemble instruction memory (32-bit byte offset from beginning of PRU instruction memory) + G - Start processor execution of instructions (at current IP) + GSS - Start processor execution using automatic single stepping - this allows running a program with breakpoints + HALT - Halt the processor + L memory_location_iwa file_name - Load program file into instruction memory + PRU pru_number - Set the active PRU where pru_number ranges from 0 to 1 + Q - Quit the debugger and return to shell prompt. + R - Display the current PRU registers. + RESET - Reset the current PRU + SS - Single step the current instruction. + WA [watch_num [address [value]]] - Clear or set a watch point + WR memory_location_ba value1 [value2 [value3 ...]] - Write a 32-bit value to a raw (offset from beginning of full PRU memory block) + WRD memory_location_ba value1 [value2 [value3 ...]] - Write a 32-bit value to PRU data memory for current PRU + WRI memory_location_ba value1 [value2 [value3 ...]] - Write a 32-bit value to PRU instruction memory for current PRU + + +Initially you are talking to PRU 0. You can enter ``pru 1`` to talk to PRU 1. +The commands I find most useful are, ``r``, to see the registers. + +.. code-block:: bash + + PRU0> *r* + Register info for PRU0 + Control register: 0x00008003 + Reset PC:0x0000 RUNNING, FREE_RUN, COUNTER_DISABLED, NOT_SLEEPING, PROC_ENABLED + + Program counter: 0x0030 + Current instruction: ADD R0.b0, R0.b0, R0.b0 + + Rxx registers not available since PRU is RUNNING. + +Notice the PRU has to be stopped to see the register contents. + +.. code-block:: bash + + PRU0> *h* + PRU0 Halted. + PRU0> *r* + Register info for PRU0 + Control register: 0x00000001 + Reset PC:0x0000 STOPPED, FREE_RUN, COUNTER_DISABLED, NOT_SLEEPING, PROC_DISABLED + + Program counter: 0x0028 + Current instruction: LBBO R15, R15, 4, 4 + + R00: 0x00000000 R08: 0x00000000 R16: 0x00000001 R24: 0x00000002 + R01: 0x00000000 R09: 0xaf40dcf2 R17: 0x00000000 R25: 0x00000003 + R02: 0x000000dc R10: 0xd8255b1b R18: 0x00000003 R26: 0x00000003 + R03: 0x000f0000 R11: 0xc50cbefd R19: 0x00000100 R27: 0x00000002 + R04: 0x00000000 R12: 0xb037c0d7 R20: 0x00000100 R28: 0x8ca9d976 + R05: 0x00000009 R13: 0xf48bbe23 R21: 0x441fb678 R29: 0x00000002 + R06: 0x00000000 R14: 0x00000134 R22: 0xc8cc0752 R30: 0x00000000 + R07: 0x00000009 R15: 0x00000200 R23: 0xe346fee9 R31: 0x00000000 + +You can resume using ``g`` which starts right where you left off, or use ``reset`` to +restart back at the beginning. + +The ``dd`` command dumps the memory. Keep in mind the following. + +Important memory locations +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-------+---------------------------------------------------------------------------+ + |Address|Contents | + +=======+===========================================================================+ + |0x00000| Start of the **stack** for PRU 0. The file **AM335x_PRU.cmd** specifies | + | | where the stack is. | + +-------+---------------------------------------------------------------------------+ + |0x00100| Start of the **heap** for PRU 0. | + +-------+---------------------------------------------------------------------------+ + |0x00200| Start of DRAM that your programs can use. The **Makefile** specifies | + +-------+---------------------------------------------------------------------------+ + | | the size of the **stack** and the **heap**. | + +-------+---------------------------------------------------------------------------+ + |0x10000| Start of the memory shared between the PRUs. | + +-------+---------------------------------------------------------------------------+ + +Using ``dd`` with no address prints the next section of memory. + +.. code-block:: bash + + PRU0> *dd* + dd + Absolute addr = 0x0000, offset = 0x0000, Len = 16 + [0x0000] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x0010] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x0020] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x0030] 0x00000000 0x00000000 0x00000000 0x00000000 + +The stack grows from higher memory to lower memory, so you often won't see +much around address ``0x0000``. + +.. code-block:: bash + + PRU0> *dd 0x100* + dd 0x100 + Absolute addr = 0x0100, offset = 0x0000, Len = 16 + [0x0100] 0x00000001 0x00000002 0x00000003 0x00000004 + [0x0110] 0x00000004 0x00000003 0x00000002 0x00000001 + [0x0120] 0x00000001 0x00000000 0x00000000 0x00000000 + [0x0130] 0x00000000 0x00000200 0x862e5c18 0xfeb21aca + +Here we see some values on the heap. + +.. code-block:: bash + + PRU0> *dd 0x200* + dd 0x200 + Absolute addr = 0x0200, offset = 0x0000, Len = 16 + [0x0200] 0x00000001 0x00000004 0x00000002 0x00000003 + [0x0210] 0x00000003 0x00000011 0x00000004 0x00000010 + [0x0220] 0x0a4fe833 0xb222ebda 0xe5575236 0xc50cbefd + [0x0230] 0xb037c0d7 0xf48bbe23 0x88c460f0 0x011550d4 + +Data written explicity to ``0x0200`` of the DRAM. + +.. code-block:: bash + + PRU0> *dd 0x10000* + dd 0x10000 + Absolute addr = 0x10000, offset = 0x0000, Len = 16 + [0x10000] 0x8ca9d976 0xebcb119e 0x3aebce31 0x68c44d8b + [0x10010] 0xc370ba7e 0x2fea993b 0x15c67fa5 0xfbf68557 + [0x10020] 0x5ad81b4f 0x4a55071a 0x48576eb7 0x1004786b + [0x10030] 0x2265ebc6 0xa27b32a0 0x340d34dc 0xbfa02d4b + +Here's the shared memory. + +You can also use ``prudebug`` to set breakpoints and single step, +but I haven't used that feature much. + +:ref:`../05blocks/blocks.html#_memory_allocation, Memory Allocation` gives examples +of how you can control where your vaiables are stored in memory. + + +UART +====== + +Problem +--------- + +I'd like to use something like ``printf()`` to debug my code. + +.. TODO Check these on the Black and Pocket without grove + +Solution +--------- + +One simple, yet effective approach to 'printing' from the PRU is +an idea taken from the Adruino playbook; +use the UART (serial port) to output debug information. The PRU has it's +own UART that can send characters to a serial port. + +You'll need a 3.3V FTDI cable to go between your Beagle and the USB port +on your host computer as shown in :ref:`debug_ftdi`. [#debug1]_ +you can get such a cable from places such as +`Sparkfun <https://www.sparkfun.com/products/9717>`_ or +`Adafruit <https://www.adafruit.com/product/70>`_. + +.. _debug_ftdi: + +FTDI cable +~~~~~~~~~~~ + +.. figure:: figures/FTDIcable.jpg + :align: center + :alt: FTDI cable + +Discussion +--------- + +The Beagle side of the FTDI cable has a small triangle on it as shown in +:ref:`debug_ftdi_connector` which marks the ground pin, pin 1. + +.. _debug_ftdi_connector: + +FTDI connector +~~~~~~~~~~~~~~~~~ + +.. figure:: figures/FTDIconnector.jpg + :align: center + :alt: FTDI connector + +The :ref:`debug_FTDI` table shows which pins connect where and :ref:`debug_ftdi_pins` +is a wiring diagram for the BeagleBone Black. + +.. _debug_FTDI: + +Wriing for FTDI cable to Beagle +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +--------+------+---------+-------------+--------+------+---------+ + |FTDI pin|Color |Black pin|AI 1 pin |AI 2 pin|Pocket|Function | + +========+======+=========+=============+========+======+=========+ + |0 |black |P9_1 |P8_1 |P8_1 |P1_16 |ground | + +--------+------+---------+-------------+--------+------+---------+ + |4 |orange|P9_24 |P8_43 |P8_33a |P1_12 |rx | + +--------+------+---------+-------------+--------+------+---------+ + |5 |yellow|P9_26 |P8_44 |P8_31a |P1_06 |tx | + +--------+------+---------+-------------+--------+------+---------+ + +.. _debug_ftdi_pins: + +FTDI to BB Black +~~~~~~~~~~~~~~~~~ + +.. figure:: figures/FTDIhookup_bb.png + :align: center + :alt: FTDI to BB Black + +Details +~~~~~~~~ + +Two examples of using the UART are presented here. The first +(:ref:`debug_uart1`) sends a character out the serial port then +waits for a character to come in. Once the new +character arrives another character is output. + +The second example (:ref:`debug_uart2`) prints out a string and then +waits for characters to arrive. +Once an ENTER appears the string is sent back. + +.. tip:: + + On the Black, either PRU0 and PRU1 can + run this code. Both have access to the same UART. + +You need to set the pin muxes. + +config-pin +~~~~~~~~~~~ + +.. code-block:: bash + + # Configure tx Black + bone$ *config-pin P9_24 pru_uart* + # Configure rx Black + bone$ *config-pin P9_26 pru_uart* + + # Configure tx Pocket + bone$ *config-pin P1_06 pru_uart* + # Configure rx Pocket + bone$ *config-pin P1_12 pru_uart* + +.. note:: + + See :ref:`../08ai/ai.html#ai_device_tree, Configuring pins on the AI via device trees` for configuring + pins on the AI. Make sure your `rx` pins are configured as input pins in the device tree. + +For example + +.. code-block:: bash + + DRA7XX_CORE_IOPAD(0x3610, *PIN_INPUT* | MUX_MODE10) // C6: P8.33a: + +.. * TODO - Add code for Blue. + +uart1.pru1_0.c +~~~~~~~~~~~~~~~ + +Set the following variables so ``make`` will know what to compile. + +make +~~~~~ + +.. code-block:: bash + + bone$ *make TARGET=uart1.pru0* + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=uart1.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/uart1.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /dev/remoteproc/pruss-core0 + +Now ``make`` will compile, load PRU0 and start it. +In a terminal window on your host computer run + +.. code-block:: bash + + host$ *screen /dev/ttyUSB0 115200* + +It will initially display the first charters (``H``) and then as you enter +characters on the keyboard, the rest of the message will appear. + +uart1.pru0.c output +~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/uart1.pru0.png + :align: center + :alt: uart1.pru0.c output + +Here's the code (``uart1.pru1_0.c``) that does it. + +.. _debug_uart1: + +uart1.pru1_0.c +~~~~~~~~~~~~~~~ + +:download:`uart1.pru1_0.c <code/uart1.pru1_0.c>` + +.. note:: + + I'm using the AI version of the code since + it uses variables with more desciptive names. + +The first part of the code initializes the UART. Then the line +``CT_UART.RBR_THR_REGISTERS = tx;`` takes a character in ``tx`` +and sends it to the transmit buffer on the UART. Think of +this as the UART version of the ``printf()``. + +Later the line ``while (!((CT_UART.INTERRUPT_IDENTIFICATION_REGISTER_FIFO_CONTROL_REGISTER & 0x2) == 0x2));`` +waits for the transmitter FIFO to be empty. This makes sure later characters +won't overwrite the buffer before they can be sent. The downside is, this will +cause your code to wait on the buffer and it might miss an important +real-time event. + +The line ``while ((CT_UART.LINE_STATUS_REGISTER & 0x1) == 0x0);`` waits for an input from the +UART (possibly missing something) and ``rx = CT_UART.RBR_THR_REGISTERS;`` reads from the +receive register on the UART. + +These simple lines should be enough to place in your code to print out +debugging information. + +uart2.pru0.c +~~~~~~~~~~~~ + +If you want to try ``uart2.pru0.c``, run the following: + +make +~~~~~ + +.. code-block:: bash + + bone$ *make TARGET=uart2.pru0* + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=uart2.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/uart2.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /dev/remoteproc/pruss-core0 + +You will see: + +uart2.pru0.c output +~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/uart2.pru0.png + :align: center + :alt: uart2.pru0.c output + +Type a few characters and hit ENTER. The PRU will playback what you typed, +but it won't echo it as you type. + +``uart2.pru0.c`` defines ``PrintMessageOut()`` which is passed a string that is +sent to the UART. It takes advantage of the eight character FIFO on the UART. +Be careful using it because it also uses ``while (!CT_UART.LSR_bit.TEMT);`` to +wait for the FIFO to empty, which may cause your code to miss something. + +:ref:`debug_uart2` is the code that does it. + +.. _debug_uart2: + +uart2.pru1_0.c +~~~~~~~~~~~~~~~ + +:download:`uart2.pru1_0.c <code/uart2.pru1_0.c>` + +More complex examples can be built using the principles shown in these examples. + +Copyright +========== + +copyright.c +~~~~~~~~~~~~~ + +:download:`copyright.c <code/copyright.c>` + +.. rubric:: Footnotes + +.. [#debug1] FTDI images are from the BeagleBone Cookbook http://shop.oreilly.com/product/0636920033899.do \ No newline at end of file diff --git a/pru-cookbook/04debug/figures/FTDIcable.jpg b/pru-cookbook/04debug/figures/FTDIcable.jpg new file mode 100644 index 0000000000000000000000000000000000000000..80d8e01563cf6781849b305f7f45de40184caca3 Binary files /dev/null and b/pru-cookbook/04debug/figures/FTDIcable.jpg differ diff --git a/pru-cookbook/04debug/figures/FTDIconnector.jpg b/pru-cookbook/04debug/figures/FTDIconnector.jpg new file mode 100644 index 0000000000000000000000000000000000000000..3517708cffa8b349f8bb6ebe06525a606d338a68 Binary files /dev/null and b/pru-cookbook/04debug/figures/FTDIconnector.jpg differ diff --git a/pru-cookbook/04debug/figures/FTDIhookup.fzz b/pru-cookbook/04debug/figures/FTDIhookup.fzz new file mode 100644 index 0000000000000000000000000000000000000000..561602594f87e6d1a778136f4fe67533bbde4316 Binary files /dev/null and b/pru-cookbook/04debug/figures/FTDIhookup.fzz differ diff --git a/pru-cookbook/04debug/figures/FTDIhookup_bb.png b/pru-cookbook/04debug/figures/FTDIhookup_bb.png new file mode 100644 index 0000000000000000000000000000000000000000..ec7b447a35ff5f1e7d6ecf5539b357833d05563a Binary files /dev/null and b/pru-cookbook/04debug/figures/FTDIhookup_bb.png differ diff --git a/pru-cookbook/04debug/figures/LED.fzz b/pru-cookbook/04debug/figures/LED.fzz new file mode 100644 index 0000000000000000000000000000000000000000..ab3d81aa9a29a2b01a4d980ca8d72a032562aa6f Binary files /dev/null and b/pru-cookbook/04debug/figures/LED.fzz differ diff --git a/pru-cookbook/04debug/figures/LED_bb.png b/pru-cookbook/04debug/figures/LED_bb.png new file mode 100644 index 0000000000000000000000000000000000000000..40533edaec88bfa756d5908aca4dddcdc8a9ca48 Binary files /dev/null and b/pru-cookbook/04debug/figures/LED_bb.png differ diff --git a/pru-cookbook/04debug/figures/uart1.pru0.png b/pru-cookbook/04debug/figures/uart1.pru0.png new file mode 100644 index 0000000000000000000000000000000000000000..beffa2c6189e7c831d0fbf8f57848bcd0995a5c7 Binary files /dev/null and b/pru-cookbook/04debug/figures/uart1.pru0.png differ diff --git a/pru-cookbook/04debug/figures/uart2.pru0.png b/pru-cookbook/04debug/figures/uart2.pru0.png new file mode 100644 index 0000000000000000000000000000000000000000..12c32441ff36ed34c811fce0ce88754103237139 Binary files /dev/null and b/pru-cookbook/04debug/figures/uart2.pru0.png differ diff --git a/pru-cookbook/05blocks/blocks.rst b/pru-cookbook/05blocks/blocks.rst new file mode 100644 index 0000000000000000000000000000000000000000..e6de06e35ef9851530791b1518ba28ef68a765c4 --- /dev/null +++ b/pru-cookbook/05blocks/blocks.rst @@ -0,0 +1,1977 @@ +.. _pru-cookbook-blocks: + +Building Blocks - Applications +############################### + +Here are some examples that use the basic PRU building blocks. + +The following are resources used in this chapter. + +Resources +~~~~~~~~~~ + +* `PRU Optimizing C/C++ Compiler, v2.2, User's Guide <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ +* `AM572x Technical Reference Manual <http://www.ti.com/lit/pdf/spruhz6l>`_ (AI) +* `AM335x Technical Reference Manual <http://www.ti.com/lit/pdf/spruh73>`_ (All others) +* `Exploring BeagleBone by Derek Molloy <http://exploringbeaglebone.com/>`_ +* `WS2812 Data Sheet <https://cdn-shop.adafruit.com/datasheets/WS2812.pdf>`_ + +Memory Allocation +================== + + +Problem +--------- + +I want to control where my variables are stored in memory. + +.. TODO Include a section on accessing DDR. + +Solution +--------- + +Each PRU has is own 8KB of data memory (Data Mem0 and Mem1) and 12KB of +shared memory (Shared RAM) as shown in :ref:`blocks_PRU_block_diagram`. + +.. _blocks_PRU_block_diagram: + +PRU Block Diagram +~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/blockDiagram.png + :align: center + :alt: PRU Block diagram + +Each PRU accesses it's own DRAM starting at location 0x0000_0000. Each PRU +can also access the other PRU's DRAM starting at 0x0000_2000. Both PRUs +access the shared RAM at 0x0001_0000. The compiler can control where each +of these memories variables are stored. + +:ref:`blocks_shared` shows how to allocate seven variable in six different locations. + +.. _blocks_shared: + +shared.pro0.c - Examples of Using Different Memory Locations +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`shared.pru0.c <code/shared.pru0.c>` + + +Discussion +----------- + +Here's the line-by-line + +Line-byline for shared.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + + +-------+---------------------------------------------------------------------------------------------------------+ + |Line | Explanation | + +=======+=========================================================================================================+ + |7 | `PRU_SRAM` is defined here. It will be used later to declare variables | + | | in the `Shared RAM` location of memory. Section 5.5.2 on page 75 of the | + | | `PRU Optimizing C/C++ Compiler, v2.2, User's Guide <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ | + | | gives details of the command. The `PRU_SHAREDMEM` | + | | refers to the memory section defined in `am335x_pru.cmd` on line 26. | + +-------+---------------------------------------------------------------------------------------------------------+ + |8, 9 | These are like the previous line except for the DMEM sections. | + +-------+---------------------------------------------------------------------------------------------------------+ + |16 | Variables declared outside of `main()` are put on the heap. | + +-------+---------------------------------------------------------------------------------------------------------+ + |17 | Adding `PRU_SRAM` has the variable stored in the shared memory. | + +-------+---------------------------------------------------------------------------------------------------------+ + |18, 19 | These are stored in the PRU's local RAM. | + +-------+---------------------------------------------------------------------------------------------------------+ + |20, 21 | These lines are for storing in the `.bss` section as declared on | + | | line 74 of `am335x_pru.cmd`. | + +-------+---------------------------------------------------------------------------------------------------------+ + |28-31 | All the previous examples direct the compiler to an area in memory | + | | and the compilers figures out what to put where. With these lines we | + | | specify the exact location. Here are start with the PRU_DRAM starting | + | | address and add 0x200 to it to avoid | + | | the **stack** and the **heap**. The advantage of this technique is you can | + | | easily share these variables between the ARM and the two PRUs. | + +-------+---------------------------------------------------------------------------------------------------------+ + |36, 37 | Variable declared inside `main()` go on the stack. | + +-------+---------------------------------------------------------------------------------------------------------+ + +.. caution:: + + Using the technique of line 28-31 you can put variables anywhere, even where the + compiler has put them. Be careful, it's easy to overwrite what the compiler has done + +Compile and run the program. + +.. code-block:: bash + + bone$ *source shared_setup.sh* + TARGET=shared.pru0 + Black Found + P9_31 + Current mode for P9_31 is: pruout + Current mode for P9_31 is: pruout + P9_29 + Current mode for P9_29 is: pruout + Current mode for P9_29 is: pruout + P9_30 + Current mode for P9_30 is: pruout + Current mode for P9_30 is: pruout + P9_28 + Current mode for P9_28 is: pruout + Current mode for P9_28 is: pruout + bone$ *make* + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=shared.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/shared.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + +Now check the **symbol table** to see where things are allocated. + +.. code-block:: bash + + bone $ *grep shared /tmp/cloud9-examples/shared.pru0.map* + .... + 1 0000011c shared_0 + 2 00010000 shared_1 + 1 00000000 shared_2 + 1 00002000 shared_3 + 1 00000118 shared_4 + 1 00000120 shared_5 + +We see, ``shared_0`` had no directives and was places in the heap that is 0x100 +to 0x1ff. ``shared_1`` was directed to go to the SHAREDMEM, ``shared_2`` to +the start of the local DRAM (which is also the top of the stack). ``shared_3`` +was placed in the DRAM of PRU 1, ``shared_4`` was placed in the ``.bss`` section, +which is in the **heap**. Finally ``shared_5`` is a pointer to where the value +is stored. + +Where are ``shared_6`` and ``shared_7``? They are declared inside ``main()`` and are +therefore placed on the stack at run time. The ``shared.map`` file shows the +compile time allocations. We have to look in the memory itself to see what +happen at run time. + +Let's fire up ``prudebug`` +(:ref:`../04debug/debug.html#debug_prudebug, prudebug - A Simple Debugger for the PRU`) +to see where things are. + +.. code-block:: bash + + bone$ *sudo ./prudebug* + PRU Debugger v0.25 + (C) Copyright 2011, 2013 by Arctica Technologies. All rights reserved. + Written by Steven Anderson + + Using /dev/mem device. + Processor type AM335x + PRUSS memory address 0x4a300000 + PRUSS memory length 0x00080000 + + offsets below are in 32-bit byte addresses (not ARM byte addresses) + PRU Instruction Data Ctrl + 0 0x00034000 0x00000000 0x00022000 + 1 0x00038000 0x00002000 0x00024000 + + PRU0> *d 0* + Absolute addr = 0x0000, offset = 0x0000, Len = 16 + [0x0000] 0x0000feed 0x00000000 0x00000000 0x00000000 + [0x0010] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x0020] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x0030] 0x00000000 0x00000000 0x00000000 0x00000000 + +The value of ``shared_2`` is in memory location 0. + +.. code-block:: bash + + PRU0> *dd 0x100* + Absolute addr = 0x0100, offset = 0x0000, Len = 16 + [0x0100] 0x00000000 0x00000001 0x00000000 0x00000000 + [0x0110] 0x00000000 0x00000000 0x0000beed 0x0000feef + [0x0120] 0x00000200 0x3ec71de3 0x1a013e1a 0xbf2a01a0 + [0x0130] 0x111110b0 0x3f811111 0x55555555 0xbfc55555 + +There are ``shared_0`` and ``shared_4`` in the heap, but where is ``shared_6`` and +``shared_7``? They are supposed to be on the **stack** that starts at 0. + +.. code-block:: bash + + PRU0> dd *0xc0* + Absolute addr = 0x00c0, offset = 0x0000, Len = 16 + [0x00c0] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x00d0] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x00e0] 0x00000000 0x00000000 0x00000000 0x00000000 + [0x00f0] 0x00000000 0x00000000 0x00004321 0x00009876 + +There they are; the stack grows from the top. (The heap grows from the bottom.) + +.. code-block:: bash + + PRU0> dd *0x2000* + Absolute addr = 0x2000, offset = 0x0000, Len = 16 + [0x2000] 0x0000deed 0x00000001 0x00000000 0x557fcfb5 + [0x2010] 0xce97bd0f 0x6afb2c8f 0xc7f35df4 0x5afb6dcb + [0x2020] 0x8dec3da3 0xe39a6756 0x642cb8b8 0xcb6952c0 + [0x2030] 0x2f22ebda 0x548d97c5 0x9241786f 0x72dfeb86 + +And there is PRU 1's memory with ``shared_3``. And finally the shared memory. + +.. code-block:: bash + + PRU0> *dd 0x10000* + Absolute addr = 0x10000, offset = 0x0000, Len = 16 + [0x10000] 0xdeadbeef 0x0000feed 0x00000000 0x68c44f8b + [0x10010] 0xc372ba7e 0x2ffa993b 0x11c66da5 0xfbf6c5d7 + [0x10020] 0x5ada3fcf 0x4a5d0712 0x48576fb7 0x1004796b + [0x10030] 0x2267ebc6 0xa2793aa1 0x100d34dc 0x9ca06d4a + +The compiler offers great control over where variables are stored. Just +be sure if you are hand picking where things are put, not to put them in places +used by the compiler. + +Auto Initialization of built-in LED Triggers +============================================= + +Problem +----------- + +I see the built-in LEDs blink to their own patterns. +How do I turn this off? Can this be automated? + +Solution +--------- + +Each built-in LED has a default action (trigger) when the Bone boots up. +This is controlled by ``/sys/class/leds``. + +.. code-block:: bash + + bone$ *cd /sys/class/leds* + bone$ *ls* + beaglebone:green:usr0 beaglebone:green:usr2 + beaglebone:green:usr1 beaglebone:green:usr3 + +Here you see a directory for each of the LEDs. Let's pick USR1. + +.. code-block:: bash + + bone$ *cd beaglebone\:green\:usr1* + bone$ *ls* + brightness device max_brightness power subsystem trigger uevent + bone$ *cat trigger* + none usb-gadget usb-host rfkill-any rfkill-none kbd-scrolllock kbd-numlock + kbd-capslock kbd-kanalock kbd-shiftlock kbd-altgrlock kbd-ctrllock kbd-altlock + kbd-shiftllock kbd-shiftrlock kbd-ctrlllock kbd-ctrlrlock *[mmc0]* timer + oneshot disk-activity disk-read disk-write ide-disk mtd nand-disk heartbeat + backlight gpio cpu cpu0 activity default-on panic netdev phy0rx phy0tx + phy0assoc phy0radio rfkill0 + +Notice ``[mmc0]`` is in brackets. This means it's the current trigger; it flashes +when the built-in flash memory is in use. You can turn this off using: + +.. code-block:: bash + + bone$ *echo none > trigger* + bone$ *cat trigger* + *[none]* usb-gadget usb-host rfkill-any rfkill-none kbd-scrolllock kbd-numlock + kbd-capslock kbd-kanalock kbd-shiftlock kbd-altgrlock kbd-ctrllock kbd-altlock + kbd-shiftllock kbd-shiftrlock kbd-ctrlllock kbd-ctrlrlock mmc0 timer + oneshot disk-activity disk-read disk-write ide-disk mtd nand-disk heartbeat + backlight gpio cpu cpu0 activity default-on panic netdev phy0rx phy0tx + phy0assoc phy0radio rfkill0 + +Now it is no longer flashing. + +How can this be automated so when code is run that needs the trigger off, it's +turned off automatically? Here's a trick. Include the following in your code. + +.. code-block:: bash + :linenos: + + #pragma DATA_SECTION(init_pins, ".init_pins") + #pragma RETAIN(init_pins) + const char init_pins[] = + "/sys/class/leds/beaglebone:green:usr3/trigger\0none\0" \ + "\0\0"; + +Lines 3 and 4 declare the array ``init_pins`` to have an entry which is the path +to ``trigger`` and the value that should be 'echoed' into it. Both are NULL +terminated. Line 1 says to put this in a section called ``.init_pins`` and +line 2 says to ``RETAIN`` it. That is don't throw it away if it appears to be +unused. + +Discussion +----------- + +The above code stores this array in the ``.out`` file thats created, but that's not +enough. You need to run :ref:`blocks_write_init_pins` on the ``.out`` file to make +the code work. Fortunately the Makefile always runs it. + +.. _blocks_write_init_pins: + +write_init_pins.sh +~~~~~~~~~~~~~~~~~~~ + +:download:`write_init_pins.sh <code/write_init_pins.sh>` + +The ``readelf`` command extracts the path and value from the ``.out`` file. + +.. code-block:: bash + + bone$ *readelf -x .init_pins /tmp/pru0-gen/shared.out* + + Hex dump of section '.init_pins': + 0x000000c0 2f737973 2f636c61 73732f6c 6564732f /sys/class/leds/ + 0x000000d0 62656167 6c65626f 6e653a67 7265656e beaglebone:green + 0x000000e0 3a757372 332f7472 69676765 72006e6f :usr3/trigger.no + 0x000000f0 6e650000 0000 ne.... + +The rest of the command formats it. Finally line 6 echos the ``none`` into the path. + +This can be generalized to initialize other things. The point is, the ``.out`` +file contains everything needed to run the executable. + + +.. _blocks_pwm: + +PWM Generator +============== + +One of the simplest things a PRU can to is generate a simple +signal starting with a single channel PWM that has a fixed frequency and +duty cycle and ending with a multi channel PWM that the ARM can change +the frequency and duty cycle on the fly. + +Problem +--------- + +I want to generate a PWM signal that has a fixed frequency and duty cycle. + +Solution +--------- + +The solution is fairly easy, but be sure to check the *Discussion* section +for details on making it work. + +:ref:`blocks_pwm1` shows the code. + +.. warning:: + This code is for the BeagleBone Black. + See ``pwm1.pru1_1.c`` for an example + that works on the AI. + +.. _blocks_pwm1: + +pwm1.pru0.c +~~~~~~~~~~~~ + +:download:`pwm1.pru0.c <code/pwm1.pru0.c>` + +To run this code you need to configure the pin muxes to output the PRU. If you are on the Black run + +.. code-block:: bash + + bone$ config-pin P9_31 pruout + +On the Pocket run + +.. code-block:: bash + + bone$ config-pin P1_36 pruout + + +.. note:: + + See :ref:`../08ai/ai.html#ai_device_tree, Configuring pins on the AI via device trees` + for configuring pins on the AI. + +Then, tell ``Makefile`` which PRU you are compiling for and what your target file is + +.. code-block:: bash + + bone$ export TARGET=pwm1.pru0 + +Now you are ready to compile + +.. code-block:: bash + + bone$ make + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=pwm1.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/pwm1.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + +Now attach an LED (or oscilloscope) to ``P9_31`` on the Black +or ``P1.36`` on the Pocket. You should see a squarewave. + +Discussion +----------- + +Since this is our first example we'll discuss the many parts in detail. + +pwm1.pru0.c +~~~~~~~~~~~~ + +:ref:`blocks_pwm1_line_by_line` is a line-by-line expanation of the c code. + +.. _blocks_pwm1_line_by_line: + +Line-by-line of pwm1.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-----+-------------------------------------------------------------------------------------+ + |Line | Explanation | + +=====+=====================================================================================+ + |1 | Standard c-header include | + +-----+-------------------------------------------------------------------------------------+ + |2 | Include for the PRU. The compiler knows where to find this since the | + | | `Makefile` says to look for includes in `/usr/lib/ti/pru-software-support-package` | + +-----+-------------------------------------------------------------------------------------+ + |3 | The file `resource_table_empty.h` is used by the PRU loader. Generally we'll | + | | use the same file, and don't need to modify it. | + +-----+-------------------------------------------------------------------------------------+ + |4 | This include has addresses for the GPIO ports and some bit positions for | + | | some of the headers. | + +-----+-------------------------------------------------------------------------------------+ + +Here's what's in ``resource_table_empty.h`` + +resource_table_empty.c +~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`resource_table_empty.h <code/resource_table_empty.h>` + +Line-by-line (continuted) +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-----+-----------------------------------------------------------------------------------------------------------------+ + |Line | Explanation | + +=====+=================================================================================================================+ + |6-7 | `pass:[__]R30` and `pass:[__]R31` are two variables that refer to the | + | | PRU output (`pass:[__]R30`) and input (`pass:[__]R31`) registers. | + | | When you write something to `pass:[__]R30` it will show up on the | + | | corresponding output pins. When you read from `pass:[__]R31` | + | | you read the data on the input pins. | + | | NOTE: Both names begin with two underscore's. Section 5.7.2 of the | + | | http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf[PRU Optimizing C/C++ Compiler, v2.2, User's Guide] | + | | gives more details. | + +-----+-----------------------------------------------------------------------------------------------------------------+ + |11 | This line selects which GPIO pin to toggle. The table below shows which bits in `pass:[__]R30` | + | | map to which pins | + +-----+-----------------------------------------------------------------------------------------------------------------+ + |14 | `CT_CFG.SYSCFG_bit.STANDBY_INIT` is set to `0` to enable the OCP master port. More details on this | + | | and thousands of other regesters see the | + | | https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf[AM335x Technical Reference Manual]. Section 4 is on the PRU | + | | and section 4.5 gives details for all the registers. | + +-----+-----------------------------------------------------------------------------------------------------------------+ + +Bit 0 is the LSB. + +.. TODO fill in Blue + +.. _blocks_mapping_bits: + +Mapping bit positions to pin names +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +---+---+---------------------+-----------+ + |PRU|Bit|Black pin |Pocket pin | + +===+===+=====================+===========+ + |0 |0 |P9_31 |P1.36 | + +---+---+---------------------+-----------+ + |0 |1 |P9_29 |P1.33 | + +---+---+---------------------+-----------+ + |0 |2 |P9_30 |P2.32 | + +---+---+---------------------+-----------+ + |0 |3 |P9_28 |P2.30 | + +---+---+---------------------+-----------+ + |0 |4 |P9_42b |P1.31 | + +---+---+---------------------+-----------+ + |0 |5 |P9_27 |P2.34 | + +---+---+---------------------+-----------+ + |0 |6 |P9_41b |P2.28 | + +---+---+---------------------+-----------+ + |0 |7 |P9_25 |P1.29 | + +---+---+---------------------+-----------+ + |0 |14 |P8_12(out) P8_16(in) |P2.24 | + +---+---+---------------------+-----------+ + |0 |15 |P8_11(out) P8_15(in) |P2.33 | + +---+---+---------------------+-----------+ + | | | | | + +---+---+---------------------+-----------+ + |1 |0 |P8_45 | | + +---+---+---------------------+-----------+ + |1 |1 |P8_46 | | + +---+---+---------------------+-----------+ + |1 |2 |P8_43 | | + +---+---+---------------------+-----------+ + |1 |3 |P8_44 | | + +---+---+---------------------+-----------+ + |1 |4 |P8_41 | | + +---+---+---------------------+-----------+ + |1 |5 |P8_42 | | + +---+---+---------------------+-----------+ + |1 |6 |P8_39 | | + +---+---+---------------------+-----------+ + |1 |7 |P8_40 | | + +---+---+---------------------+-----------+ + |1 |8 |P8_27 |P2.35 | + +---+---+---------------------+-----------+ + |1 |9 |P8_29 |P2.01 | + +---+---+---------------------+-----------+ + |1 |10 |P8_28 |P1.35 | + +---+---+---------------------+-----------+ + |1 |11 |P8_30 |P1.04 | + +---+---+---------------------+-----------+ + |1 |12 |P8_21 | | + +---+---+---------------------+-----------+ + |1 |13 |P8_20 | | + +---+---+---------------------+-----------+ + |1 |14 | |P1.32 | + +---+---+---------------------+-----------+ + |1 |15 | |P1.30 | + +---+---+---------------------+-----------+ + |1 |16 |P9_26(in)| | | + +---+---+---------------------+-----------+ + +.. note:: + + See :ref:`../08ai/ai.html#ai_device_tree, Configuring pins on the AI via device trees` + for all the PRU pins on the AI. + +Since we are running on PRU 0, and we're using ``0x0001``, +that is bit 0, we'll be toggling ``P9_31``. + +Line-by-line (continued again) +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-----+-----------------------------------------------------------------------+ + |Line | Explanation | + +=====+=======================================================================+ + |17 | Here is where the action is. This line reads ``pass:[__]R30`` and | + | | then ORs it with ``gpio``, setting the bits where there is | + | | a 1 in ``gpio`` and leaving the bits where there is a 0. | + | | Thus we are setting the bit we selected. Finally the new | + | | value is written back to ``pass:[__]R30``. | + +-----+-----------------------------------------------------------------------+ + |18 | ``pass:[__]delay_cycles`` is an ((instrinsic function)) that delays | + | | with number of cycles passed to it. Each cycle is 5ns, | + | | and we are delaying 100,000,000 cycles which is | + | | 500,000,000ns, or 0.5 seconds. | + +-----+-----------------------------------------------------------------------+ + |19 | This is like line 17, but ``~gpio`` inverts all the bits in ``gpio`` | + | | so that where we had a 1, there is now a 0. This 0 | + | | is then ANDed with ``pass:[__]R30`` setting the corresponding | + | | bit to 0. Thus we are clearing the bit we selected. | + +-----+-----------------------------------------------------------------------+ + + +.. tip:: + + You can read more about instrinsics in section 5.11 of the + (`PRU Optimizing C/C++ Compiler, v2.2, User's Guide <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_.) + + +When you run this code and look at the output you will see something like the following figure. + +Output of pwm1.pru0.c with 100,000,000 delays cycles giving a 1s period +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm1.png + :align: center + :alt: pwm1.pru0.c output + +Notice the on time (``+Width(1)``) is 500ms, just as we predicted. +The off time is 498ms, which is only 2ms off from our prediction. +The standard deviation is 0, or only 380as, which is 380 * 10^-18^!. + +You can see how fast the PRU can run by setting both of the +``pass:[__]delay_cycles`` to 0. This results in the next figure. + +Output of pwm1.pru0c with 0 delay cycles +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm2.png + :align: center + :alt: pwm1.pru0.c output with 0 delay + +Notice the period is 15ns which gives us a frequency of about 67MHz. At this high +frequency the breadboard that I'm using distorts the waveform so it's no longer a squarewave. +The **on** time is 5.3ns and the **off** time is 9.8ns. That means **pass:[__]R30 |= gpio** +took only one 5ns cycle and ``pass:[__]R30 &= ~gpio`` also only took one cycle, but there +is also an extra cycle needed for the loop. This means the compiler was able to implement +the ``while`` loop in just three 5ns instructions! Not bad. + +We want a square wave, so we need to add a delay to correct for the delay of looping back. + +Here's the code that does just that. + +pwm2.pru0.c +~~~~~~~~~~~~~ + +:download:`pwm2.pru0.c <code/pwm2.pru0.c>` + +The output now looks like: + +Output of pwm2.pru0.c corrected delay +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm3.png + :align: center + :alt: pwm2.c corrected delay + +It's not hard to adjust the two ``pass:[__]delay_cycles`` +to get the desired frequency and duty cycle. + +### Controlling the PWM Frequency +#### Problem +You would like to control the frequency and +duty cycle of the PWM without recompiling. + +#### Solution +Have the PRU read the **on** and **off** times from a shared memory location. +Each PRU has is own 8KB of data memory (DRAM) and 12KB of shared memory +(SHAREDMEM) that the ARM processor can also access. See :ref:`blocks_PRU_block_diagram`. + +The DRAM 0 address is 0x0000 for PRU 0. The same DRAM appears at address +0x4A300000 as seen from the ARM processor. + +.. tip:: + + See page + 184 of the https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf[AM335x Technical Reference Manual]. + +We take the previous PRU code and add the lines + +.. code-block:: C + + #define PRU0_DRAM 0x00000 // Offset to DRAM + volatile unsigned int *pru0_dram = PRU0_DRAM; + +to define a pointer to the DRAM. + +.. note:: + + The `volatile` keyword is used here to tell the compiler the value this + points to may change, so don't make any assumptions while optimizing. + + +Later in the code we use + +.. code-block:: C + + pru0_dram[ch] = on[ch]; // Copy to DRAM0 so the ARM can change it + pru0_dram[ch+MAXCH] = off[ch]; // Copy after the on array + +to write the `on` and `off` times to the DRAM. Then inside the `while` loop we use + +.. code-block:: c + + onCount[ch] = pru0_dram[2*ch]; // Read from DRAM0 + offCount[ch]= pru0_dram[2*ch+1]; + +to read from the DRAM when reseting the counters. Now, while the PRU is running, +the ARM can write values into the DRAM and change the PWM on and off times. +:ref:`blocks_pwm4` is the whole code. + +.. _blocks_pwm4: + +pwm4.pru0.c +~~~~~~~~~~~~ + +:download:`pwm4.pru0.c <code/pwm4.pru0.c>` + +Here is code that runs on the ARM side to set the on and off time values. + +pwm-test.c +~~~~~~~~~~~~ + +:download:`pwm-test.c <code/pwm-test.c>` + +A quick check on the 'scope shows :ref:`blocks_pwm_arm_control`. + +.. _blocks_pwm_arm_control: + +Four Channel PWM with ARM control +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm4.png + :align: center + :alt: pwm4.png + +From the 'scope you see a 1 cycle **on** time results in a 450ns wide pulse and a +3.06us period is 326KHz, much slower than the 10ns pulse we saw before. But it +may be more than fast enough for many applications. For example, most servos run at 50Hz. + +But we can do better. + +Loop Unrolling for Better Performance +====================================== + +Problem +----------- + +The ARM controlled PRU code runs too slowly. + +Solution +----------- + +Simple loop unrolling can greatly improve the speed. ``pwm5.pru0.c`` is our unrolled +version. + +pwm5.pru0.c Unrolled +~~~~~~~~~~~~~~~~~~~~~ + +:download:`pwm5.pru0.c <code/pwm5.pru0.c>` + +The output of ``pwm5.pru0.c`` is in the figure below. + +pwm5.pru0.c Unrolled version of pwm4.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm5_no_loop.png + :align: center + :alt: pwm5.pru0.c Unrolled version of pwm4.pru0.c + +It's running about 6 times faster than ``pwm4.pru0.c``. + +pwm4.pru0.c vs. pwm5.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +---------+-----------------+-----------------+---------+-----------------------+---------+ + |Measure |pwm4.pru0.c time |pwm5.pru0.c time |Speedup |pwm5.pru0.c w/o UNROLL |Speedup | + +---------+-----------------+-----------------+---------+-----------------------+---------+ + |Period |3.06μs |510ns |6x |1.81μs |~1.7x | + +---------+-----------------+-----------------+---------+-----------------------+---------+ + |Width+ |450ns |70ns |~6x |1.56μs |~.3x | + +---------+-----------------+-----------------+---------+-----------------------+---------+ + +Not a bad speed up for just a couple of simple changes. + +Discussion +----------- + +Here's how it works. +First look at line 39. You see ``#pragma UNROLL(MAXCH)`` which is a ``pragma`` +that tells the compiler to unroll the loop that follows. We are unrolling it +``MAXCH`` times (four times in this example). Just removing the ``pragma`` causes +the speedup compared to the ``pwm4.pru0.c`` case to drop from 6x to only 1.7x. + +We also have our ``for`` loop inside the ``while`` loop that can be unrolled. +Unfortunately ``UNROLL()`` doesn't work on it, therefore we have to do it by +hand. We could take the loop and just copy it three times, but that would +make it harder to maintain the code. Instead I convered the loop into a +``#define`` (lines 14-24) and invoked ``update()`` as needed (lines 48-51). +This is not a function call. Whenever the preprocessor sees the ``update()`` +it copies the code an then it's compiled. + +This unrolling gets us an impressive 6x speedup. + +Making All the Pulses Start at the Same Time +============================================= + +Problem +----------- + +I have a mutlichannel PWM working, but the pulses aren't synchronized, that is +they don't all start at the same time. + +Solution +----------- + +:ref:`blocks_zoomed` is a zoomed in version of the previous figure. Notice the pulse +in each channel starts about 15ns later than the channel above it. + +.. _blocks_zoomed: + +pwm5.pru0 Zoomed In +~~~~~~~~~~~~~~~~~~~~~ +.. figure:: figures/pwm5_zoomed.png + :align: center + :alt: pwm5.pru0 zoomed.png + +The solution is to declare ``Rtmp`` (line 35) which holds the value for ``pass:[__]R30``. + +pwm6.pru0.c Sync'ed Version of pwm5.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`pwm6.pru0.c Sync'ed Version of pwm5.pru0.c <code/pwm6.pru0.c>` + +Each channel writes it's value to ``Rtmp`` (lines 17 and 20) and then after +each channel has updated, ``Rtmp`` is copied to ``pass:[__]R30`` (line 54). + +Discussion +----------- + +The following figure shows the channel are sync'ed. Though the period is slightly +longer than before. + +pwm6.pru0 Synchronized Channels +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm6_synced.png + :align: center + :alt: pwm6.pru0 Synchronized Channels + + +Adding More Channels via PRU 1 +================================ + +Problem +----------- + +You need more output channels, or you need to shorten the period. + +Solution +----------- + +PRU 0 can output up to eight output pins (see +:ref:`blocks_mapping_bits`). The code presented so far +can be easily extended to use the eight output pins. + +But what if you need more channels? +You can always use PRU1, it has 14 output pins. + +Or, what if four channels is enough, but you need a shorter period. Everytime +you add a channel, the overall period gets longer. Twice as many channels +means twice as long a period. If you move half the channels to PRU 1, you +will make the period half as long. + +Here's the code (``pwm7.pru0.c``) + +pwm7.pru0.c Using Both PRUs +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`pwm7.pru0.c Using Both PRUs <code/pwm7.pru0.c>` + +Be sure to run ``pwm7_setup.sh`` to get the correct pins configured. + +pwm7_setup.sh +~~~~~~~~~~~~~~~ + +:download:`pw7_setup.sh <code/pwm7_setup.sh>` + +This makes sure the PRU 1 pins are properly configured. + +Here we have a second ``pwm7`` file. ``pwm7.pru1.c`` is identical to ``pwm7.pru0.c`` +except ``PRUNUM`` is set to 1, instead of 0. + +Compile and run the two files with: + +.. code-block:: bash + + bone$ *make TARGET=pwm7.pru0; make TARGET=pwm7.pru1* + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=pwm7.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/pwm7.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=pwm7.pru1 + - Stopping PRU 1 + - copying firmware file /tmp/cloud9-examples/pwm7.pru1.out to /lib/firmware/am335x-pru1-fw + write_init_pins.sh + - Starting PRU 1 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 1 + PRU_DIR = /sys/class/remoteproc/remoteproc2 + +This will first stop, compile and start PRU 0, then do the same for PRU 1. + +Moving half of the channels to PRU1 dropped the period from 510ns to 370ns, so +we gained a bit. + +Discussion +----------- + +There weren't many changes to be made. Line 15 we set MAXCH to 2. Lines 44-48 +is where the big change is. +.. code-block:: c + + pru0_dram[2*ch ] = on [ch+PRUNUN*MAXCH]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch+PRUNUN*MAXCH]; // Interleave the on and off values + onCount[ch] = on [ch+PRUNUN*MAXCH]; + offCount[ch]= off[ch+PRUNUN*MAXCH]; + +If we are compiling for PRU 0, ``on[ch+PRUNUN*MAXCH]`` becomes ``on[ch+0*2]`` which is +``on[ch]`` which is what we had before. But now if we are on PRU 1 it becomes +``on[ch+1*2]`` which is ``on[ch+2]``. That means we are picking up the second +half of the ``on`` and ``off`` arrays. The first half goes to PRU 0, the second to +PRU 1. So the same code can be used for both PRUs, but we get slightly different +behavior. + +Running the code you will see the next figure. + +pwm7.pru0 Two PRUs running +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm7_two_prus_running.png + :align: center + :alt: pwm7.pru0 Two PRUs running + +What's going on there, the first channels look fine, but the PRU 1 channels +are blurred. To see what's happening, let's stop the oscilloscope. + +pwm7.pru0 Two PRUs stopped +~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm7_two_prus_stopped.png + :align: center + :alt: pwm7 Two PRUs stopped + +The stopped display shows that the four channels are doing what we wanted, except +The PRU 0 channels have a period of 370ns while the PRU 1 channels at 330ns. +It appears the compiler has optimied the two PRUs slightly differenty. + + +Synchronizing Two PRUs +======================= + +Problem +----------- + +I need to synchronize the two PRUs so they run together. + +Solution +----------- + +Use the Interrupt Controller (INTC). It allows one PRU to signal the other. +Page 225 of the `AM335x Technical Reference Manual <https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf>`_ +has details of how it works. Here's the code for PRU 0, which at the end of the +``while`` loop signals PRU 1 to start(``pwm8.pru0.c``). + +pwm8.pru0.c PRU 0 using INTC to send a signal to PRU 1 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`pwm8.pru0.c PRU 0 using INTC to send a signal to PRU 1 <code/pwm8.pru0.c>` + +PRU 2's code waits for PRU 0 before going. + +pwm8.pru1.c PRU 1 waiting for INTC from PRU 0 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`pwm8.pru1.c PRU 1 waiting for INTC from PRU 0 <code/pwm8.pru1.c>` + +In ``pwm8.pru0.c`` PRU 1 waits for a signal from PRU 0, so be sure to start PRU 1 first. + +.. code-block:: bash + + bone$ *make TARGET=pwm8.pru0; make TARGET=pwm8.pru1* + +Discussion +----------- + +The figure below shows the two PRUs are synchronized, though there is some extra +overhead in the process so the period is longer. + +pwm8.pru0 PRUs sycned +~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pwm8_prus_sycned.png + :align: center + :alt: pwm8.pru0 PRUs sycned + +This isn't much different from the previous examples. + +pwm8.pru0.c changes from pwm7.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-----+-------+---------------------------------------------------------------------------------------+ + |PRU |Line |Change | + +=====+=======+=======================================================================================+ + |0 |37-45 |For PRU 0 these define ``configInitc()`` which initializes the interupts. | + | | |See page 226 of the | + | | |`AM335x Technical Reference Manual <https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf>`_ | + | | |for a diagram explaining events, channels, hosts, etc. | + +-----+-------+---------------------------------------------------------------------------------------+ + |0 |55-56 |Set a configuration register and call `configInitc`. | + +-----+-------+---------------------------------------------------------------------------------------+ + |1 |59-61 |PRU 1 then waits for PRU 0 to signal it. Bit 31 of ``pass:[__]R31`` corresponds | + | | |to the Host-1 channel which ``configInitc()`` set up. We also clear event 16 so | + | | |PRU 0 can set it again. | + +-----+-------+---------------------------------------------------------------------------------------+ + |0 |74-75 |On PRU 0 this generates the interupt to send to PRU 1. I found PRU 1 was | + | | |slow to respond to the interupt, so I put this code at the end of the loop to | + | | |give time for the signal to get to PRU 1. | + +-----+-------+---------------------------------------------------------------------------------------+ + +This ends the multipart pwm example. + +Reading an Input at Regular Intervals +====================================== + +Problem +----------- + +You have an input pin that needs to be read at regular intervals. + +Solution +----------- + +You can use the ``pass:[__]R31`` register to read an input pin. Let's use the following +pins. + +.. _blocks_io_pins: + +Input/Output pins +~~~~~~~~~~~~~~~~~~ + +.. table:: + + +---------+----------+-----+----------+---------+ + |Direction|Bit number|Black|AI (ICSS2)|Pocket | + +=========+==========+=====+==========+=========+ + |out |0 |P9_31|P8_44 |P1.36 | + +---------+----------+-----+----------+---------+ + |in |7 |P9_25|P8_36 |P1.29 | + +---------+----------+-----+----------+---------+ + +These values came from :ref:`blocks_mapping_bits`. + +Configure the pins with ``input_setup.sh``. + +input_setup.sh +~~~~~~~~~~~~~~~ + +:download:`input_setup.sh <code/input_setup.sh>` + +The following code reads the input pin and writes its value to the output pin. + +input.c +~~~~~~~~~~ + +:download:`input.pru0.c <code/input.pru0.c>` + +Discussion +----------- + +Just remember that ``pass:[__]R30`` is for outputs and ``pass:[__]R31`` is for inputs. + +Analog Wave Generator +======================= + +Problem +----------- + +I want to generate an analog output, but only have GPIO pins. + +Solution +----------- + +The Beagle doesn't have a built-in analog to digital converter. You could get a +`USB Audio Dongle <https://www.amazon.com/external-Adapter-Windows-Microphone-SD-CM-UAUD/dp/B001MSS6CS/0&keywords=audio+dongle>`_ +which are under $10. But here we'll take another approach. + +Earlier we generated a PWM signal. Here we'll generate a PWM whose duty cycle +changes with time. A small duty cycle for when the output signal is small and +a large duty cycle for when it is large. + +This example was inspired by +`A PRU Sin Wave Generator <https://github.com/derekmolloy/exploringBB/tree/master/chp13/sineWave>`_ +in chapter 13 of +`Exploring BeagleBone by Derek Molloy<http://exploringbeaglebone.com/>`_. + +Here's the code. + +sine.pru0.c +~~~~~~~~~~~~~ + +:download:`sine.pru0.c <code/sine.pru0.c>` + + +Set the ``#define`` at line 7 to the number of samples in one cycle of the waveform +and set the ``#define`` at line 8 to which waveform and then run ``make``. + +Discussion +----------- + +The code has two parts. The first part (lines 21 to 39) generate the waveform +to be output. The ``#define``s let you select which waveform you want to +generate. Since the output is a percent duty cycle, the values in ``waveform[]`` +must be between 0 and 100 inclusive. The waveform is only generated once, so +this part of the code isn't time critical. + +The second part (lines 44 to 54) uses the generated data to set the +duty cycle of the PWM on a cycle-by-cycle basis. This part is time critical; +the faster we can output the values, the higher the frequency of the output +signal. + +Suppose you want to generate a sawtooth waveform like the one shown in :ref:`blocks_sawtooth`. + +.. _blocks_sawtooth: + +Continuous Sawtooth Waveform +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sawtoothsmooth.png + :align: center + :alt: Continuous Sawtooth Waveform + +You need to sample the waveform and store one cycle. :ref:`blocks_sawtoothsampled` +shows a sampled version of the sawtooth. You need to generate ``MAXT`` samples; +here we show 20 samples, which may be enough. In the code ``MAXT`` is set to 100. + +.. _blocks_sawtoothsampled: + +Sampled Sawtooth Waveform +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sawtoothsampled.png + :align: center + :alt: Sampled Sawtooth Waveform + +There's a lot going on here; let's take it line by line. + +Line-by-line of sine.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-------+---------------------------------------------------------------------------------+ + |Line | Explanation | + +=======+=================================================================================+ + |2-5 | Standard c-header includes | + +-------+---------------------------------------------------------------------------------+ + |7 | Number for samples in one cycle of the analog waveform | + +-------+---------------------------------------------------------------------------------+ + |8 | Which waveform to use. We've defined SAWTOOTH, TRIANGLE and SINE, but | + | | you can define your own too. | + +-------+---------------------------------------------------------------------------------+ + |10-11 | Declaring registers ``pass:[__R30]`` and ``pass:[__R31]``. | + +-------+---------------------------------------------------------------------------------+ + |15-16 | ``onCount`` counts how many cycles the PWM should be 1 and ``offCount`` counts | + | | how many it should be off. | + +-------+---------------------------------------------------------------------------------+ + |18 | ``waveform[]`` stores the analog waveform being ouput. | + +-------+---------------------------------------------------------------------------------+ + |21-24 | ``SAWTOOTH`` is the simplest of the waveforms. Each sample is the duty cycle | + | | at that time and must therefore be between 0 and 100. | + +-------+---------------------------------------------------------------------------------+ + |26-31 | ``TRIANGLE`` is also a simple waveform. | + +-------+---------------------------------------------------------------------------------+ + |32-39 | ``SINE`` generates a sine wave and also introduces floating point. Yes, | + | | you can use floating point, but the PRUs don't have floating point hardware, | + | | rather, it's all done in software. This mean using floating point will make | + | | your code much bigger and slower. Slower doesn't matter in this part, and | + | | bigger isn't bigger than our instruction memory, so we're OK. | + +-------+---------------------------------------------------------------------------------+ + |47 | Here the ``for`` loop looks up each value of the generated waveform. | + +-------+---------------------------------------------------------------------------------+ + |48,49 | ``onCount`` is the number of cycles to be at 1 and ``offCount`` is the number | + | | of cycles to be 0. The two add to 100, one full cycle. | + +-------+---------------------------------------------------------------------------------+ + |50-52 | Stay on for ``onCount`` cycles. | + +-------+---------------------------------------------------------------------------------+ + |53-55 | Now turn off for ``offCount`` cycles, then loop back and look up the next | + | | cycle count. | + +-------+---------------------------------------------------------------------------------+ + +:ref:`blocks_sawunfiltered` shows the output of the code. + +.. _blocks_sawunfiltered: + +Unfiltered Sawtooth Waveform +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sawunfiltered.png + :align: center + :alt: Unfiltered Sawtooth Waveform + +It doesn't look like a sawtooth; but if you look at the left side you will +see each cycle has a longer and longer on time. The duty cycle is increasing. +Once it's almost 100% duty cycle, it switches to a very small duty cycle. +Therefore it's output what we programmed, but what we want is the average of +the signal. The left hand side has a large (and increasing) average which +would be for top of the sawtooth. The right hand side has a small average, +which is what you want for the start of the sawtooth. + +A simple low-pass filter, built with one resistor and one capacitor will do it. +:ref:`blocks_filterwiring` shows how to wire it up. + +.. _blocks_filterwiring: + +Low-Pass Filter Wiring Diagram +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/filter_bb.png + :align: center + :alt: Low-Pass Filter Wiring Diagram + +.. note:: + + I used a 10K variable resistor and a 0.022uF capacitor. + Probe the circuit between the resistor and the capacitor + and adjust the resistor until you get a good looking waveform. + +:ref:`blocks_sawscope` shows the results for filtered the SAWTOOTH. + +.. _blocks_sawscope: + +Reconstructed Sawtooth Waveform +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sawscope.png + :align: center + :alt: Reconstructed Sawtooth Waveform + +Now that looks more like a sawtooth wave. The top plot is the time-domain +plot of the output of the low-pass filter. The bottom plot is the FFT of the top +plot, therefore it's the frequency domain. We are getting a sawtooth with a +frequency of about 6.1KHz. You can see the fundamental frequency on the +bottom plot along with several harmonics. + + +The top looks like a sawtooth wave, +but there is a high freqnecy superimposed on it. We are only using a simple +first-order filter. You could lower the cutoff freqnecy by adjusting the +resistor. You'll see something like :ref:`blocks_lowercutoff`. + +.. _blocks_lowercutoff: + +Reconstructed Sawtooth Waveform with Lower Cutoff Frequency +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sawlowercutoff.png + :align: center + :alt: Reconstructed Sawtooth Waveform with Lower Cutoff Frequency + +The high freqencies have been reduced, but the corner of the waveform has +been rounded. You can also adjust the cutoff to a higher frequency and you'll +get a sharper corner, but you'll also get more high frequencies. See +:ref:`blocks_highercutoff` + +.. _blocks_highercutoff: + +Reconstructed Sawtooth Waveform with Higher Cutoff Frequency +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sawhighercutoff.png + :align: center + :alt: Reconstructed Sawtooth Waveform with Higher Cutoff Frequency + +Adjust to taste, though the real solution is to build a higher order filter. +Search for _second order **filter** and you'll find some nice circuits. + +You can adjust the frequency of the signal by adjusting ``MAXT``. A smaller +``MAXT`` will give a higher frequency. I've gotten good results with ``MAXT`` +as small as 20. + +You can also get a triangle waveform by setting the ``#define``. +:ref:`blocks_triangle` shows the output signal. + +.. _blocks_triangle: + +Reconstructed Triangle Waveform +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/triangle.png + :align: center + :alt: Reconstructed Triangle Waveform + +And also the sine wave as shown in :ref:`blocks_sine`. + +.. _blocks_sine: + +Reconstructed Sinusoid Waveform +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/sine.png + :align: center + :alt: Reconstructed Sinusoid Waveform + +Notice on the bottom plot the harmonics are much more suppressed. + +Generating the sine waveform uses **floats**. This requires much more code. +You can look in `/tmp/cloud9-examples/sine.pru0.map` to see how much memory is being used. +:ref:`blocks_sine_map` shows the first few lines for the sine wave. + +.. _blocks_sine_map: + +/tmp/cloud9-examples/sine.pru0.map for Sine Wave +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`lines=1..22 <code/sine.map>` + +Notice line 15 shows 0x18c0 bytes are being used for instructions. That's 6336 +in decimal. + +Now compile for the sawtooth and you see only 444 byes are used. Floating-point +requires over 5K more bytes. Use with care. If you are short on instruction +space, you can move the table generation to the ARM and just copy the table +to the PRU. + + +.. ToDo Ultrasonic Sensor Application + +.. _blocks_ws2812: + +WS2812 (NeoPixel) driver +========================== + +Problem +----------- + +You have an `Adafruit NeoPixel LED string <http://www.adafruit.com/products/1138>`_ +or `Adafruit NeoPixel LED matrix <http://www.adafruit.com/products/1487>`_ +and want to light it up. + +Solution +----------- + +NeoPixel is Adafruit's name for the WS2812 Intelligent control LED. Each +NeoPixel contains a Red, Green and Blue LED with a PWM controller that can +dim each one individually making a rainbow of colors possible. The +NeoPixel is driven by a single serial line. The timing on the line is very +sensesitive, which make the PRU a perfect candidate for driving it. + +Wire the input to ``P9_29`` and power to 3.3V and ground to ground as shown in +:ref:`blocks_neo_wiring`. + +.. _blocks_neo_wiring: + +.NeoPixel Wiring +.. figure:: figures/neo_bb.png + :align: center + :alt: NeoPixel Wiring + +Test your wiring with the simple code in :ref:`blocks_neo1` +which to turns all pixels white. + +.. _blocks_neo1: + +neo1.pru0.c - Code to turn all NeoPixels's white +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: bash + +:download:`neo1.pru0.c <code/neo1.pru0.c>` + +Discussion +----------- + +:ref:`blocks_sequence` (taken from `WS2812 Data Sheet <https://cdn-shop.adafruit.com/datasheets/WS2812.pdf>`_) shows the following waveforms are used to send a bit of data. + +.. _blocks_sequence: + +NeoPixel bit sequence +~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/neo_sequence.png + :align: center + :alt: NeoPixel bit sequence + +Where the times are: + +.. table:: + + +-------+-------------+ + |Label | Time in ns | + +=======+=============+ + |T0H | 350 | + +-------+-------------+ + |T0L | 800 | + +-------+-------------+ + |T1H | 700 | + +-------+-------------+ + |T1L | 600 | + +-------+-------------+ + |Treset | >50,000 | + +-------+-------------+ + +The code in :ref:`blocks_neo1` define these times in lines 7-10. +The `/5` is because +each instruction take 5ns. Lines 27-30 then set the output to 1 for +the desired time and then to 0 and keeps repeating it for the entire +string length. :ref:`blocks_zero_scope` +shows the waveform for sending a 0 value. Note the times are spot on. + +.. _blocks_zero_scope: + +NeoPixel zero timing +~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/neo_scope.png + :align: center + :alt: + +Each NeoPixel listens for a RGB value. Once a value has arrived all other values +that follow are passed on to the next NeoPixel which does the same thing. +That way you can individually control all of the NeoPixels. + +Lines 38-40 send out a reset pulse. If a NeoPixel sees a reset pulse it will +grab the next value for itself and start over again. + + + +Setting NeoPixels to Different Colors +====================================== + +Problem +----------- + +I want to set the LEDs to different colors. + +Solution +----------- + +Wire your NeoPixels as shown in :ref:`blocks_neo_wiring` then run the code in +:ref:`blocks_neo2`. + +.. _blocks_neo2: + +neo2.pru0.c - Code to turn on green, red, blue +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`neo2.pru0.c <code/neo2.pru0.c>` + +This will make the first LED green, the second red and the third blue. + +Discussion +----------- + +:ref:`blocks_new_data_seq` shows the sequence of bits +used to control the green, red and blue values. + +.. _blocks_new_data_seq: + +NeoPixel data sequence +~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/neo_data_seq.png + :align: center + :alt: + +.. note:: + +The usual order for colors is RGB (red, green, blue), but the NeoPixels use GRB (green, red, blue). + +:ref:`blocks_neo2_line` is the line-by-line for ``neo2.pru0.c``. + +.. _blocks_neo2_line: + +Line-by-line for neo2.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-------+---------------------------------------------------------------------------------+ + |Line | Explanation | + |23 | Define the string of colors to be output. Here the ordering of the | + | | bits is the same as :ref:`blocks_new_data_seq`, GRB. | + +-------+---------------------------------------------------------------------------------+ + |26 | Loop for each color to output. | + +-------+---------------------------------------------------------------------------------+ + |27 | Loop for each bit in an GRB color. | + +-------+---------------------------------------------------------------------------------+ + |28 | Get the j^th^ color and mask off all but the i^th^ bit. `(0x1:ref:`i)` takes | + | | the value `0x1` and shifts it left `i` bits. When anded (&) with `color[j]` | + | | it will zero out all but the i^th^ bit. If the result of the operation is | + | | 1, the `if` is done, otherwise the `else` is done. | + +-------+---------------------------------------------------------------------------------+ + |29-32 | Send a 1. | + +-------+---------------------------------------------------------------------------------+ + |34-37 | Send a 0. | + +-------+---------------------------------------------------------------------------------+ + |42-43 | Send a reset pulse once all the colors have been sent. | + +-------+---------------------------------------------------------------------------------+ + +.. note:: + +This will only change the first ``STR_LEN`` LEDs. The LEDs that follow will not +be changed. +==== + +Controlling Arbitrary LEDs +============================ + +Problem +----------- + +I want to change the 10^th^ LED and not have to change the others. + +Solution +----------- + +You need to keep an array of colors for the whole string in the PRU. Change +the color of any pixels you want in the array and then send out the whole +string to the LEDs. :ref:`blocks_neo3` shows an example animates a red pixel +running around a ring of blue background. :ref:`blocks_neo3_video` shows +the code in action. + +.. _blocks_neo3_video: + +neo3.pru0.c - Simple animation + +:download:`ring_around.mp4 <figures/ring_around.mp4>` + +.. _blocks_neo3: + +neo3.pru0.c - Code to animate a red pixel running around a ring of blue +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`neo3.pru0.c <code/neo3.pru0.c>` + + +Discussion +----------- + +Here's the highlights. + +.. table:: + + +-----+---------------------------------+ + |Line |Explanation | + +=====|=================================+ + |32,33|Initiallize the array of colors. | + +-----+---------------------------------+ + |38-41|Update the array. | + +-----+---------------------------------+ + |44-58|Send the array to the LEDs. | + +-----+---------------------------------+ + |60-61|Send a reset. | + +-----+---------------------------------+ + |64 |Wait a bit. | + +-----+---------------------------------+ + +Controlling NeoPixels Through a Kernel Driver +====================================== + +Problem +----------- + +You want to control your NeoPixels through a kernel driver so you can control it +through a ``/dev`` interface. + +Solution +----------- + +The `rpmsg_pru <https://github.com/beagleboard/linux/raw/4.9/drivers/rpmsg/rpmsg_pru.c>`_ +driver provides a way to pass data between the ARM processor and +the PRUs. It's already included on current images. :ref:`blocks_neo4` shows +an example. + +.. _blocks_neo4: + +neo4.pru0.c - Code to talk to the PRU via rpmsg_pru +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`neo4.pru0.c <code/neo4.pru0.c>` + +Run the code as usual. + +.. code-block:: bash + + bone$ make TARGET=neo4.pru0 + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=neo4.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/neo4.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + + bone$ echo 0 0xff 0 127 > /dev/rpmsg_pru30 + bone$ echo -1 > /dev/rpmsg_pru30 + + +.. TODO get this working on the 5.10 kernel + +``/dev/rpmsg_pru30`` is a device driver that lets the ARM talk to the PRU. +The first ``echo`` says to set the 0^th^ LED to RGB value 0xff 0 127. (Note: you can +mix hex and decimal.) The second ``echo`` tells the driver to send the data to the +LEDs. Your 0^th^ LED should now be lit. + +Discussion +----------- + +There's a lot here. I'll just hit some of the highlights in :ref:`blocks_neo4_lines`. + +.. _blocks_neo4_lines: + +Line-by-line for neo4.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + +|Line |Explanation + +|30 |The `CHAN_NAME` of `rpmsg-pru` matches that `prmsg_pru` driver that is +is already installed. This connects this PRU to the driver. +|32 |The `CHAN_PORT` tells it to use port 30. That's why we use +`/dev/rpmsg_pru30` +|40 |`payload[]` is the buffer that receives the data from the ARM. +|42-48|Same as the previous NeoPixel examples. +|52 |`color[]` is the state to be sent to the LEDs. +|66-68|`color[]` is initialized. +|70-85|Here are a number of details needed to set up the channel between +the PRU and the ARM. +|88 |Here we wait until the ARM sends us some numbers. +|99 |Receive all the data from the ARM, store it in `payload[]`. +|101-111|The data sent is: index red green blue. Pull off the index. If it's +in the right range, pull off the red, green and blue values. +|113 |The NeoPixels want the data in GRB order. Shift and OR everything +together. +|116-133|If the `index` = -1, send the contents of `color` to the LEDs. This +code is same as before. +|==== + +You can now use programs running on the ARM to send colors to the PRU. + +:ref:`blocks_neo-rainbow` shows an example. + +.. _blocks_neo-rainbow: + +neo-rainbow.py - A python program using /dev/rpmsg_pru30 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`neo-rainbow.py <code/neo-rainbow.py>` + + +Line 19 writes the data to the PRU. Be sure to have a newline, or space after +the last number, or you numbers will get blurred together. + +Switching from pru0 to pru1 with rpmsg_pru +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + + +There are three things you need to change when switching from pru0 to pru1 +when using rpmsg_pru. + +1. The include on line 10 is switched to ``#include "resource_table_1.h"`` (0 is switched to a 1) +2. Line 17 is switched to ``#define HOST_INT ((uint32_t) 1 << 31)`` (30 is switched to 31.) +3. Lines 23 and 24 are switched to: + +.. code-block:: c + + #define TO_ARM_HOST 18 + #define FROM_ARM_HOST 19 + +These changes switch to the proper channel numbers to use pru1 instead of pru0. + +RGB LED Matrix - No Integrated Drivers +====================================== + + +Problem +----------- + +You have a RGB LED matrix +(:ref:`../01case/case.html#case_rgb_matrix, 1.4. RGB LED Matrix - No Integrated Drivers`) and want to know +at a low level how the PRU works. + +Solution +----------- + +Here is the +`datasheet <https://cdn-shop.adafruit.com/product-files/2277/MI-T35P5RGBE-AE.pdf>`_, +but the best description I've found for the RGB Matrix is from +`Adafruit <https://www.adafruit.com/>`_. I've reproduced it here, with +adjustments for the 64x32 matrix we are using. + + +.. admonition:: information + + There's zero documention out there on how these matrices work, and no public + datasheets or spec sheets so we are going to try to document how they work. + + First thing to notice is that there are 2048 RGB LEDs in a 64x32 matrix. + Like pretty much every matrix out there, you can't drive all 2048 at once. + One reason is that would require a lot of current, another reason is that + it would be really expensive to have so many pins. Instead, the matrix is + divided into 16 interleaved sections/strips. The first section is the + 1^st^ 'line' and the 17^th^ 'line' (64 x 2 RGB LEDs = 128 RGB LEDs), + the second is the 2^nd^ and 18^th^ line, etc until the last section + which is the 16^th^ and 32^nd^ line. You might be asking, why are the + lines paired this way? wouldn't it be nicer to have the first section + be the 1^st^ and 2^nd^ line, then 3^rd^ and 4^th^, until the 15^th^ and + 16^th^? The reason they do it this way is so that the lines are interleaved + and look better when refreshed, otherwise we'd see the stripes more clearly. + + So, on the PCB is 24 LED driver chips. These are like 74HC595s but they + have 16 outputs and they are constant current. 16 outputs * 24 chips = + 384 LEDs that can be controlled at once, and 128 * 3 (R G and B) = 384. + So now the design comes together: You have 384 outputs that can control + one line at a time, with each of 384 R, G and B LEDs either on or off. + The controller (say an FPGA or microcontroller) selects which section + to currently draw (using LA, LB, LC and LD address pins - 4 bits can + have 16 values). Once the address is set, the controller clocks out + 384 bits of data (48 bytes) and latches it. Then it increments the + address and clocks out another 384 bits, etc until it gets to address + #15, then it sets the address back to #0 + + https://cdn-learn.adafruit.com/downloads/pdf/32x16-32x32-rgb-led-matrix.pdf + +That gives a good overview, but there are a few details missing. +:ref:`blocks_rgb_python` is a functioning python program that gives a nice +high-level view of how to drive the display. + +.. TODO Test this + +.. _blocks_rgb_python: + +rgb_python.py - Python code for driving RGB LED matrix +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`rgb_python.py <code/rgb_python.py>` + +Be sure to run the :ref:`blocks_rgb_setup` script before running the python code. + +.. _blocks_rgb_setup: + +rgb_python_setup.sh +~~~~~~~~~~~~~~~~~~~~ + +:download:`rgb_python_setup.sh <code/rgb_python_setup.sh>` + +Make sure line 29 is commented out and line 30 is uncommented. +Later we'll configure for _pruout_, but for now the python code doesn't use +the PRU outs. + +.. code-block:: bash + + # config-pin $pin pruout + config-pin $pin out + +Your display should look like :ref:`blocks_rgb_python_jpg`. + +.. _blocks_rgb_python_jpg: + +Display running rgb_python.py +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/rgb_python.jpg + :align: center + :alt: Display running rgb_python.py + +So why do only two lines appear at a time? That's how the display works. +Currently lines 6 and 22 are showing, then a moment later 7 and 23 show, etc. +The display can only display two lines at a time, so it cycles through all the +lines. Unfortunately, python is too slow to make the display appear +all at once. Here's where the PRU comes in. + +:ref:``blocks_rgb1`` is the PRU code to drive the RGB LED matrix. Be sure to run +``bone$ source rgb_setup.sh`` first. + +.. _blocks_rgb1: + +PRU code for driving the RGB LED matrix +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`rgb1.pru0.c <code/rgb1.pru0.c>` + + +The results are shown in :ref:`blocks_rgb_pru`. + +.. _blocks_rgb_pru: + +Display running rgb1.c on PRU 0 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/rgb_pru.jpg + :align: center + :alt: Display running rgb1.pru0.c on PRU 0 + +The PRU is fast enough to quickly write to the display so that it appears +as if all the LEDs are on at once. + +Discussion +----------- + +There are a lot of details needed to make this simple display work. Let's +go over some of them. + +First, the connector looks like :ref:`blocks_matrix_j1`. + +.. _blocks_matrix_j1: + +RGB Matrix J1 connector +~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/matrix_j1.jpg + :align: center + :alt: RGB Matrix J1 connector, 200 + +Notice the labels on the connect match the labels in the code. +:ref:`blocks_pocket_scroller_pins` shows how the pins on the display are +mapped to the pins on the Pocket Beagle. + +.. ToDo Make a mapping table for the Black +.. https://github.com/FalconChristmas/fpp/blob/master/src/pru/OctoscrollerV2.hp + +.. _blocks_pocket_scroller_pins: + +PocketScroller pin table +~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-----------------+---------------+------------------------+------------------+-------------------+ + |J1 Connector Pin |Pocket Headers |gpio port and bit number|Linux gpio number |PRU R30 bit number | + +=================+===============+========================+==================+===================+ + |R1 |P2_10 |1-20 |52 | | + +-----------------+---------------+------------------------+------------------+-------------------+ + |B1 |P2_06 |1-25 |57 | | + +-----------------+---------------+------------------------+------------------+-------------------+ + |R2 |P2_04 |1-26 |58 | | + +-----------------+---------------+------------------------+------------------+-------------------+ + |B2 |P2_01 |1-18 |50 | | + +-----------------+---------------+------------------------+------------------+-------------------+ + |LA |P2_32 |3-16 |112 |PRU0.2 | + +-----------------+---------------+------------------------+------------------+-------------------+ + |LC |P1_31 |3-18 |114 |PRU0.4 | + +-----------------+---------------+------------------------+------------------+-------------------+ + |CLK |P1_33 |3-15 |111 |PRU0.1 | + +-----------------+---------------+------------------------+------------------+-------------------+ + |OE |P1_29 |3-21 |117 |PRU0.7 | + +-----------------+---------------+------------------------+------------------+-------------------+ + |G1 |P2_08 |1-28 |60 | | + +-----------------+---------------+------------------------+------------------+-------------------+ + |G2 |P2_02 |1-27 |59 | | + +-----------------+---------------+------------------------+------------------+-------------------+ + |LB |P2_30 |3-17 |113 |PRU0.3 | + +-----------------+---------------+------------------------+------------------+-------------------+ + |LD |P2_34 |3-19 |115 |PRU0.5 | + +-----------------+---------------+------------------------+------------------+-------------------+ + |LAT |P1_36 |3-14 |110 |PRU0.0 | + +-----------------+---------------+------------------------+------------------+-------------------+ + +The J1 mapping to gpio port and bit number comes from +https://github.com/FalconChristmas/fpp/blob/master/capes/pb/panels/PocketScroller.json. +The gpio port and bit number mapping to Pocket Headers comes from +https://docs.google.com/spreadsheets/d/1FRGvYOyW1RiNSEVprvstfJAVeapnASgDXHtxeDOjgqw/edit#gid=0. + +:ref:`blocks_rgb_waveforms` shows four of the signal waveforms driving the RGB LED matrix. + +.. _blocks_rgb_waveforms: + +Oscilloscope display of CLK, OE, LAT and R1 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/rgb_waveforms.png + :align: center + :alt: .Oscilloscope display of CLK, OE, LAT and R1 + +The top waveform is the CLK, the next is OE, followed by LAT and finally R1. +The OE (output enable) is active low, so most of the time the display is visible. +The sequence is: + +* Put data on the R1, G1, B1, R2, G2 and B2 lines +* Toggle the clock. +* Repeat the first two steps as one row of data is transfered. There are +384 LEDs (2 rows of 32 RGB LEDs times 3 LED per RGB), but we are clocking in +six bits (R1, G1, etc.) at a time, so 384/6=64 values need to be clocked in. +* Once all the values are in, disable the display (OE goes high) +* Then toggle the latch (LAT) to latch the new data. +* Turn the display back on. +* Increment the address lines (LA, LB, LC and LD) to point to the next rows. +* Keep repeating the above to keep the display lit. + +Using the PRU we are able to run the clock a about 2.9 MKHz. +:ref:`blocks_rgb_fpp` shows the optimized assembler code used by FPP clocks in +at some 6.3 MHz. So the compiler is doing a pretty good job, but you can run +some two times faster if you want to use assembly code. In fairness to FPP, +it's having to pull it's data out of RAM to display it, so isn't not a good +comparision. + +.. _blocks_rgb_fpp: + +FPP waveforms +~~~~~~~~~~~~~~ + +.. figure:: figures/rgb_fpp.png + :align: center + :alt: FPP waveforms + +Getting More Colors +~~~~~~~~~~~~~~~~~~~~~ + +The Adafruit description goes on to say: + +.. admonition:: information + + The only downside of this technique is that despite being very simple and fast, + it has no PWM control built-in! The controller can only set the LEDs on or off. + So what do you do when you want full color? You actually need to draw the entire + matrix over and over again at very high speeds to PWM the matrix manually. For + that reason, you need to have a very fast controller (50 MHz is a minimum) if + you want to do a lot of colors and motion video and have it look good. + + https://cdn-learn.adafruit.com/downloads/pdf/32x16-32x32-rgb-led-matrix.pdf + +This is what FPP does, but it's beyond the scope of this project. + +Compiling and Inserting rpmsg_pru +====================================== + +Problem +----------- + +Your Beagle doesn't have rpmsg_pru. + +Solution +----------- + +Do the following. + +.. code-block:: bash + + bone$ *cd 05blocks/code/module* + bone$ *sudo apt install linux-headers-\`uname -r`* + bone$ *wget https://github.com/beagleboard/linux/raw/4.9/drivers/rpmsg/rpmsg_pru.c* + bone$ *make* + make -C /lib/modules/4.9.88-ti-r111/build M=$PWD + make[1]: Entering directory '/usr/src/linux-headers-4.9.88-ti-r111' + LD /home/debian/PRUCookbook/docs/05blocks/code/module/built-in.o + CC [M] /home/debian/PRUCookbook/docs/05blocks/code/module/rpmsg_client_sample.o + CC [M] /home/debian/PRUCookbook/docs/05blocks/code/module/rpmsg_pru.o + Building modules, stage 2. + MODPOST 2 modules + CC /home/debian/PRUCookbook/docs/05blocks/code/module/rpmsg_client_sample.mod.o + LD [M] /home/debian/PRUCookbook/docs/05blocks/code/module/rpmsg_client_sample.ko + CC /home/debian/PRUCookbook/docs/05blocks/code/module/rpmsg_pru.mod.o + LD [M] /home/debian/PRUCookbook/docs/05blocks/code/module/rpmsg_pru.ko + make[1]: Leaving directory '/usr/src/linux-headers-4.9.88-ti-r111' + bone$ *sudo insmod rpmsg_pru.ko* + bone$ *lsmod | grep rpm* + rpmsg_pru 5799 2 + virtio_rpmsg_bus 13620 0 + rpmsg_core 8537 2 rpmsg_pru,virtio_rpmsg_bus + + +It's now installed and ready to go. + + +Copyright +========== + +copyright.c +~~~~~~~~~~~~~ + +:download:`copyright.c <code/copyright.c>` + diff --git a/pru-cookbook/05blocks/code/Makefile b/pru-cookbook/05blocks/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/05blocks/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/05blocks/code/copyright.c b/pru-cookbook/05blocks/code/copyright.c new file mode 100644 index 0000000000000000000000000000000000000000..a34918faa44a789c3624c0e5551704e630017dae --- /dev/null +++ b/pru-cookbook/05blocks/code/copyright.c @@ -0,0 +1,32 @@ +/* + * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ diff --git a/pru-cookbook/05blocks/code/input.pru0.c b/pru-cookbook/05blocks/code/input.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..9ed5d3b759eb866fdbaa57fed9258c71ed32263e --- /dev/null +++ b/pru-cookbook/05blocks/code/input.pru0.c @@ -0,0 +1,26 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t led; + uint32_t sw; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + led = 0x1<<0; // P9_31 or P1_36 + sw = 0x1<<7; // P9_25 or P1_29 + + while (1) { + if((__R31&sw) == sw) { + __R30 |= led; // Turn on LED + } else + __R30 &= ~led; // Turn off LED + } +} + diff --git a/pru-cookbook/05blocks/code/input_setup.sh b/pru-cookbook/05blocks/code/input_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..35a0d9c3598288c14894a90907a69d62009eed1f --- /dev/null +++ b/pru-cookbook/05blocks/code/input_setup.sh @@ -0,0 +1,27 @@ +#!/bin/bash +# +export TARGET=input.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + config-pin P9_31 pruout + config-pin -q P9_31 + config-pin P9_25 pruin + config-pin -q P9_25 +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + config-pin P1_36 pruout + config-pin -q P1_36 + config-pin P1_29 pruin + config-pin -q P1_29 +else + echo " Not Found" + pins="" +fi diff --git a/pru-cookbook/05blocks/code/module/.gitignore b/pru-cookbook/05blocks/code/module/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..9b6dfc433724ec94f8905520931dcf33d409ea87 --- /dev/null +++ b/pru-cookbook/05blocks/code/module/.gitignore @@ -0,0 +1 @@ +*.ko diff --git a/pru-cookbook/05blocks/code/module/Makefile b/pru-cookbook/05blocks/code/module/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..9ca05d057c153a2d9b653fc64f5504a2d5aeea2e --- /dev/null +++ b/pru-cookbook/05blocks/code/module/Makefile @@ -0,0 +1,25 @@ +# Makefile for compiling out-of-tree +# From Free Electrons +# If on the Bone run apt install linux-headers-`uname -r` + +KDIR := /lib/modules/$(shell uname -r)/build + +obj-m += rpmsg_client_sample.o +obj-m += rpmsg_pru.o + +all: + $(MAKE) -C $(KDIR) M=$$PWD + +install: + scp hello*.ko bone:. + +print: + @echo KERNELRELEASE= $(KERNELRELEASE) + @echo CONFIG_EXAMPLES= $(CONFIG_EXAMPLES) + @echo obj-m= $(obj-m) + +clean: + rm -rf .tmp_versions *.o + rm -rf .rpmsg_client_sample* rpmsg_client_sample*.mod.c + rm -rf .rpmsg_pru* rpmsg_pru*.mod.c + rm modules.order Module.symvers .built-in.o.cmd \ No newline at end of file diff --git a/pru-cookbook/05blocks/code/module/install.sh b/pru-cookbook/05blocks/code/module/install.sh new file mode 100755 index 0000000000000000000000000000000000000000..89f497f038396b8bfaddcec73330b74255e0c5ab --- /dev/null +++ b/pru-cookbook/05blocks/code/module/install.sh @@ -0,0 +1,6 @@ +# This is a example of ARM to PRU communication from Lab 5 of +# http://processors.wiki.ti.com/index.php/PRU_Training:_Hands-on_Labs#LAB_5:_RPMsg_Communication_between_ARM_and_PRU + +sudo apt install linux-headers-`uname -r` + +wget https://github.com/beagleboard/linux/raw/4.9/drivers/rpmsg/rpmsg_pru.c diff --git a/pru-cookbook/05blocks/code/module/rpmsg_client_sample.c b/pru-cookbook/05blocks/code/module/rpmsg_client_sample.c new file mode 100644 index 0000000000000000000000000000000000000000..72b24524c626cca0291d93b5d765ecf515dac0f3 --- /dev/null +++ b/pru-cookbook/05blocks/code/module/rpmsg_client_sample.c @@ -0,0 +1,102 @@ +/* + * Remote processor messaging - sample client driver + * + * Copyright (C) 2011 Texas Instruments, Inc. + * Copyright (C) 2011 Google, Inc. + * + * Ohad Ben-Cohen <ohad@wizery.com> + * Brian Swetland <swetland@google.com> + * + * This software is licensed under the terms of the GNU General Public + * License version 2, as published by the Free Software Foundation, and + * may be copied, distributed, and modified under those terms. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#include <linux/kernel.h> +#include <linux/module.h> +#include <linux/rpmsg.h> + +#define MSG "hello world!" +#define MSG_LIMIT 10 + +struct instance_data { + int rx_count; +}; + +static int rpmsg_sample_cb(struct rpmsg_device *rpdev, void *data, int len, + void *priv, u32 src) +{ + int ret; + struct instance_data *idata = dev_get_drvdata(&rpdev->dev); + + dev_info(&rpdev->dev, "incoming msg %d (src: 0x%x)\n", + ++idata->rx_count, src); + + print_hex_dump(KERN_DEBUG, __func__, DUMP_PREFIX_NONE, 16, 1, + data, len, true); + + /* samples should not live forever */ + if (idata->rx_count >= MSG_LIMIT) { + dev_info(&rpdev->dev, "goodbye!\n"); + return 0; + } + + /* send a new message now */ + ret = rpmsg_send(rpdev->ept, MSG, strlen(MSG)); + if (ret) + dev_err(&rpdev->dev, "rpmsg_send failed: %d\n", ret); + + return 0; +} + +static int rpmsg_sample_probe(struct rpmsg_device *rpdev) +{ + int ret; + struct instance_data *idata; + + dev_info(&rpdev->dev, "new channel: 0x%x -> 0x%x!\n", + rpdev->src, rpdev->dst); + + idata = devm_kzalloc(&rpdev->dev, sizeof(*idata), GFP_KERNEL); + if (!idata) + return -ENOMEM; + + dev_set_drvdata(&rpdev->dev, idata); + + /* send a message to our remote processor */ + ret = rpmsg_send(rpdev->ept, MSG, strlen(MSG)); + if (ret) { + dev_err(&rpdev->dev, "rpmsg_send failed: %d\n", ret); + return ret; + } + + return 0; +} + +static void rpmsg_sample_remove(struct rpmsg_device *rpdev) +{ + dev_info(&rpdev->dev, "rpmsg sample client driver is removed\n"); +} + +static struct rpmsg_device_id rpmsg_driver_sample_id_table[] = { + { .name = "rpmsg-client-sample" }, + { }, +}; +MODULE_DEVICE_TABLE(rpmsg, rpmsg_driver_sample_id_table); + +static struct rpmsg_driver rpmsg_sample_client = { + .drv.name = KBUILD_MODNAME, + .id_table = rpmsg_driver_sample_id_table, + .probe = rpmsg_sample_probe, + .callback = rpmsg_sample_cb, + .remove = rpmsg_sample_remove, +}; +module_rpmsg_driver(rpmsg_sample_client); + +MODULE_DESCRIPTION("Remote processor messaging sample client driver"); +MODULE_LICENSE("GPL v2"); diff --git a/pru-cookbook/05blocks/code/module/rpmsg_pru.c b/pru-cookbook/05blocks/code/module/rpmsg_pru.c new file mode 100644 index 0000000000000000000000000000000000000000..5a96b20965035e5b12139f391a41529c94ff0dbc --- /dev/null +++ b/pru-cookbook/05blocks/code/module/rpmsg_pru.c @@ -0,0 +1,359 @@ +/* + * PRU Remote Processor Messaging Driver + * + * Copyright (C) 2015-2017 Texas Instruments Incorporated - http://www.ti.com/ + * Jason Reeder <jreeder@ti.com> + * Suman Anna <s-anna@ti.com> + * + * This software is licensed under the terms of the GNU General Public + * License version 2, as published by the Free Software Foundation, and + * may be copied, distributed, and modified under those terms. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#include <linux/kernel.h> +#include <linux/rpmsg.h> +#include <linux/slab.h> +#include <linux/fs.h> +#include <linux/init.h> +#include <linux/cdev.h> +#include <linux/module.h> +#include <linux/kfifo.h> +#include <linux/uaccess.h> +#include <linux/mutex.h> +#include <linux/poll.h> +#include <linux/rpmsg/virtio_rpmsg.h> + +#define PRU_MAX_DEVICES (8) +/* Matches the definition in virtio_rpmsg_bus.c */ +#define RPMSG_BUF_SIZE (512) +#define MAX_FIFO_MSG (32) +#define FIFO_MSG_SIZE RPMSG_BUF_SIZE + +/** + * struct rpmsg_pru_dev - Structure that contains the per-device data + * @rpdev: rpmsg channel device that is associated with this rpmsg_pru device + * @dev: device + * @cdev: character device + * @locked: boolean used to determine whether or not the device file is in use + * @devt: dev_t structure for the rpmsg_pru device + * @msg_fifo: kernel fifo used to buffer the messages between userspace and PRU + * @msg_len: array storing the lengths of each message in the kernel fifo + * @msg_idx_rd: kernel fifo read index + * @msg_idx_wr: kernel fifo write index + * @wait_list: wait queue used to implement the poll operation of the character + * device + * + * Each rpmsg_pru device provides an interface, using an rpmsg channel (rpdev), + * between a user space character device (cdev) and a PRU core. A kernel fifo + * (msg_fifo) is used to buffer the messages in the kernel that are + * being passed between the character device and the PRU. + */ +struct rpmsg_pru_dev { + struct rpmsg_device *rpdev; + struct device *dev; + struct cdev cdev; + bool locked; + dev_t devt; + struct kfifo msg_fifo; + u32 msg_len[MAX_FIFO_MSG]; + int msg_idx_rd; + int msg_idx_wr; + wait_queue_head_t wait_list; +}; + +static struct class *rpmsg_pru_class; +static dev_t rpmsg_pru_devt; +static DEFINE_MUTEX(rpmsg_pru_lock); +static DEFINE_IDR(rpmsg_pru_minors); + +static int rpmsg_pru_open(struct inode *inode, struct file *filp) +{ + struct rpmsg_pru_dev *prudev; + int ret = -EACCES; + + prudev = container_of(inode->i_cdev, struct rpmsg_pru_dev, cdev); + + mutex_lock(&rpmsg_pru_lock); + if (!prudev->locked) { + prudev->locked = true; + filp->private_data = prudev; + ret = 0; + } + mutex_unlock(&rpmsg_pru_lock); + + if (ret) + dev_err(prudev->dev, "Device already open\n"); + + return ret; +} + +static int rpmsg_pru_release(struct inode *inode, struct file *filp) +{ + struct rpmsg_pru_dev *prudev; + + prudev = container_of(inode->i_cdev, struct rpmsg_pru_dev, cdev); + mutex_lock(&rpmsg_pru_lock); + prudev->locked = false; + mutex_unlock(&rpmsg_pru_lock); + return 0; +} + +static ssize_t rpmsg_pru_read(struct file *filp, char __user *buf, + size_t count, loff_t *f_pos) +{ + int ret; + u32 length; + struct rpmsg_pru_dev *prudev; + + prudev = filp->private_data; + + if (kfifo_is_empty(&prudev->msg_fifo) && + (filp->f_flags & O_NONBLOCK)) + return -EAGAIN; + + ret = wait_event_interruptible(prudev->wait_list, + !kfifo_is_empty(&prudev->msg_fifo)); + if (ret) + return -EINTR; + + ret = kfifo_to_user(&prudev->msg_fifo, buf, + prudev->msg_len[prudev->msg_idx_rd], &length); + prudev->msg_idx_rd = (prudev->msg_idx_rd + 1) % MAX_FIFO_MSG; + + return ret ? ret : length; +} + +static ssize_t rpmsg_pru_write(struct file *filp, const char __user *buf, + size_t count, loff_t *f_pos) +{ + int ret; + struct rpmsg_pru_dev *prudev; + static char rpmsg_pru_buf[RPMSG_BUF_SIZE]; + + prudev = filp->private_data; + + if (count > RPMSG_BUF_SIZE - sizeof(struct rpmsg_hdr)) { + dev_err(prudev->dev, "Data too large for RPMsg Buffer\n"); + return -EINVAL; + } + + if (copy_from_user(rpmsg_pru_buf, buf, count)) { + dev_err(prudev->dev, "Error copying buffer from user space"); + return -EFAULT; + } + + ret = rpmsg_send(prudev->rpdev->ept, (void *)rpmsg_pru_buf, count); + if (ret) + dev_err(prudev->dev, "rpmsg_send failed: %d\n", ret); + + return ret ? ret : count; +} + +static unsigned int rpmsg_pru_poll(struct file *filp, + struct poll_table_struct *wait) +{ + int mask; + struct rpmsg_pru_dev *prudev; + + prudev = filp->private_data; + + poll_wait(filp, &prudev->wait_list, wait); + + mask = POLLOUT | POLLWRNORM; + + if (!kfifo_is_empty(&prudev->msg_fifo)) + mask |= POLLIN | POLLRDNORM; + + return mask; +} + +static const struct file_operations rpmsg_pru_fops = { + .owner = THIS_MODULE, + .open = rpmsg_pru_open, + .release = rpmsg_pru_release, + .read = rpmsg_pru_read, + .write = rpmsg_pru_write, + .poll = rpmsg_pru_poll, +}; + +static int rpmsg_pru_cb(struct rpmsg_device *rpdev, void *data, int len, + void *priv, u32 src) +{ + u32 length; + struct rpmsg_pru_dev *prudev; + + prudev = dev_get_drvdata(&rpdev->dev); + + if (kfifo_avail(&prudev->msg_fifo) < len) { + dev_err(&rpdev->dev, "Not enough space on the FIFO\n"); + return -ENOSPC; + } + + if ((prudev->msg_idx_wr + 1) % MAX_FIFO_MSG == + prudev->msg_idx_rd) { + dev_err(&rpdev->dev, "Message length table is full\n"); + return -ENOSPC; + } + + length = kfifo_in(&prudev->msg_fifo, data, len); + prudev->msg_len[prudev->msg_idx_wr] = length; + prudev->msg_idx_wr = (prudev->msg_idx_wr + 1) % MAX_FIFO_MSG; + + wake_up_interruptible(&prudev->wait_list); + + return 0; +} + +static int rpmsg_pru_probe(struct rpmsg_device *rpdev) +{ + int ret; + struct rpmsg_pru_dev *prudev; + int minor_got; + + prudev = devm_kzalloc(&rpdev->dev, sizeof(*prudev), GFP_KERNEL); + if (!prudev) + return -ENOMEM; + + mutex_lock(&rpmsg_pru_lock); + minor_got = idr_alloc(&rpmsg_pru_minors, prudev, 0, PRU_MAX_DEVICES, + GFP_KERNEL); + mutex_unlock(&rpmsg_pru_lock); + if (minor_got < 0) { + ret = minor_got; + dev_err(&rpdev->dev, "Failed to get a minor number for the rpmsg_pru device: %d\n", + ret); + goto fail_alloc_minor; + } + + prudev->devt = MKDEV(MAJOR(rpmsg_pru_devt), minor_got); + + cdev_init(&prudev->cdev, &rpmsg_pru_fops); + prudev->cdev.owner = THIS_MODULE; + ret = cdev_add(&prudev->cdev, prudev->devt, 1); + if (ret) { + dev_err(&rpdev->dev, "Unable to add cdev for the rpmsg_pru device\n"); + goto fail_add_cdev; + } + + prudev->dev = device_create(rpmsg_pru_class, &rpdev->dev, prudev->devt, + NULL, "rpmsg_pru%d", rpdev->dst); + if (IS_ERR(prudev->dev)) { + dev_err(&rpdev->dev, "Unable to create the rpmsg_pru device\n"); + ret = PTR_ERR(prudev->dev); + goto fail_create_device; + } + + prudev->rpdev = rpdev; + + ret = kfifo_alloc(&prudev->msg_fifo, MAX_FIFO_MSG * FIFO_MSG_SIZE, + GFP_KERNEL); + if (ret) { + dev_err(&rpdev->dev, "Unable to allocate fifo for the rpmsg_pru device\n"); + goto fail_alloc_fifo; + } + + init_waitqueue_head(&prudev->wait_list); + + dev_set_drvdata(&rpdev->dev, prudev); + + dev_info(&rpdev->dev, "new rpmsg_pru device: /dev/rpmsg_pru%d", + rpdev->dst); + + return 0; + +fail_alloc_fifo: + device_destroy(rpmsg_pru_class, prudev->devt); +fail_create_device: + cdev_del(&prudev->cdev); +fail_add_cdev: + mutex_lock(&rpmsg_pru_lock); + idr_remove(&rpmsg_pru_minors, minor_got); + mutex_unlock(&rpmsg_pru_lock); +fail_alloc_minor: + return ret; +} + +static void rpmsg_pru_remove(struct rpmsg_device *rpdev) +{ + struct rpmsg_pru_dev *prudev; + + prudev = dev_get_drvdata(&rpdev->dev); + + kfifo_free(&prudev->msg_fifo); + device_destroy(rpmsg_pru_class, prudev->devt); + cdev_del(&prudev->cdev); + mutex_lock(&rpmsg_pru_lock); + idr_remove(&rpmsg_pru_minors, MINOR(prudev->devt)); + mutex_unlock(&rpmsg_pru_lock); +} + +/* .name matches on RPMsg Channels and causes a probe */ +static const struct rpmsg_device_id rpmsg_driver_pru_id_table[] = { + { .name = "rpmsg-pru" }, + { }, +}; +MODULE_DEVICE_TABLE(rpmsg, rpmsg_driver_pru_id_table); + +static struct rpmsg_driver rpmsg_pru_driver = { + .drv.name = KBUILD_MODNAME, + .id_table = rpmsg_driver_pru_id_table, + .probe = rpmsg_pru_probe, + .callback = rpmsg_pru_cb, + .remove = rpmsg_pru_remove, +}; + +static int __init rpmsg_pru_init(void) +{ + int ret; + + rpmsg_pru_class = class_create(THIS_MODULE, "rpmsg_pru"); + if (IS_ERR(rpmsg_pru_class)) { + pr_err("Unable to create class\n"); + ret = PTR_ERR(rpmsg_pru_class); + goto fail_create_class; + } + + ret = alloc_chrdev_region(&rpmsg_pru_devt, 0, PRU_MAX_DEVICES, + "rpmsg_pru"); + if (ret) { + pr_err("Unable to allocate chrdev region\n"); + goto fail_alloc_region; + } + + ret = register_rpmsg_driver(&rpmsg_pru_driver); + if (ret) { + pr_err("Unable to register rpmsg driver"); + goto fail_register_rpmsg_driver; + } + + return 0; + +fail_register_rpmsg_driver: + unregister_chrdev_region(rpmsg_pru_devt, PRU_MAX_DEVICES); +fail_alloc_region: + class_destroy(rpmsg_pru_class); +fail_create_class: + return ret; +} + +static void __exit rpmsg_pru_exit(void) +{ + unregister_rpmsg_driver(&rpmsg_pru_driver); + idr_destroy(&rpmsg_pru_minors); + mutex_destroy(&rpmsg_pru_lock); + class_destroy(rpmsg_pru_class); + unregister_chrdev_region(rpmsg_pru_devt, PRU_MAX_DEVICES); +} + +module_init(rpmsg_pru_init); +module_exit(rpmsg_pru_exit); + +MODULE_AUTHOR("Jason Reeder <jreeder@ti.com>"); +MODULE_ALIAS("rpmsg:rpmsg-pru"); +MODULE_DESCRIPTION("PRU Remote Processor Messaging Driver"); +MODULE_LICENSE("GPL v2"); diff --git a/pru-cookbook/05blocks/code/module/setup.sh b/pru-cookbook/05blocks/code/module/setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..3b313b366052438b0db8e7a7c0f00a04313b8687 --- /dev/null +++ b/pru-cookbook/05blocks/code/module/setup.sh @@ -0,0 +1 @@ +insmod rpmsg_pru.ko diff --git a/pru-cookbook/05blocks/code/neo-colors.py b/pru-cookbook/05blocks/code/neo-colors.py new file mode 100755 index 0000000000000000000000000000000000000000..736a47b3c7e6ae2507da296fdae5ac01be7f7a14 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo-colors.py @@ -0,0 +1,41 @@ +#!/usr/bin/python3 +from time import sleep +import math + +length = 24 +max = 25 + +# Open a file +fo = open("/dev/rpmsg_pru30", "wb", 0) + +colors = [[1,0,0],[1,1,0],[0,1,0],[0,1,1],[0,0,1],[1,0,1]]# colors = [[1,0,0],[1,0,0]] + +oldr=0 +oldb=0 +oldg=0 + +while True: + for color in colors: + newr = color[0] + newg = color[1] + newb = color[2] + maxtime=20 + for time in range(0, maxtime): + r = (max*oldr+(newr-oldr)*max*time/maxtime) + g = (max*oldg+(newg-oldg)*max*time/maxtime) + b = (max*oldb+(newb-oldb)*max*time/maxtime) + for i in range(0, length): + fo.write(b"%d %d %d %d\n" % (i, r, g, b)) + # print("0 0 127 %d" % (i)) + fo.write(b"-1 0 0 0\n"); # Send colors to LEDs + + # print (r,g,b) + + sleep(0.05) + + oldr=newr + oldg=newg + oldb=newb + +# Close opened file +fo.close() \ No newline at end of file diff --git a/pru-cookbook/05blocks/code/neo-rainbow.py b/pru-cookbook/05blocks/code/neo-rainbow.py new file mode 100755 index 0000000000000000000000000000000000000000..ffd706a643a9c71609e8b8893b6ff91b53cde08b --- /dev/null +++ b/pru-cookbook/05blocks/code/neo-rainbow.py @@ -0,0 +1,27 @@ +#!/usr/bin/python3 +from time import sleep +import math + +len = 24 +amp = 12 +f = 25 +shift = 3 +phase = 0 + +# Open a file +fo = open("/dev/rpmsg_pru30", "wb", 0) + +while True: + for i in range(0, len): + r = (amp * (math.sin(2*math.pi*f*(i-phase-0*shift)/len) + 1)) + 1; + g = (amp * (math.sin(2*math.pi*f*(i-phase-1*shift)/len) + 1)) + 1; + b = (amp * (math.sin(2*math.pi*f*(i-phase-2*shift)/len) + 1)) + 1; + fo.write(b"%d %d %d %d\n" % (i, r, g, b)) + # print("0 0 127 %d" % (i)) + + fo.write(b"-1 0 0 0\n"); + phase = phase + 1 + sleep(0.05) + +# Close opened file +fo.close() \ No newline at end of file diff --git a/pru-cookbook/05blocks/code/neo1.pru0.c b/pru-cookbook/05blocks/code/neo1.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..5f2ce8248d73706b8f78c7b76e5df9009ffd4b02 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo1.pru0.c @@ -0,0 +1,44 @@ +// Control a ws2812 (NeoPixel) display, All on or all off +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define STR_LEN 24 +#define oneCyclesOn 700/5 // Stay on 700ns +#define oneCyclesOff 800/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 600/5 +#define resetCycles 60000/5 // Must be at least 50u, use 60u +#define gpio P9_29 // output pin + +#define ONE + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + uint32_t i; + for(i=0; i<STR_LEN*3*8; i++) { +#ifdef ONE + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-2); +#else + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-2); +#endif + } + // Send Reset + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(resetCycles); + + __halt(); +} diff --git a/pru-cookbook/05blocks/code/neo1.pru1_1.c b/pru-cookbook/05blocks/code/neo1.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..2ad8d56d1cff431ab3c006788440454083b26e96 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo1.pru1_1.c @@ -0,0 +1,44 @@ +// Control a ws2812 (NeoPixel) display, All on or all off +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define STR_LEN 24 +#define oneCyclesOn 700/5 // Stay on 700ns +#define oneCyclesOff 800/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 600/5 +#define resetCycles 60000/5 // Must be at least 50u, use 60u +#define gpio P9_16 // output pin + +#define ONE + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + uint32_t i; + for(i=0; i<STR_LEN*3*8; i++) { +#ifdef ONE + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-2); +#else + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-2); +#endif + } + // Send Reset + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(resetCycles); + + __halt(); +} diff --git a/pru-cookbook/05blocks/code/neo2.pru0.c b/pru-cookbook/05blocks/code/neo2.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..a30077ceb090f7e3722cb0fe1a61162bdde9bb9d --- /dev/null +++ b/pru-cookbook/05blocks/code/neo2.pru0.c @@ -0,0 +1,46 @@ +// Control a ws2812 (neo pixel) display, green, red, blue, green, ... +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define STR_LEN 3 +#define oneCyclesOn 700/5 // Stay on 700ns +#define oneCyclesOff 800/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 600/5 +#define resetCycles 60000/5 // Must be at least 50u, use 60u +#define gpio P9_29 // output pin + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + uint32_t color[STR_LEN] = {0x0f0000, 0x000f00, 0x0000f}; // green, red, blue + int i, j; + + for(j=0; j<STR_LEN; j++) { + for(i=23; i>=0; i--) { + if(color[j] & (0x1<<i)) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-2); + } else { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-2); + } + } + } + // Send Reset + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(resetCycles); + + __halt(); +} diff --git a/pru-cookbook/05blocks/code/neo2.pru1_1.c b/pru-cookbook/05blocks/code/neo2.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..6b243dcfb53c2139f44d4a2aabd113a274c331e8 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo2.pru1_1.c @@ -0,0 +1,46 @@ +// Control a ws2812 (neo pixel) display, green, red, blue, green, ... +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define STR_LEN 3 +#define oneCyclesOn 700/5 // Stay on 700ns +#define oneCyclesOff 800/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 600/5 +#define resetCycles 60000/5 // Must be at least 50u, use 60u +#define gpio P9_16 // output pin + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + uint32_t color[STR_LEN] = {0x0f0000, 0x000f00, 0x0000f}; // green, red, blue + int i, j; + + for(j=0; j<STR_LEN; j++) { + for(i=23; i>=0; i--) { + if(color[j] & (0x1<<i)) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-2); + } else { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-2); + } + } + } + // Send Reset + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(resetCycles); + + __halt(); +} diff --git a/pru-cookbook/05blocks/code/neo3.pru0.c b/pru-cookbook/05blocks/code/neo3.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..8c1a9f1323d564d7894b299f1bd4f8a4020bc5e8 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo3.pru0.c @@ -0,0 +1,67 @@ +// Control a ws2812 (neo pixel) display, green, red, blue, green, ... +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define STR_LEN 24 +#define oneCyclesOn 700/5 // Stay on 700ns +#define oneCyclesOff 800/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 600/5 +#define resetCycles 60000/5 // Must be at least 50u, use 60u +#define gpio P9_29 // output pin + +#define SPEED 20000000/5 // Time to wait between updates + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t background = 0x00000f; + uint32_t foreground = 0x000f00; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + uint32_t color[STR_LEN]; // green, red, blue + int i, j; + int k, oldk = 0;; + // Set everything to background + for(i=0; i<STR_LEN; i++) { + color[i] = background; + } + + while(1) { + // Move forward one position + for(k=0; k<STR_LEN; k++) { + color[oldk] = background; + color[k] = foreground; + oldk=k; + + // Output the string + for(j=0; j<STR_LEN; j++) { + for(i=23; i>=0; i--) { + if(color[j] & (0x1<<i)) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-2); + } else { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-2); + } + } + } + // Send Reset + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(resetCycles); + + // Wait + __delay_cycles(SPEED); + } + } +} diff --git a/pru-cookbook/05blocks/code/neo3.pru1_1.c b/pru-cookbook/05blocks/code/neo3.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..ee8dd63859680a930b3dca6c8c61895a87051ac9 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo3.pru1_1.c @@ -0,0 +1,67 @@ +// Control a ws2812 (neo pixel) display, green, red, blue, green, ... +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define STR_LEN 24 +#define oneCyclesOn 700/5 // Stay on 700ns +#define oneCyclesOff 800/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 600/5 +#define resetCycles 60000/5 // Must be at least 50u, use 60u +#define gpio P9_16 // output pin + +#define SPEED 20000000/5 // Time to wait between updates + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t background = 0x00000f; + uint32_t foreground = 0x000f00; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + uint32_t color[STR_LEN]; // green, red, blue + int i, j; + int k, oldk = 0;; + // Set everything to background + for(i=0; i<STR_LEN; i++) { + color[i] = background; + } + + while(1) { + // Move forward one position + for(k=0; k<STR_LEN; k++) { + color[oldk] = background; + color[k] = foreground; + oldk=k; + + // Output the string + for(j=0; j<STR_LEN; j++) { + for(i=23; i>=0; i--) { + if(color[j] & (0x1<<i)) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-2); + } else { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-2); + } + } + } + // Send Reset + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(resetCycles); + + // Wait + __delay_cycles(SPEED); + } + } +} diff --git a/pru-cookbook/05blocks/code/neo4.pru0.c b/pru-cookbook/05blocks/code/neo4.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..7fb5455781d8dc47cd414a433b1578ea84645ad9 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo4.pru0.c @@ -0,0 +1,145 @@ +// Use rpmsg to control the NeoPixels via /dev/rpmsg_pru30 +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> // atoi +#include <string.h> +#include <pru_cfg.h> +#include <pru_intc.h> +#include <rsc_types.h> +#include <pru_rpmsg.h> +#include "resource_table_0.h" +#include "prugpio.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +/* Host-0 Interrupt sets bit 30 in register R31 */ +#define HOST_INT ((uint32_t) 1 << 30) + +/* The PRU-ICSS system events used for RPMsg are defined in the Linux device tree + * PRU0 uses system event 16 (To ARM) and 17 (From ARM) + * PRU1 uses system event 18 (To ARM) and 19 (From ARM) + */ +#define TO_ARM_HOST 16 +#define FROM_ARM_HOST 17 + +/* +* Using the name 'rpmsg-pru' will probe the rpmsg_pru driver found +* at linux-x.y.z/drivers/rpmsg/rpmsg_pru.c +*/ +#define CHAN_NAME "rpmsg-pru" +#define CHAN_DESC "Channel 30" +#define CHAN_PORT 30 + +/* + * Used to make sure the Linux drivers are ready for RPMsg communication + * Found at linux-x.y.z/include/uapi/linux/virtio_config.h + */ +#define VIRTIO_CONFIG_S_DRIVER_OK 4 + +char payload[RPMSG_BUF_SIZE]; + +#define STR_LEN 24 +#define oneCyclesOn 700/5 // Stay on for 700ns +#define oneCyclesOff 600/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 800/5 +#define resetCycles 51000/5 // Must be at least 50u, use 51u +#define out P9_29 // Bit number to output on + +#define SPEED 20000000/5 // Time to wait between updates + +uint32_t color[STR_LEN]; // green, red, blue + +/* + * main.c + */ +void main(void) +{ + struct pru_rpmsg_transport transport; + uint16_t src, dst, len; + volatile uint8_t *status; + + uint8_t r, g, b; + int i, j; + // Set everything to background + for(i=0; i<STR_LEN; i++) { + color[i] = 0x010000; + } + + /* Allow OCP master port access by the PRU so the PRU can read external memories */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + /* Clear the status of the PRU-ICSS system event that the ARM will use to 'kick' us */ +#ifdef CHIP_IS_am57xx + CT_INTC.SICR_bit.STATUS_CLR_INDEX = FROM_ARM_HOST; +#else + CT_INTC.SICR_bit.STS_CLR_IDX = FROM_ARM_HOST; +#endif + + /* Make sure the Linux drivers are ready for RPMsg communication */ + status = &resourceTable.rpmsg_vdev.status; + while (!(*status & VIRTIO_CONFIG_S_DRIVER_OK)); + + /* Initialize the RPMsg transport structure */ + pru_rpmsg_init(&transport, &resourceTable.rpmsg_vring0, &resourceTable.rpmsg_vring1, TO_ARM_HOST, FROM_ARM_HOST); + + /* Create the RPMsg channel between the PRU and ARM user space using the transport structure. */ + while (pru_rpmsg_channel(RPMSG_NS_CREATE, &transport, CHAN_NAME, CHAN_DESC, CHAN_PORT) != PRU_RPMSG_SUCCESS); + while (1) { + /* Check bit 30 of register R31 to see if the ARM has kicked us */ + if (__R31 & HOST_INT) { + /* Clear the event status */ +#ifdef CHIP_IS_am57xx + CT_INTC.SICR_bit.STATUS_CLR_INDEX = FROM_ARM_HOST; +#else + CT_INTC.SICR_bit.STS_CLR_IDX = FROM_ARM_HOST; +#endif + /* Receive all available messages, multiple messages can be sent per kick */ + while (pru_rpmsg_receive(&transport, &src, &dst, payload, &len) == PRU_RPMSG_SUCCESS) { + char *ret; // rest of payload after front character is removed + int index; // index of LED to control + // Input format is: index red green blue + index = atoi(payload); + // Update the array, but don't write it out. + if((index >=0) & (index < STR_LEN)) { + ret = strchr(payload, ' '); // Skip over index + r = strtol(&ret[1], NULL, 0); + ret = strchr(&ret[1], ' '); // Skip over r, etc. + g = strtol(&ret[1], NULL, 0); + ret = strchr(&ret[1], ' '); + b = strtol(&ret[1], NULL, 0); + + color[index] = (g<<16)|(r<<8)|b; // String wants GRB + } + // When index is -1, send the array to the LED string + if(index == -1) { + // Output the string + for(j=0; j<STR_LEN; j++) { + // Cycle through each bit + for(i=23; i>=0; i--) { + if(color[j] & (0x1<<i)) { + __R30 |= out; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~out; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-14); + } else { + __R30 |= out; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~(out); // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-14); + } + } + } + // Send Reset + __R30 &= ~out; // Clear the GPIO pin + __delay_cycles(resetCycles); + + // Wait + __delay_cycles(SPEED); + } + + } + } + } +} diff --git a/pru-cookbook/05blocks/code/neo4.pru1_1.c b/pru-cookbook/05blocks/code/neo4.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..7c120937aa92b19fbc65971aa8a6ec3748c69ca4 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo4.pru1_1.c @@ -0,0 +1,145 @@ +// Use rpmsg to control the NeoPixels via /dev/rpmsg_pru30 +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> // atoi +#include <string.h> +#include <pru_cfg.h> +#include <pru_intc.h> +#include <rsc_types.h> +#include <pru_rpmsg.h> +#include "resource_table_1.h" +#include "prugpio.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +/* Host-1 Interrupt sets bit 31 in register R31 */ +#define HOST_INT ((uint32_t) 1 << 31) + +/* The PRU-ICSS system events used for RPMsg are defined in the Linux device tree + * PRU0 uses system event 16 (To ARM) and 17 (From ARM) + * PRU1 uses system event 18 (To ARM) and 19 (From ARM) + */ +#define TO_ARM_HOST 18 +#define FROM_ARM_HOST 19 + +/* +* Using the name 'rpmsg-pru' will probe the rpmsg_pru driver found +* at linux-x.y.z/drivers/rpmsg/rpmsg_pru.c +*/ +#define CHAN_NAME "rpmsg-pru" +#define CHAN_DESC "Channel 31" +#define CHAN_PORT 31 + +/* + * Used to make sure the Linux drivers are ready for RPMsg communication + * Found at linux-x.y.z/include/uapi/linux/virtio_config.h + */ +#define VIRTIO_CONFIG_S_DRIVER_OK 4 + +char payload[RPMSG_BUF_SIZE]; + +#define STR_LEN 24 +#define oneCyclesOn 700/5 // Stay on for 700ns +#define oneCyclesOff 600/5 +#define zeroCyclesOn 350/5 +#define zeroCyclesOff 800/5 +#define resetCycles 51000/5 // Must be at least 50u, use 51u +#define out P9_16 // Bit number to output on + +#define SPEED 20000000/5 // Time to wait between updates + +uint32_t color[STR_LEN]; // green, red, blue + +/* + * main.c + */ +void main(void) +{ + struct pru_rpmsg_transport transport; + uint16_t src, dst, len; + volatile uint8_t *status; + + uint8_t r, g, b; + int i, j; + // Set everything to background + for(i=0; i<STR_LEN; i++) { + color[i] = 0x010000; + } + + /* Allow OCP master port access by the PRU so the PRU can read external memories */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + /* Clear the status of the PRU-ICSS system event that the ARM will use to 'kick' us */ +#ifdef CHIP_IS_am57xx + CT_INTC.SICR_bit.STATUS_CLR_INDEX = FROM_ARM_HOST; +#else + CT_INTC.SICR_bit.STS_CLR_IDX = FROM_ARM_HOST; +#endif + + /* Make sure the Linux drivers are ready for RPMsg communication */ + status = &resourceTable.rpmsg_vdev.status; + while (!(*status & VIRTIO_CONFIG_S_DRIVER_OK)); + + /* Initialize the RPMsg transport structure */ + pru_rpmsg_init(&transport, &resourceTable.rpmsg_vring0, &resourceTable.rpmsg_vring1, TO_ARM_HOST, FROM_ARM_HOST); + + /* Create the RPMsg channel between the PRU and ARM user space using the transport structure. */ + while (pru_rpmsg_channel(RPMSG_NS_CREATE, &transport, CHAN_NAME, CHAN_DESC, CHAN_PORT) != PRU_RPMSG_SUCCESS); + while (1) { + /* Check bit 30 of register R31 to see if the ARM has kicked us */ + if (__R31 & HOST_INT) { + /* Clear the event status */ +#ifdef CHIP_IS_am57xx + CT_INTC.SICR_bit.STATUS_CLR_INDEX = FROM_ARM_HOST; +#else + CT_INTC.SICR_bit.STS_CLR_IDX = FROM_ARM_HOST; +#endif + /* Receive all available messages, multiple messages can be sent per kick */ + while (pru_rpmsg_receive(&transport, &src, &dst, payload, &len) == PRU_RPMSG_SUCCESS) { + char *ret; // rest of payload after front character is removed + int index; // index of LED to control + // Input format is: index red green blue + index = atoi(payload); + // Update the array, but don't write it out. + if((index >=0) & (index < STR_LEN)) { + ret = strchr(payload, ' '); // Skip over index + r = strtol(&ret[1], NULL, 0); + ret = strchr(&ret[1], ' '); // Skip over r, etc. + g = strtol(&ret[1], NULL, 0); + ret = strchr(&ret[1], ' '); + b = strtol(&ret[1], NULL, 0); + + color[index] = (g<<16)|(r<<8)|b; // String wants GRB + } + // When index is -1, send the array to the LED string + if(index == -1) { + // Output the string + for(j=0; j<STR_LEN; j++) { + // Cycle through each bit + for(i=23; i>=0; i--) { + if(color[j] & (0x1<<i)) { + __R30 |= out; // Set the GPIO pin to 1 + __delay_cycles(oneCyclesOn-1); + __R30 &= ~out; // Clear the GPIO pin + __delay_cycles(oneCyclesOff-14); + } else { + __R30 |= out; // Set the GPIO pin to 1 + __delay_cycles(zeroCyclesOn-1); + __R30 &= ~(out); // Clear the GPIO pin + __delay_cycles(zeroCyclesOff-14); + } + } + } + // Send Reset + __R30 &= ~out; // Clear the GPIO pin + __delay_cycles(resetCycles); + + // Wait + __delay_cycles(SPEED); + } + + } + } + } +} diff --git a/pru-cookbook/05blocks/code/neo_setup.sh b/pru-cookbook/05blocks/code/neo_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..8f98cbb2a9ae5f915cafc47af006b958fffe44f7 --- /dev/null +++ b/pru-cookbook/05blocks/code/neo_setup.sh @@ -0,0 +1,28 @@ +#!/bin/bash +# +export TARGET=neo1.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_29 " +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_33" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done diff --git a/pru-cookbook/05blocks/code/pwm-test.c b/pru-cookbook/05blocks/code/pwm-test.c new file mode 100644 index 0000000000000000000000000000000000000000..3bff5f71b6598b3f6ee1fd0fc04bdac16793cda8 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm-test.c @@ -0,0 +1,74 @@ +/* + * + * pwm tester + * The on cycle and off cycles are stored in each PRU's Data memory + * + */ + +#include <stdio.h> +#include <fcntl.h> +#include <sys/mman.h> + +#define MAXCH 4 + +#define PRU_ADDR 0x4A300000 // Start of PRU memory Page 184 am335x TRM +#define PRU_LEN 0x80000 // Length of PRU memory +#define PRU0_DRAM 0x00000 // Offset to DRAM +#define PRU1_DRAM 0x02000 +#define PRU_SHAREDMEM 0x10000 // Offset to shared memory + +unsigned int *pru0DRAM_32int_ptr; // Points to the start of local DRAM +unsigned int *pru1DRAM_32int_ptr; // Points to the start of local DRAM +unsigned int *prusharedMem_32int_ptr; // Points to the start of the shared memory + +/******************************************************************************* +* int start_pwm_count(int ch, int countOn, int countOff) +* +* Starts a pwm pulse on for countOn and off for countOff to a single channel (ch) +*******************************************************************************/ +int start_pwm_count(int ch, int countOn, int countOff) { + unsigned int *pruDRAM_32int_ptr = pru0DRAM_32int_ptr; + + printf("countOn: %d, countOff: %d, count: %d\n", + countOn, countOff, countOn+countOff); + // write to PRU shared memory + pruDRAM_32int_ptr[2*(ch)+0] = countOn; // On time + pruDRAM_32int_ptr[2*(ch)+1] = countOff; // Off time + return 0; +} + +int main(int argc, char *argv[]) +{ + unsigned int *pru; // Points to start of PRU memory. + int fd; + printf("Servo tester\n"); + + fd = open ("/dev/mem", O_RDWR | O_SYNC); + if (fd == -1) { + printf ("ERROR: could not open /dev/mem.\n\n"); + return 1; + } + pru = mmap (0, PRU_LEN, PROT_READ | PROT_WRITE, MAP_SHARED, fd, PRU_ADDR); + if (pru == MAP_FAILED) { + printf ("ERROR: could not map memory.\n\n"); + return 1; + } + close(fd); + printf ("Using /dev/mem.\n"); + + pru0DRAM_32int_ptr = pru + PRU0_DRAM/4 + 0x200/4; // Points to 0x200 of PRU0 memory + pru1DRAM_32int_ptr = pru + PRU1_DRAM/4 + 0x200/4; // Points to 0x200 of PRU1 memory + prusharedMem_32int_ptr = pru + PRU_SHAREDMEM/4; // Points to start of shared memory + + int i; + for(i=0; i<MAXCH; i++) { + start_pwm_count(i, i+1, 20-(i+1)); + } + + if(munmap(pru, PRU_LEN)) { + printf("munmap failed\n"); + } else { + printf("munmap succeeded\n"); + } +} + diff --git a/pru-cookbook/05blocks/code/pwm1.pru0.c b/pru-cookbook/05blocks/code/pwm1.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..2ea5caa17efd570bad1cbc658d5e2c96557b1b0f --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm1.pru0.c @@ -0,0 +1,22 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_31; // Select which pin to toggle.; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while(1) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(100000000); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(100000000); + } +} diff --git a/pru-cookbook/05blocks/code/pwm1.pru1_1.c b/pru-cookbook/05blocks/code/pwm1.pru1_1.c new file mode 100644 index 0000000000000000000000000000000000000000..1f8b737e678691af3c4ab83cfa2d1d5bc442fe64 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm1.pru1_1.c @@ -0,0 +1,22 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_16; // Select which pin to toggle.; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while(1) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(100000000); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(100000000); + } +} diff --git a/pru-cookbook/05blocks/code/pwm2.pru0.c b/pru-cookbook/05blocks/code/pwm2.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..bfa0d0761411212a8a0f4469ca02dfd08f5a20bd --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm2.pru0.c @@ -0,0 +1,22 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_31; // Select which pin to toggle.; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while (1) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(1); // Delay one cycle to correct for loop time + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(0); + } +} diff --git a/pru-cookbook/05blocks/code/pwm3.pru0.c b/pru-cookbook/05blocks/code/pwm3.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..4b2955a0573171e7e3525a9381e4a4a2acb15e36 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm3.pru0.c @@ -0,0 +1,43 @@ +// This code does MAXCH parallel PWM channels. +// It's period is 3 us +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +#define MAXCH 4 // Maximum number of channels + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; // Number of cycles to stay on + uint32_t off[] = {4, 3, 2, 1}; // Number to stay off + uint32_t onCount[MAXCH]; // Current count + uint32_t offCount[MAXCH]; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + // Initialize the channel counters. + for(ch=0; ch<MAXCH; ch++) { + onCount[ch] = on[ch]; + offCount[ch]= off[ch]; + } + + while (1) { + for(ch=0; ch<MAXCH; ch++) { + if(onCount[ch]) { + onCount[ch]--; + __R30 |= 0x1<<ch; // Set the GPIO pin to 1 + } else if(offCount[ch]) { + offCount[ch]--; + __R30 &= ~(0x1<<ch); // Clear the GPIO pin + } else { + onCount[ch] = on[ch]; + offCount[ch]= off[ch]; + } + } + } +} diff --git a/pru-cookbook/05blocks/code/pwm4.pru0.c b/pru-cookbook/05blocks/code/pwm4.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..ee0f7ec6adf542562b8740ce410226840b01ac50 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm4.pru0.c @@ -0,0 +1,50 @@ +// This code does MAXCH parallel PWM channels. +// It's period is 3 us +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 4 // Maximum number of channels per PRU + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; // Number of cycles to stay on + uint32_t off[] = {4, 3, 2, 1}; // Number to stay off + uint32_t onCount[MAXCH]; // Current count + uint32_t offCount[MAXCH]; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + // Initialize the channel counters. + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on[ch]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch]; // Interleave the on and off values + onCount[ch] = on[ch]; + offCount[ch]= off[ch]; + } + + while (1) { + for(ch=0; ch<MAXCH; ch++) { + if(onCount[ch]) { + onCount[ch]--; + __R30 |= 0x1<<ch; // Set the GPIO pin to 1 + } else if(offCount[ch]) { + offCount[ch]--; + __R30 &= ~(0x1<<ch); // Clear the GPIO pin + } else { + onCount[ch] = pru0_dram[2*ch]; // Read from DRAM0 + offCount[ch]= pru0_dram[2*ch+1]; + } + } + } +} diff --git a/pru-cookbook/05blocks/code/pwm5.pru0.c b/pru-cookbook/05blocks/code/pwm5.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..60a97b817c32ca25fcbd9445ec6d612151ed21c6 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm5.pru0.c @@ -0,0 +1,53 @@ +// This code does MAXCH parallel PWM channels. +// It's period is 510ns. +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 4 // Maximum number of channels per PRU + +#define update(ch) \ + if(onCount[ch]) { \ + onCount[ch]--; \ + __R30 |= 0x1<<ch; \ + } else if(offCount[ch]) { \ + offCount[ch]--; \ + __R30 &= ~(0x1<<ch); \ + } else { \ + onCount[ch] = pru0_dram[2*ch]; \ + offCount[ch]= pru0_dram[2*ch+1]; \ + } + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; + uint32_t off[] = {4, 3, 2, 1}; + uint32_t onCount[MAXCH], offCount[MAXCH]; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +#pragma UNROLL(MAXCH) + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on[ch]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch]; // Interleave the on and off values + onCount[ch] = on[ch]; + offCount[ch]= off[ch]; + } + + while (1) { + update(0) + update(1) + update(2) + update(3) + } +} diff --git a/pru-cookbook/05blocks/code/pwm6.pru0.c b/pru-cookbook/05blocks/code/pwm6.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..6df321f5f0d7094b6ec93e57cbf65a62994094a1 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm6.pru0.c @@ -0,0 +1,56 @@ +// This code does MAXCH parallel PWM channels. +// All channels start at the same time. It's period is 510ns +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 4 // Maximum number of channels per PRU + +#define update(ch) \ + if(onCount[ch]) { \ + onCount[ch]--; \ + Rtmp |= 0x1<<ch; \ + } else if(offCount[ch]) { \ + offCount[ch]--; \ + Rtmp &= ~(0x1<<ch); \ + } else { \ + onCount[ch] = pru0_dram[2*ch]; \ + offCount[ch]= pru0_dram[2*ch+1]; \ + } + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; + uint32_t off[] = {4, 3, 2, 1}; + uint32_t onCount[MAXCH], offCount[MAXCH]; + register uint32_t Rtmp; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +#pragma UNROLL(MAXCH) + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on[ch]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch]; // Interleave the on and off values + onCount[ch] = on[ch]; + offCount[ch]= off[ch]; + } + Rtmp = __R30; + + while (1) { + update(0) + update(1) + update(2) + update(3) + __R30 = Rtmp; + } +} diff --git a/pru-cookbook/05blocks/code/pwm7-test.c b/pru-cookbook/05blocks/code/pwm7-test.c new file mode 100644 index 0000000000000000000000000000000000000000..cbb6769819050b95dec2fd402db0f69a4beaa752 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm7-test.c @@ -0,0 +1,83 @@ +/* + * + * pwm tester + * (c) Copyright 2016 + * Mark A. Yoder, 20-July-2016 + * The channels 0-11 are on PRU1 and channels 12-17 are on PRU0 + * The period and duty cycle values are stored in each PRU's Data memory + * The enable bits are stored in the shared memory + * + */ + +#include <stdio.h> +#include <fcntl.h> +#include <sys/mman.h> + +#define MAXCH 2 + +#define PRU_ADDR 0x4A300000 // Start of PRU memory Page 184 am335x TRM +#define PRU_LEN 0x80000 // Length of PRU memory +#define PRU0_DRAM 0x00000 // Offset to DRAM +#define PRU1_DRAM 0x02000 +#define PRU_SHAREDMEM 0x10000 // Offset to shared memory + +unsigned int *pru0DRAM_32int_ptr; // Points to the start of local DRAM +unsigned int *pru1DRAM_32int_ptr; // Points to the start of local DRAM +unsigned int *prusharedMem_32int_ptr; // Points to the start of the shared memory + +/******************************************************************************* +* int start_pwm_count(int ch, int countOn, int countOff) +* +* Starts a pwm pulse on for countOn and off for countOff to a single channel (ch) +*******************************************************************************/ +int start_pwm_count(int ch, int countOn, int countOff, unsigned int *ptr) { + unsigned int *pruDRAM_32int_ptr = ptr; + + printf("countOn: %d, countOff: %d, count: %d\n", + countOn, countOff, countOn+countOff); + // write to PRU shared memory + pruDRAM_32int_ptr[2*(ch)+0] = countOn; // On time + pruDRAM_32int_ptr[2*(ch)+1] = countOff; // Off time + return 0; +} + +int main(int argc, char *argv[]) +{ + unsigned int *pru; // Points to start of PRU memory. + int fd; + printf("Servo tester\n"); + + fd = open ("/dev/mem", O_RDWR | O_SYNC); + if (fd == -1) { + printf ("ERROR: could not open /dev/mem.\n\n"); + return 1; + } + pru = mmap (0, PRU_LEN, PROT_READ | PROT_WRITE, MAP_SHARED, fd, PRU_ADDR); + if (pru == MAP_FAILED) { + printf ("ERROR: could not map memory.\n\n"); + return 1; + } + close(fd); + printf ("Using /dev/mem.\n"); + + pru0DRAM_32int_ptr = pru + PRU0_DRAM/4 + 0x200/4; // Points to 0x200 of PRU0 memory + pru1DRAM_32int_ptr = pru + PRU1_DRAM/4 + 0x200/4; // Points to 0x200 of PRU1 memory + prusharedMem_32int_ptr = pru + PRU_SHAREDMEM/4; // Points to start of shared memory + + + int on[] = {1, 2, 3, 4}; + int off[] = {4, 3, 2, 1}; + + int ch; + for(ch=0; ch<MAXCH; ch++) { + start_pwm_count(ch, on[ch], off[ch], pru0DRAM_32int_ptr); + start_pwm_count(ch, on[ch+MAXCH], off[ch+MAXCH], pru1DRAM_32int_ptr); + } + + if(munmap(pru, PRU_LEN)) { + printf("munmap failed\n"); + } else { + printf("munmap succeeded\n"); + } +} + diff --git a/pru-cookbook/05blocks/code/pwm7.pru0.c b/pru-cookbook/05blocks/code/pwm7.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..b95c5caab89d3614e9dfa964d38c162921f67a4b --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm7.pru0.c @@ -0,0 +1,57 @@ +// This code does MAXCH parallel PWM channels on both PRU 0 and PRU 1 +// All channels start at the same time. But the PRU 1 ch have a difference period +// It's period is 370ns +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +#define PRUNUM 0 + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 2 // Maximum number of channels per PRU + +#define update(ch) \ + if(onCount[ch]) { \ + onCount[ch]--; \ + Rtmp |= 0x1<<ch; \ + } else if(offCount[ch]) { \ + offCount[ch]--; \ + Rtmp &= ~(0x1<<ch); \ + } else { \ + onCount[ch] = pru0_dram[2*ch]; \ + offCount[ch]= pru0_dram[2*ch+1]; \ + } + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; + uint32_t off[] = {4, 3, 2, 1}; + uint32_t onCount[MAXCH], offCount[MAXCH]; + register uint32_t Rtmp; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +#pragma UNROLL(MAXCH) + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on [ch+PRUNUM*MAXCH]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch+PRUNUM*MAXCH]; // Interleave the on and off values + onCount[ch] = on [ch+PRUNUM*MAXCH]; + offCount[ch]= off[ch+PRUNUM*MAXCH]; + } + Rtmp = __R30; + + while (1) { + update(0) + update(1) + __R30 = Rtmp; + } +} diff --git a/pru-cookbook/05blocks/code/pwm7.pru1.c b/pru-cookbook/05blocks/code/pwm7.pru1.c new file mode 100644 index 0000000000000000000000000000000000000000..50d4788b8c4d07848a0f0ff651fc78a7545ff67f --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm7.pru1.c @@ -0,0 +1,57 @@ +// This code does MAXCH parallel PWM channels on both PRU 0 and PRU 1 +// All channels start at the same time. But the PRU 1 ch have a difference period +// It's period is 370ns +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +#define PRUNUM 1 + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 2 // Maximum number of channels per PRU + +#define update(ch) \ + if(onCount[ch]) { \ + onCount[ch]--; \ + Rtmp |= 0x1<<ch; \ + } else if(offCount[ch]) { \ + offCount[ch]--; \ + Rtmp &= ~(0x1<<ch); \ + } else { \ + onCount[ch] = pru0_dram[2*ch]; \ + offCount[ch]= pru0_dram[2*ch+1]; \ + } + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; + uint32_t off[] = {4, 3, 2, 1}; + uint32_t onCount[MAXCH], offCount[MAXCH]; + register uint32_t Rtmp; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +#pragma UNROLL(MAXCH) + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on [ch+PRUNUM*MAXCH]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch+PRUNUM*MAXCH]; // Interleave the on and off values + onCount[ch] = on [ch+PRUNUM*MAXCH]; + offCount[ch]= off[ch+PRUNUM*MAXCH]; + } + Rtmp = __R30; + + while (1) { + update(0) + update(1) + __R30 = Rtmp; + } +} diff --git a/pru-cookbook/05blocks/code/pwm7_setup.sh b/pru-cookbook/05blocks/code/pwm7_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..aae4c778542c5209a799411b393b2d5b2130e76e --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm7_setup.sh @@ -0,0 +1,28 @@ +#!/bin/bash +# +export TARGET=pwm7.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_31 P9_29 P8_45 P8_46" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_36 P1_33" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done diff --git a/pru-cookbook/05blocks/code/pwm8.pru0.c b/pru-cookbook/05blocks/code/pwm8.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..c566573f6f681f7642e61d5c159c8558d3895d33 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm8.pru0.c @@ -0,0 +1,78 @@ +// This code does MAXCH parallel PWM channels on both PRU 0 and PRU 1 +// All channels start at the same time. +// It's period is 430ns +#include <stdint.h> +#include <pru_cfg.h> +#include <pru_intc.h> +#include <pru_ctrl.h> +#include "resource_table_empty.h" + +#define PRUNUM 0 + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 2 // Maximum number of channels per PRU + +#define update(ch) \ + if(onCount[ch]) { \ + onCount[ch]--; \ + Rtmp |= 0x1<<ch; \ + } else if(offCount[ch]) { \ + offCount[ch]--; \ + Rtmp &= ~(0x1<<ch); \ + } else { \ + onCount[ch] = pru0_dram[2*ch]; \ + offCount[ch]= pru0_dram[2*ch+1]; \ + } + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +// Initialize interupts so the PRUs can be syncronized. +// PRU1 is started first and then waits for PRU0 +// PRU0 is then started and tells PRU1 when to start going +void configIntc(void) { + __R31 = 0x00000000; // Clear any pending PRU-generated events + CT_INTC.CMR4_bit.CH_MAP_16 = 1; // Map event 16 to channel 1 + CT_INTC.HMR0_bit.HINT_MAP_1 = 1; // Map channel 1 to host 1 + CT_INTC.SICR = 16; // Ensure event 16 is cleared + CT_INTC.EISR = 16; // Enable event 16 + CT_INTC.HIEISR |= (1 << 0); // Enable Host interrupt 1 + CT_INTC.GER = 1; // Globally enable host interrupts +} + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; + uint32_t off[] = {4, 3, 2, 1}; + uint32_t onCount[MAXCH], offCount[MAXCH]; + register uint32_t Rtmp; + + CT_CFG.GPCFG0 = 0x0000; // Configure GPI and GPO as Mode 0 (Direct Connect) + configIntc(); // Configure INTC + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +#pragma UNROLL(MAXCH) + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on [ch+PRUNUM*MAXCH]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch+PRUNUM*MAXCH]; // Interleave the on and off values + onCount[ch] = on [ch+PRUNUM*MAXCH]; + offCount[ch]= off[ch+PRUNUM*MAXCH]; + } + Rtmp = __R30; + + while (1) { + __R30 = Rtmp; + update(0) + update(1) +#define PRU0_PRU1_EVT 16 + __R31 = (PRU0_PRU1_EVT-16) | (0x1<<5); //Tell PRU 1 to start + __delay_cycles(1); + } +} diff --git a/pru-cookbook/05blocks/code/pwm8.pru1.c b/pru-cookbook/05blocks/code/pwm8.pru1.c new file mode 100644 index 0000000000000000000000000000000000000000..cf5c241c99addc0d8b958724554e078f9a53412c --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm8.pru1.c @@ -0,0 +1,66 @@ +// This code does MAXCH parallel PWM channels on both PRU 0 and PRU 1 +// All channels start at the same time. +// It's period is 430ns +#include <stdint.h> +#include <pru_cfg.h> +#include <pru_intc.h> +#include <pru_ctrl.h> +#include "resource_table_empty.h" + +#define PRUNUM 1 + +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 byte of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *pru0_dram = (unsigned int *) (PRU0_DRAM + 0x200); + +#define MAXCH 2 // Maximum number of channels per PRU + +#define update(ch) \ + if(onCount[ch]) { \ + onCount[ch]--; \ + Rtmp |= 0x1<<ch; \ + } else if(offCount[ch]) { \ + offCount[ch]--; \ + Rtmp &= ~(0x1<<ch); \ + } else { \ + onCount[ch] = pru0_dram[2*ch]; \ + offCount[ch]= pru0_dram[2*ch+1]; \ + } + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +// Initialize interupts so the PRUs can be syncronized. +// PRU1 is started first and then waits for PRU0 +// PRU0 is then started and tells PRU1 when to start going + +void main(void) +{ + uint32_t ch; + uint32_t on[] = {1, 2, 3, 4}; + uint32_t off[] = {4, 3, 2, 1}; + uint32_t onCount[MAXCH], offCount[MAXCH]; + register uint32_t Rtmp; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +#pragma UNROLL(MAXCH) + for(ch=0; ch<MAXCH; ch++) { + pru0_dram[2*ch ] = on [ch+PRUNUM*MAXCH]; // Copy to DRAM0 so the ARM can change it + pru0_dram[2*ch+1] = off[ch+PRUNUM*MAXCH]; // Interleave the on and off values + onCount[ch] = on [ch+PRUNUM*MAXCH]; + offCount[ch]= off[ch+PRUNUM*MAXCH]; + } + Rtmp = __R30; + + while (1) { + while((__R31 & (0x1<<31))==0) { // Wait for PRU 0 + } + CT_INTC.SICR = 16; // Clear event 16 + __R30 = Rtmp; + update(0) + update(1) + } +} diff --git a/pru-cookbook/05blocks/code/pwm_setup.sh b/pru-cookbook/05blocks/code/pwm_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..b69ed1297ab4a73d309a7f46253c3b9b5f97add1 --- /dev/null +++ b/pru-cookbook/05blocks/code/pwm_setup.sh @@ -0,0 +1,28 @@ +#!/bin/bash +# +export TARGET=pwm1.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_31 P9_29 P9_30 P9_28" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_36 P1_33 P2_32 P2_30" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done diff --git a/pru-cookbook/05blocks/code/resource_table_empty.h b/pru-cookbook/05blocks/code/resource_table_empty.h new file mode 100644 index 0000000000000000000000000000000000000000..07e97d9b383c9d937d4b55e705024d6c28dc748a --- /dev/null +++ b/pru-cookbook/05blocks/code/resource_table_empty.h @@ -0,0 +1,39 @@ +/* + * ======== resource_table_empty.h ======== + * + * Define the resource table entries for all PRU cores. This will be + * incorporated into corresponding base images, and used by the remoteproc + * on the host-side to allocated/reserve resources. Note the remoteproc + * driver requires that all PRU firmware be built with a resource table. + * + * This file contains an empty resource table. It can be used either as: + * + * 1) A template, or + * 2) As-is if a PRU application does not need to configure PRU_INTC + * or interact with the rpmsg driver + * + */ + +#ifndef _RSC_TABLE_PRU_H_ +#define _RSC_TABLE_PRU_H_ + +#include <stddef.h> +#include <rsc_types.h> + +struct my_resource_table { + struct resource_table base; + + uint32_t offset[1]; /* Should match 'num' in actual definition */ +}; + +#pragma DATA_SECTION(pru_remoteproc_ResourceTable, ".resource_table") +#pragma RETAIN(pru_remoteproc_ResourceTable) +struct my_resource_table pru_remoteproc_ResourceTable = { + 1, /* we're the first version that implements this */ + 0, /* number of entries in the table */ + 0, 0, /* reserved, must be zero */ + 0, /* offset[0] */ +}; + +#endif /* _RSC_TABLE_PRU_H_ */ + diff --git a/pru-cookbook/05blocks/code/rgb1.pru0.c b/pru-cookbook/05blocks/code/rgb1.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..9936d627336863daf9c4c1d5e92f70afd602b3a9 --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb1.pru0.c @@ -0,0 +1,80 @@ +// This code drives the RGB LED Matrix on the 1st Connector +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" +#include "rgb_pocket.h" + +#define DELAY 10 // Number of cycles (5ns each) to wait after a write + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + // Set up the pointers to each of the GPIO ports + uint32_t *gpio[] = { + (uint32_t *) GPIO0, + (uint32_t *) GPIO1, + (uint32_t *) GPIO2, + (uint32_t *) GPIO3 + }; + + uint32_t i, row; + + while(1) { + for(row=0; row<16; row++) { + // Set the row address + // Here we take advantage of the select bits (LA,LB,LC,LD) + // being sequential in the R30 register (bits 2,3,4,5) + // We shift row over so it lines up with the select bits + // Oring (|=) with R30 sets bits to 1 and + // Anding (&=) clears bits to 0, the 0xffc mask makes sure the + // other bits aren't changed. + __R30 |= row<<pru_sel0; + __R30 &= (row<<pru_sel0)|0xffc3; + + for(i=0; i<64; i++) { + // Top row white + // Combining these to one write works because they are all in + // the same gpio port + gpio[r11_gpio][GPIO_SETDATAOUT] = r11_pin | g11_pin | b11_pin; + __delay_cycles(DELAY);; + + // Bottom row red + gpio[r12_gpio][GPIO_SETDATAOUT] = r12_pin; + __delay_cycles(DELAY); + gpio[r12_gpio][GPIO_CLEARDATAOUT] = g12_pin | b12_pin; + __delay_cycles(DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + + // Top row black + gpio[r11_gpio][GPIO_CLEARDATAOUT] = r11_pin | g11_pin | b11_pin; + __delay_cycles(DELAY); + + // Bottom row green + gpio[r12_gpio][GPIO_CLEARDATAOUT] = r12_pin | b12_pin; + __delay_cycles(DELAY); + gpio[r12_gpio][GPIO_SETDATAOUT] = g12_pin; + __delay_cycles(DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + } + __R30 |= pru_oe; // Disable display + __delay_cycles(DELAY); + __R30 |= pru_latch; // Toggle latch + __delay_cycles(DELAY); + __R30 &= ~pru_latch; + __delay_cycles(DELAY); + __R30 &= ~pru_oe; // Enable display + __delay_cycles(DELAY); + } + } +} diff --git a/pru-cookbook/05blocks/code/rgb2.pru0.c b/pru-cookbook/05blocks/code/rgb2.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..eb43b357bb3cdb92f6853a0d46600b1e53a7b6ac --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb2.pru0.c @@ -0,0 +1,81 @@ +// This code drives the RGB LED Matrix on J1 connector +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" +#include "rgb_pocket.h" + +#define DELAY 100 + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + // Set up the pointers to each of the GPIO ports + uint32_t *gpio[] = { + (uint32_t *) GPIO0, + (uint32_t *) GPIO1, + (uint32_t *) GPIO2, + (uint32_t *) GPIO3 + }; + + uint32_t i, row; + + while(1) { + for(row=0; row<16; row++) { + if(row&(0x1<<0)) __R30|=(0x1<<pru_sel0); else __R30&=~(0x1<<pru_sel0); + __delay_cycles(DELAY); + if(row&(0x1<<1)) __R30|=(0x1<<pru_sel1); else __R30&=~(0x1<<pru_sel1); + __delay_cycles(DELAY); + if(row&(0x1<<2)) __R30|=(0x1<<pru_sel2); else __R30&=~(0x1<<pru_sel2); + __delay_cycles(DELAY); + if(row&(0x1<<3)) __R30|=(0x1<<pru_sel3); else __R30&=~(0x1<<pru_sel3); + __delay_cycles(DELAY); + + for(i=0; i<64; i++) { + // red + gpio[r11_gpio][GPIO_SETDATAOUT] = r11_pin; + __delay_cycles(DELAY); + gpio[r11_gpio][GPIO_CLEARDATAOUT] = g11_pin | b11_pin; + __delay_cycles(DELAY); + + // green + gpio[r11_gpio][GPIO_CLEARDATAOUT] = r12_pin | b12_pin; + __delay_cycles(DELAY); + gpio[r11_gpio][GPIO_SETDATAOUT] = g12_pin; + __delay_cycles(DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + + // blue + gpio[r11_gpio][GPIO_CLEARDATAOUT] = r11_pin | g11_pin; + __delay_cycles(DELAY); + gpio[r11_gpio][GPIO_SETDATAOUT] = b11_pin; + __delay_cycles(DELAY); + + //blue + gpio[r11_gpio][GPIO_CLEARDATAOUT] = r12_pin | g12_pin; + __delay_cycles(DELAY); + gpio[r11_gpio][GPIO_SETDATAOUT] = b12_pin; + __delay_cycles(DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + } + __R30 |= pru_oe; // Disable display + __delay_cycles(DELAY); + __R30 |= pru_latch; // Toggle latch + __delay_cycles(DELAY); + __R30 &= ~pru_latch; + __delay_cycles(DELAY); + __R30 &= ~pru_oe; // Enable display + __delay_cycles(DELAY); + } + } +} diff --git a/pru-cookbook/05blocks/code/rgb3.pru0.c b/pru-cookbook/05blocks/code/rgb3.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..b14ad864c4de0c621561f1cb031aae0af5367dd4 --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb3.pru0.c @@ -0,0 +1,114 @@ +// This code drives the RGB LED Matrix on the 1st and 2nd Connectors +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" +#include "rgb_pocket.h" + +#define DELAY 10 // Number of cycles (5ns each) to wait after a write + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + // Set up the pointers to each of the GPIO ports + uint32_t *gpio[] = { + (uint32_t *) GPIO0, + (uint32_t *) GPIO1, + (uint32_t *) GPIO2, + (uint32_t *) GPIO3 + }; + + uint32_t i, row; + + while(1) { + for(row=0; row<16; row++) { + // Set the row address + // Here we take advantage of the select bits (LA,LB,LC,LD) + // being sequential in the R30 register (bits 2,3,4,5) + // We shift row over so it lines up with the select bits + // Oring (|=) with R30 sets bits to 1 and + // Anding (&=) clears bits to 0, the 0xffc mask makes sure the + // other bits aren't changed. + __R30 |= row<<pru_sel0; + __R30 &= (row<<pru_sel0)|0xffc3; + + for(i=0; i<64; i++) { + // Panel 1 Upper + // Top row white + // Combining these to one write works because they are all in + // the same gpio port + gpio[r11_gpio][GPIO_SETDATAOUT] = r11_pin | g11_pin | b11_pin; + __delay_cycles(DELAY); + + // Bottom row red + gpio[r12_gpio][GPIO_SETDATAOUT] = r12_pin; + // __delay_cycles(DELAY); + gpio[g12_gpio][GPIO_CLEARDATAOUT] = g12_pin | b12_pin; + __delay_cycles(DELAY); + + // Panel 2 Upper + // Top row blue + // Combining these to one write works because they are all in + // the same gpio port except b12 + gpio[r21_gpio][GPIO_CLEARDATAOUT] = r21_pin | g21_pin; + // __delay_cycles(DELAY); + gpio[b21_gpio][GPIO_SETDATAOUT] = b21_pin; + __delay_cycles(DELAY); + + // Bottom row red + gpio[r22_gpio][GPIO_SETDATAOUT] = r22_pin; + __delay_cycles(DELAY); + gpio[g22_gpio][GPIO_CLEARDATAOUT] = g22_pin; + __delay_cycles(DELAY); + gpio[b22_gpio][GPIO_CLEARDATAOUT] = b22_pin; + __delay_cycles(DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + + // Panel 1 Lower + // Top row black + gpio[r11_gpio][GPIO_CLEARDATAOUT] = r11_pin | g11_pin | b11_pin; + __delay_cycles(DELAY); + + // Bottom row green + gpio[r12_gpio][GPIO_CLEARDATAOUT] = r12_pin | b12_pin; + __delay_cycles(DELAY); + gpio[g12_gpio][GPIO_SETDATAOUT] = g12_pin; + __delay_cycles(DELAY); + + // Panel 2 Lower + // Top row reg+green = yellow + gpio[r21_gpio][GPIO_SETDATAOUT] = r21_pin | g21_pin; + __delay_cycles(DELAY); + gpio[b21_gpio][GPIO_CLEARDATAOUT] = b21_pin; + __delay_cycles(DELAY); + + // Bottom row green + gpio[r22_gpio][GPIO_CLEARDATAOUT] = r22_pin; + __delay_cycles(DELAY); + gpio[b22_gpio][GPIO_CLEARDATAOUT] = b22_pin; + __delay_cycles(DELAY); + gpio[g22_gpio][GPIO_SETDATAOUT] = g22_pin; + __delay_cycles(2*DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + } + __R30 |= pru_oe; // Disable display + __delay_cycles(DELAY); + __R30 |= pru_latch; // Toggle latch + __delay_cycles(DELAY); + __R30 &= ~pru_latch; + __delay_cycles(DELAY); + __R30 &= ~pru_oe; // Enable display + __delay_cycles(DELAY); + } + } +} diff --git a/pru-cookbook/05blocks/code/rgb4.pru0.c b/pru-cookbook/05blocks/code/rgb4.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..6d06f1210e18bb3bbf74bf20b9c0b0c876c365e0 --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb4.pru0.c @@ -0,0 +1,84 @@ +// This code drives the RGB LED Matrix on the 2nd Connector +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" +#include "rgb_pocket.h" + +#define DELAY 10 // Number of cycles (5ns each) to wait after a write + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + // Set up the pointers to each of the GPIO ports + uint32_t *gpio[] = { + (uint32_t *) GPIO0, + (uint32_t *) GPIO1, + (uint32_t *) GPIO2, + (uint32_t *) GPIO3 + }; + + uint32_t i, row; + + while(1) { + for(row=0; row<16; row++) { + // Set the row address + // Here we take advantage of the select bits (LA,LB,LC,LD) + // being sequential in the R30 register (bits 2,3,4,5) + // We shift row over so it lines up with the select bits + // Oring (|=) with R30 sets bits to 1 and + // Anding (&=) clears bits to 0, the 0xffc mask makes sure the + // other bits aren't changed. + __R30 |= row<<pru_sel0; + __R30 &= (row<<pru_sel0)|0xffc3; + + for(i=0; i<64; i++) { + // Top row white + // Combining these to one write works because they are all in + // the same gpio port except b12 + gpio[r21_gpio][GPIO_SETDATAOUT] = r21_pin | g21_pin; + __delay_cycles(DELAY); + gpio[b21_gpio][GPIO_SETDATAOUT] = b21_pin; + __delay_cycles(DELAY); + + // Bottom row red + gpio[r22_gpio][GPIO_SETDATAOUT] = r22_pin; + __delay_cycles(DELAY); + gpio[g22_gpio][GPIO_CLEARDATAOUT] = g22_pin | b22_pin; + __delay_cycles(2*DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + + // Top row black + gpio[r21_gpio][GPIO_CLEARDATAOUT] = r21_pin | g21_pin; + __delay_cycles(DELAY); + gpio[b21_gpio][GPIO_CLEARDATAOUT] = b21_pin; + __delay_cycles(DELAY); + + // Bottom row green + gpio[r22_gpio][GPIO_CLEARDATAOUT] = r22_pin | b22_pin; + __delay_cycles(DELAY); + gpio[g22_gpio][GPIO_SETDATAOUT] = g22_pin; + __delay_cycles(2*DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + } + __R30 |= pru_oe; // Disable display + __delay_cycles(DELAY); + __R30 |= pru_latch; // Toggle latch + __delay_cycles(DELAY); + __R30 &= ~pru_latch; + __delay_cycles(DELAY); + __R30 &= ~pru_oe; // Enable display + __delay_cycles(DELAY); + } + } +} diff --git a/pru-cookbook/05blocks/code/rgb_black.h b/pru-cookbook/05blocks/code/rgb_black.h new file mode 100644 index 0000000000000000000000000000000000000000..be7cf4ba8ab0da67ee7ce7301fe49a90bfad539e --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb_black.h @@ -0,0 +1,31 @@ +//Black - Not tested +// These are from https://github.com/FalconChristmas/fpp/blob/master/src/pru/PocketScrollerV1.hp +// _gpio tells which gpio port and _pin tells which bit in the port +// The first 1 in r11 is for the J1 connector +// See the githuub file for the other connectors + +#define r11_gpio 2 +#define r11_pin 9 +#define g11_gpio 2 +#define g11_pin 11 +#define b11_gpio 2 +#define b11_pin 10 + +#define r12_gpio 2 +#define r12_pin 12 +#define g12_gpio 2 +#define g12_pin 22 +#define b12_gpio 2 +#define b12_pin 23 + +#define pru_latch 1 // These are the bit positions in R30 +#define pru_oe 0 +#define pru_clock 2 + +// Control pins are all in GPIO2 +// The pocket has these on R0, the code needs to be changed for this work work +#define gpio_sel0 13 /* must be sequential with sel1 and sel2 */ +#define gpio_sel1 14 +#define gpio_sel2 15 +#define gpio_sel3 16 +#define gpio_sel4 17 diff --git a/pru-cookbook/05blocks/code/rgb_pocket.h b/pru-cookbook/05blocks/code/rgb_pocket.h new file mode 100644 index 0000000000000000000000000000000000000000..45a91118038898605939bc4493a3337cb4fd86a0 --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb_pocket.h @@ -0,0 +1,45 @@ +// Pocket +// These are from https://github.com/FalconChristmas/fpp/blob/master/src/pru/PocketScrollerV1.hp +// _gpio tells which gpio port and _pin tells which bit in the port +// The first 1 in r11 is for the J1 connector +// See the githuub file for the other connectors + +// J1 +#define r11_gpio 1 +#define r11_pin (0x1<<20) +#define g11_gpio 1 +#define g11_pin (0x1<<28) +#define b11_gpio 1 +#define b11_pin (0x1<<25) + +#define r12_gpio 1 +#define r12_pin (0x1<<26) +#define g12_gpio 1 +#define g12_pin (0x1<<27) +#define b12_gpio 1 +#define b12_pin (0x1<<18) + +// J2 +#define r21_gpio 1 +#define r21_pin (0x1<<8) +#define g21_gpio 1 +#define g21_pin (0x1<<9) +#define b21_gpio 0 +#define b21_pin (0x1<<30) + +#define r22_gpio 1 +#define r22_pin (0x1<<12) +#define g22_gpio 1 +#define g22_pin (0x1<<14) +#define b22_gpio 1 +#define b22_pin (0x1<<15) + +// All connectors +#define pru_latch (0x1<<0) // These are the bit positions in R30 +#define pru_oe (0x1<<7) +#define pru_clock (0x1<<1) + +#define pru_sel0 2 // These are called LA, LB, LC and LD in the python code +#define pru_sel1 3 // Also bit positions +#define pru_sel2 4 +#define pru_sel3 5 diff --git a/pru-cookbook/05blocks/code/rgb_python.py b/pru-cookbook/05blocks/code/rgb_python.py new file mode 100755 index 0000000000000000000000000000000000000000..25e353343f3c8f829a60376f4c3121e8ada1bbd1 --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb_python.py @@ -0,0 +1,78 @@ +#!/usr/bin/env python3 +import Adafruit_BBIO.GPIO as GPIO + +# Define which functions are connect to which pins +OE="P1_29" # Output Enable, active low +LAT="P1_36" # Latch, toggle after clocking in a row of pixels +CLK="P1_33" # Clock, toggle after each pixel + +# Input data pins +R1="P2_10" # R1, G1, B1 are for the top rows (1-16) of pixels +G1="P2_8" +B1="P2_6" + +R2="P2_4" # R2, G2, B2 are for the bottom rows (17-32) of pixels +G2="P2_2" +B2="P2_1" + +LA="P2_32" # Address lines for which row (1-16 or 17-32) to update +LB="P2_30" +LC="P1_31" +LD="P2_34" + +# Set everything as output ports +GPIO.setup(OE, GPIO.OUT) +GPIO.setup(LAT, GPIO.OUT) +GPIO.setup(CLK, GPIO.OUT) + +GPIO.setup(R1, GPIO.OUT) +GPIO.setup(G1, GPIO.OUT) +GPIO.setup(B1, GPIO.OUT) +GPIO.setup(R2, GPIO.OUT) +GPIO.setup(G2, GPIO.OUT) +GPIO.setup(B2, GPIO.OUT) + +GPIO.setup(LA, GPIO.OUT) +GPIO.setup(LB, GPIO.OUT) +GPIO.setup(LC, GPIO.OUT) +GPIO.setup(LD, GPIO.OUT) + +GPIO.output(OE, 0) # Enable the display +GPIO.output(LAT, 0) # Set latch to low + +while True: + for bank in range(64): + GPIO.output(LA, bank>>0&0x1) # Select rows + GPIO.output(LB, bank>>1&0x1) + GPIO.output(LC, bank>>2&0x1) + GPIO.output(LD, bank>>3&0x1) + + # Shift the colors out. Here we only have four different + # colors to keep things simple. + for i in range(16): + GPIO.output(R1, 1) # Top row, white + GPIO.output(G1, 1) + GPIO.output(B1, 1) + + GPIO.output(R2, 1) # Bottom row, red + GPIO.output(G2, 0) + GPIO.output(B2, 0) + + GPIO.output(CLK, 0) # Toggle clock + GPIO.output(CLK, 1) + + GPIO.output(R1, 0) # Top row, black + GPIO.output(G1, 0) + GPIO.output(B1, 0) + + GPIO.output(R2, 0) # Bottom row, green + GPIO.output(G2, 1) + GPIO.output(B2, 0) + + GPIO.output(CLK, 0) # Toggle clock + GPIO.output(CLK, 1) + + GPIO.output(OE, 1) # Disable display while updating + GPIO.output(LAT, 1) # Toggle latch + GPIO.output(LAT, 0) + GPIO.output(OE, 0) # Enable display diff --git a/pru-cookbook/05blocks/code/rgb_python_setup.sh b/pru-cookbook/05blocks/code/rgb_python_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..721067461771c4d0426a0646f28f748e384ec22d --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb_python_setup.sh @@ -0,0 +1,41 @@ +#!/bin/bash +# Setup for 64x32 RGB Matrix +export TARGET=rgb1.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + prupins="P2_32 P1_31 P1_33 P1_29 P2_30 P2_34 P1_36" + gpiopins="P2_10 P2_06 P2_04 P2_01 P2_08 P2_02" + # Uncomment for J2 + # gpiopins="$gpiopins P2_27 P2_25 P2_05 P2_24 P2_22 P2_18" +else + echo " Not Found" + pins="" +fi + +for pin in $prupins +do + echo $pin + # config-pin $pin pruout + config-pin $pin gpio + config-pin $pin out + config-pin -q $pin +done + +for pin in $gpiopins +do + echo $pin + config-pin $pin gpio + config-pin $pin out + config-pin -q $pin +done diff --git a/pru-cookbook/05blocks/code/rgb_setup.sh b/pru-cookbook/05blocks/code/rgb_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..fc5451444ad10b848f7ec42bb56f4b8f0ffd9195 --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb_setup.sh @@ -0,0 +1,41 @@ +#!/bin/bash +# Setup for 64x32 RGB Matrix +export TARGET=rgb1.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + prupins="P2_32 P1_31 P1_33 P1_29 P2_30 P2_34 P1_36" + gpiopins="P2_10 P2_06 P2_04 P2_01 P2_08 P2_02" + # Uncomment for J2 + # gpiopins="$gpiopins P2_27 P2_25 P2_05 P2_24 P2_22 P2_18" +else + echo " Not Found" + pins="" +fi + +for pin in $prupins +do + echo $pin + config-pin $pin pruout + # config-pin $pin gpio + # config-pin $pin out + config-pin -q $pin +done + +for pin in $gpiopins +do + echo $pin + config-pin $pin gpio + config-pin $pin out + config-pin -q $pin +done diff --git a/pru-cookbook/05blocks/code/rgb_white.pru0.c b/pru-cookbook/05blocks/code/rgb_white.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..e62d98e000c746e37a11c7a0402b13fb5f92b2df --- /dev/null +++ b/pru-cookbook/05blocks/code/rgb_white.pru0.c @@ -0,0 +1,64 @@ +// This code drives the RGB LED Matrix +// Turns all the LEDs on +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" +#include "rgb_pocket.h" + +#define DELAY 10 // Number of cycles (5ns each) to wait after a write + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + // Set up the pointers to each of the GPIO ports + uint32_t *gpio[] = { + (uint32_t *) GPIO0, + (uint32_t *) GPIO1, + (uint32_t *) GPIO2, + (uint32_t *) GPIO3 + }; + + uint32_t i, row; + + while(1) { + for(row=0; row<16; row++) { + // Set the row address + // Here we take advantage of the select bits (LA,LB,LC,LD) + // being sequential in the R30 register (bits 2,3,4,5) + // We shift row over so it lines up with the select bits + // Oring (|=) with R30 sets bits to 1 and + // Anding (&=) clears bits to 0, the 0xffc mask makes sure the + // other bits aren't changed. + __R30 |= row<<pru_sel0; + __R30 &= (row<<pru_sel0)|0xffc3; + + for(i=0; i<64; i++) { + // Top row white + // Combining these to one write works because they are all in + // the same gpio port + gpio[r11_gpio][GPIO_SETDATAOUT] = r11_pin | g11_pin | b11_pin; + __delay_cycles(DELAY);; + + // Bottom row white + gpio[r12_gpio][GPIO_SETDATAOUT] = r12_pin | g12_pin | b12_pin; + __delay_cycles(DELAY); + + __R30 |= pru_clock; // Toggle clock + __delay_cycles(DELAY); + __R30 &= ~pru_clock; + __delay_cycles(DELAY); + } + __R30 |= pru_oe; // Disable display + __delay_cycles(DELAY); + __R30 |= pru_latch; // Toggle latch + __delay_cycles(DELAY); + __R30 &= ~pru_latch; + __delay_cycles(DELAY); + __R30 &= ~pru_oe; // Enable display + __delay_cycles(DELAY); + } + } +} diff --git a/pru-cookbook/05blocks/code/shared.pru0.c b/pru-cookbook/05blocks/code/shared.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..95d87c5de147334e439c18bc6cadc68f6ebac98d --- /dev/null +++ b/pru-cookbook/05blocks/code/shared.pru0.c @@ -0,0 +1,63 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/blobs/master/examples/am335x/PRU_access_const_table/PRU_access_const_table.c +#include <stdint.h> +#include <pru_cfg.h> +#include <pru_ctrl.h> +#include "resource_table_empty.h" + +#define PRU_SRAM __far __attribute__((cregister("PRU_SHAREDMEM", near))) +#define PRU_DMEM0 __far __attribute__((cregister("PRU_DMEM_0_1", near))) +#define PRU_DMEM1 __far __attribute__((cregister("PRU_DMEM_1_0", near))) + +/* NOTE: Allocating shared_x to PRU Shared Memory means that other PRU cores on + * the same subsystem must take care not to allocate data to that memory. + * Users also cannot rely on where in shared memory these variables are placed + * so accessing them from another PRU core or from the ARM is an undefined behavior. + */ +volatile uint32_t shared_0; +PRU_SRAM volatile uint32_t shared_1; +PRU_DMEM0 volatile uint32_t shared_2; +PRU_DMEM1 volatile uint32_t shared_3; +#pragma DATA_SECTION(shared_4, ".bss") +volatile uint32_t shared_4; + +/* NOTE: Here we pick where in memory to store shared_5. The stack and + * heap take up the first 0x200 words, so we must start after that. + * Since we are hardcoding where things are stored we can share + * this between the PRUs and the ARM. +*/ +#define PRU0_DRAM 0x00000 // Offset to DRAM +// Skip the first 0x200 bytes of DRAM since the Makefile allocates +// 0x100 for the STACK and 0x100 for the HEAP. +volatile unsigned int *shared_5 = (unsigned int *) (PRU0_DRAM + 0x200); + + +int main(void) +{ + volatile uint32_t shared_6; + volatile uint32_t shared_7; + /*****************************************************************/ + /* Access PRU peripherals using Constant Table & PRU header file */ + /*****************************************************************/ + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + /*****************************************************************/ + /* Access PRU Shared RAM using Constant Table */ + /*****************************************************************/ + + /* C28 defaults to 0x00000000, we need to set bits 23:8 to 0x0100 in order to have it point to 0x00010000 */ + PRU0_CTRL.CTPPR0_bit.C28_BLK_POINTER = 0x0100; + + shared_0 = 0xfeef; + shared_1 = 0xdeadbeef; + shared_2 = shared_2 + 0xfeed; + shared_3 = 0xdeed; + shared_4 = 0xbeed; + shared_5[0] = 0x1234; + shared_6 = 0x4321; + shared_7 = 0x9876; + + /* Halt PRU core */ + __halt(); +} diff --git a/pru-cookbook/05blocks/code/shared_setup.sh b/pru-cookbook/05blocks/code/shared_setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..eefbcd553ac53d0fb0c289fefc536c5f5eb0a7f7 --- /dev/null +++ b/pru-cookbook/05blocks/code/shared_setup.sh @@ -0,0 +1,28 @@ +#!/bin/bash +# +export TARGET=shared.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_31 P9_29 P9_30 P9_28" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_36 P1_33 P2_32 P2_30" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done diff --git a/pru-cookbook/05blocks/code/sine.map b/pru-cookbook/05blocks/code/sine.map new file mode 100644 index 0000000000000000000000000000000000000000..45872d39e001a44e2a2b38b1ed1bf38c85b8837d --- /dev/null +++ b/pru-cookbook/05blocks/code/sine.map @@ -0,0 +1,338 @@ +****************************************************************************** +PRU Linker Unix v2.1.5 +****************************************************************************** +>> Linked Fri Jun 29 13:58:08 2018 + +OUTPUT FILE NAME: </tmp/pru0-gen/sine1.out> +ENTRY POINT SYMBOL: "_c_int00_noinit_noargs_noexit" address: 00000000 + + +MEMORY CONFIGURATION + + name origin length used unused attr fill +---------------------- -------- --------- -------- -------- ---- -------- +PAGE 0: + PRU_IMEM 00000000 00002000 000018c0 00000740 RWIX + +PAGE 1: + PRU_DMEM_0_1 00000000 00002000 00000154 00001eac RWIX + PRU_DMEM_1_0 00002000 00002000 00000000 00002000 RWIX + +PAGE 2: + PRU_SHAREDMEM 00010000 00003000 00000000 00003000 RWIX + PRU_INTC 00020000 00001504 00000000 00001504 RWIX + PRU_CFG 00026000 00000044 00000044 00000000 RWIX + PRU_UART 00028000 00000038 00000000 00000038 RWIX + PRU_IEP 0002e000 0000031c 00000000 0000031c RWIX + PRU_ECAP 00030000 00000060 00000000 00000060 RWIX + RSVD27 00032000 00000100 00000000 00000100 RWIX + RSVD21 00032400 00000100 00000000 00000100 RWIX + L3OCMC 40000000 00010000 00000000 00010000 RWIX + MCASP0_DMA 46000000 00000100 00000000 00000100 RWIX + UART1 48022000 00000088 00000000 00000088 RWIX + UART2 48024000 00000088 00000000 00000088 RWIX + I2C1 4802a000 000000d8 00000000 000000d8 RWIX + MCSPI0 48030000 000001a4 00000000 000001a4 RWIX + DMTIMER2 48040000 0000005c 00000000 0000005c RWIX + MMCHS0 48060000 00000300 00000000 00000300 RWIX + MBX0 480c8000 00000140 00000000 00000140 RWIX + SPINLOCK 480ca000 00000880 00000000 00000880 RWIX + I2C2 4819c000 000000d8 00000000 000000d8 RWIX + MCSPI1 481a0000 000001a4 00000000 000001a4 RWIX + DCAN0 481cc000 000001e8 00000000 000001e8 RWIX + DCAN1 481d0000 000001e8 00000000 000001e8 RWIX + PWMSS0 48300000 000002c4 00000000 000002c4 RWIX + PWMSS1 48302000 000002c4 00000000 000002c4 RWIX + PWMSS2 48304000 000002c4 00000000 000002c4 RWIX + RSVD13 48310000 00000100 00000000 00000100 RWIX + RSVD10 48318000 00000100 00000000 00000100 RWIX + TPCC 49000000 00001098 00000000 00001098 RWIX + GEMAC 4a100000 0000128c 00000000 0000128c RWIX + DDR 80000000 00000100 00000000 00000100 RWIX + + +SECTION ALLOCATION MAP + + output attributes/ +section page origin length input sections +-------- ---- ---------- ---------- ---------------- +.text:_c_int00* +* 0 00000000 00000014 + 00000000 00000014 rtspruv3_le.lib : boot_special.obj (.text:_c_int00_noinit_noargs_noexit) + +.text 0 00000014 000018ac + 00000014 00000374 rtspruv3_le.lib : sin.obj (.text:sin) + 00000388 00000314 : frcmpyd.obj (.text:__TI_frcmpyd) + 0000069c 00000258 : frcaddd.obj (.text:__TI_frcaddd) + 000008f4 00000254 : mpyd.obj (.text:__pruabi_mpyd) + 00000b48 00000248 : addd.obj (.text:__pruabi_addd) + 00000d90 000001c8 : mpyf.obj (.text:__pruabi_mpyf) + 00000f58 00000100 : modf.obj (.text:modf) + 00001058 000000b4 : gtd.obj (.text:__pruabi_gtd) + 0000110c 000000b0 : ged.obj (.text:__pruabi_ged) + 000011bc 000000b0 : ltd.obj (.text:__pruabi_ltd) + 0000126c 000000b0 sine1.obj (.text:main) + 0000131c 000000a8 rtspruv3_le.lib : frcmpyf.obj (.text:__TI_frcmpyf) + 000013c4 000000a0 : fixdu.obj (.text:__pruabi_fixdu) + 00001464 0000009c : round.obj (.text:__pruabi_nround) + 00001500 00000090 : eqld.obj (.text:__pruabi_eqd) + 00001590 0000008c : renormd.obj (.text:__TI_renormd) + 0000161c 0000008c : fixdi.obj (.text:__pruabi_fixdi) + 000016a8 00000084 : fltid.obj (.text:__pruabi_fltid) + 0000172c 00000078 : cvtfd.obj (.text:__pruabi_cvtfd) + 000017a4 00000050 : fltuf.obj (.text:__pruabi_fltuf) + 000017f4 0000002c : asri.obj (.text:__pruabi_asri) + 00001820 0000002c : subd.obj (.text:__pruabi_subd) + 0000184c 00000024 : mpyi.obj (.text:__pruabi_mpyi) + 00001870 00000020 : negd.obj (.text:__pruabi_negd) + 00001890 00000020 : trunc.obj (.text:__pruabi_trunc) + 000018b0 00000008 : exit.obj (.text:abort) + 000018b8 00000008 : exit.obj (.text:loader_exit) + +.stack 1 00000000 00000100 UNINITIALIZED + 00000000 00000004 rtspruv3_le.lib : boot.obj (.stack) + 00000004 000000fc --HOLE-- + +.cinit 1 00000000 00000000 UNINITIALIZED + +.fardata 1 00000100 00000040 + 00000100 00000040 rtspruv3_le.lib : sin.obj (.fardata:R$1) + +.resource_table +* 1 00000140 00000014 + 00000140 00000014 sine1.obj (.resource_table:retain) + +.creg.PRU_CFG.noload.near +* 2 00026000 00000044 NOLOAD SECTION + 00026000 00000044 sine1.obj (.creg.PRU_CFG.noload.near) + +.creg.PRU_CFG.near +* 2 00026044 00000000 UNINITIALIZED + +.creg.PRU_CFG.noload.far +* 2 00026044 00000000 NOLOAD SECTION + +.creg.PRU_CFG.far +* 2 00026044 00000000 UNINITIALIZED + + +SEGMENT ATTRIBUTES + + id tag seg value + -- --- --- ----- + 0 PHA_PAGE 1 1 + 1 PHA_PAGE 2 1 + + +GLOBAL SYMBOLS: SORTED ALPHABETICALLY BY Name + +page address name +---- ------- ---- +0 000018b8 C$$EXIT +2 00026000 CT_CFG +abs 481cc000 __PRU_CREG_BASE_DCAN0 +abs 481d0000 __PRU_CREG_BASE_DCAN1 +abs 80000000 __PRU_CREG_BASE_DDR +abs 48040000 __PRU_CREG_BASE_DMTIMER2 +abs 4a100000 __PRU_CREG_BASE_GEMAC +abs 4802a000 __PRU_CREG_BASE_I2C1 +abs 4819c000 __PRU_CREG_BASE_I2C2 +abs 40000000 __PRU_CREG_BASE_L3OCMC +abs 480c8000 __PRU_CREG_BASE_MBX0 +abs 46000000 __PRU_CREG_BASE_MCASP0_DMA +abs 48030000 __PRU_CREG_BASE_MCSPI0 +abs 481a0000 __PRU_CREG_BASE_MCSPI1 +abs 48060000 __PRU_CREG_BASE_MMCHS0 +abs 00026000 __PRU_CREG_BASE_PRU_CFG +abs 00000000 __PRU_CREG_BASE_PRU_DMEM_0_1 +abs 00002000 __PRU_CREG_BASE_PRU_DMEM_1_0 +abs 00030000 __PRU_CREG_BASE_PRU_ECAP +abs 0002e000 __PRU_CREG_BASE_PRU_IEP +abs 00020000 __PRU_CREG_BASE_PRU_INTC +abs 00010000 __PRU_CREG_BASE_PRU_SHAREDMEM +abs 00028000 __PRU_CREG_BASE_PRU_UART +abs 48300000 __PRU_CREG_BASE_PWMSS0 +abs 48302000 __PRU_CREG_BASE_PWMSS1 +abs 48304000 __PRU_CREG_BASE_PWMSS2 +abs 48318000 __PRU_CREG_BASE_RSVD10 +abs 48310000 __PRU_CREG_BASE_RSVD13 +abs 00032400 __PRU_CREG_BASE_RSVD21 +abs 00032000 __PRU_CREG_BASE_RSVD27 +abs 480ca000 __PRU_CREG_BASE_SPINLOCK +abs 49000000 __PRU_CREG_BASE_TPCC +abs 48022000 __PRU_CREG_BASE_UART1 +abs 48024000 __PRU_CREG_BASE_UART2 +abs 0000000e __PRU_CREG_DCAN0 +abs 0000000f __PRU_CREG_DCAN1 +abs 0000001f __PRU_CREG_DDR +abs 00000001 __PRU_CREG_DMTIMER2 +abs 00000009 __PRU_CREG_GEMAC +abs 00000002 __PRU_CREG_I2C1 +abs 00000011 __PRU_CREG_I2C2 +abs 0000001e __PRU_CREG_L3OCMC +abs 00000016 __PRU_CREG_MBX0 +abs 00000008 __PRU_CREG_MCASP0_DMA +abs 00000006 __PRU_CREG_MCSPI0 +abs 00000010 __PRU_CREG_MCSPI1 +abs 00000005 __PRU_CREG_MMCHS0 +abs 00000004 __PRU_CREG_PRU_CFG +abs 00000018 __PRU_CREG_PRU_DMEM_0_1 +abs 00000019 __PRU_CREG_PRU_DMEM_1_0 +abs 00000003 __PRU_CREG_PRU_ECAP +abs 0000001a __PRU_CREG_PRU_IEP +abs 00000000 __PRU_CREG_PRU_INTC +abs 0000001c __PRU_CREG_PRU_SHAREDMEM +abs 00000007 __PRU_CREG_PRU_UART +abs 00000012 __PRU_CREG_PWMSS0 +abs 00000013 __PRU_CREG_PWMSS1 +abs 00000014 __PRU_CREG_PWMSS2 +abs 0000000a __PRU_CREG_RSVD10 +abs 0000000d __PRU_CREG_RSVD13 +abs 00000015 __PRU_CREG_RSVD21 +abs 0000001b __PRU_CREG_RSVD27 +abs 00000017 __PRU_CREG_SPINLOCK +abs 0000001d __PRU_CREG_TPCC +abs 0000000b __PRU_CREG_UART1 +abs 0000000c __PRU_CREG_UART2 +1 00000100 __TI_STACK_END +abs 00000100 __TI_STACK_SIZE +0 0000069c __TI_frcaddd +0 00000388 __TI_frcmpyd +0 0000131c __TI_frcmpyf +0 00001590 __TI_renormd +abs ffffffff __binit__ +abs ffffffff __c_args__ +0 00000b48 __pruabi_addd +0 000017f4 __pruabi_asri +0 0000172c __pruabi_cvtfd +0 00001500 __pruabi_eqd +0 0000161c __pruabi_fixdi +0 000013c4 __pruabi_fixdu +0 000016a8 __pruabi_fltid +0 000017a4 __pruabi_fltuf +0 0000110c __pruabi_ged +0 00001058 __pruabi_gtd +0 000011bc __pruabi_ltd +0 000008f4 __pruabi_mpyd +0 00000d90 __pruabi_mpyf +0 0000184c __pruabi_mpyi +0 00001870 __pruabi_negd +0 00001464 __pruabi_nround +0 00001820 __pruabi_subd +0 00001890 __pruabi_trunc +0 00000000 _c_int00_noinit_noargs_noexit +1 00000000 _stack +0 000018b0 abort +abs ffffffff binit +0 0000126c main +0 00000f58 modf +1 00000140 pru_remoteproc_ResourceTable +0 00000014 sin + + +GLOBAL SYMBOLS: SORTED BY Symbol Address + +page address name +---- ------- ---- +0 00000000 _c_int00_noinit_noargs_noexit +0 00000014 sin +0 00000388 __TI_frcmpyd +0 0000069c __TI_frcaddd +0 000008f4 __pruabi_mpyd +0 00000b48 __pruabi_addd +0 00000d90 __pruabi_mpyf +0 00000f58 modf +0 00001058 __pruabi_gtd +0 0000110c __pruabi_ged +0 000011bc __pruabi_ltd +0 0000126c main +0 0000131c __TI_frcmpyf +0 000013c4 __pruabi_fixdu +0 00001464 __pruabi_nround +0 00001500 __pruabi_eqd +0 00001590 __TI_renormd +0 0000161c __pruabi_fixdi +0 000016a8 __pruabi_fltid +0 0000172c __pruabi_cvtfd +0 000017a4 __pruabi_fltuf +0 000017f4 __pruabi_asri +0 00001820 __pruabi_subd +0 0000184c __pruabi_mpyi +0 00001870 __pruabi_negd +0 00001890 __pruabi_trunc +0 000018b0 abort +0 000018b8 C$$EXIT +1 00000000 _stack +1 00000100 __TI_STACK_END +1 00000140 pru_remoteproc_ResourceTable +2 00026000 CT_CFG +abs 00000000 __PRU_CREG_BASE_PRU_DMEM_0_1 +abs 00000000 __PRU_CREG_PRU_INTC +abs 00000001 __PRU_CREG_DMTIMER2 +abs 00000002 __PRU_CREG_I2C1 +abs 00000003 __PRU_CREG_PRU_ECAP +abs 00000004 __PRU_CREG_PRU_CFG +abs 00000005 __PRU_CREG_MMCHS0 +abs 00000006 __PRU_CREG_MCSPI0 +abs 00000007 __PRU_CREG_PRU_UART +abs 00000008 __PRU_CREG_MCASP0_DMA +abs 00000009 __PRU_CREG_GEMAC +abs 0000000a __PRU_CREG_RSVD10 +abs 0000000b __PRU_CREG_UART1 +abs 0000000c __PRU_CREG_UART2 +abs 0000000d __PRU_CREG_RSVD13 +abs 0000000e __PRU_CREG_DCAN0 +abs 0000000f __PRU_CREG_DCAN1 +abs 00000010 __PRU_CREG_MCSPI1 +abs 00000011 __PRU_CREG_I2C2 +abs 00000012 __PRU_CREG_PWMSS0 +abs 00000013 __PRU_CREG_PWMSS1 +abs 00000014 __PRU_CREG_PWMSS2 +abs 00000015 __PRU_CREG_RSVD21 +abs 00000016 __PRU_CREG_MBX0 +abs 00000017 __PRU_CREG_SPINLOCK +abs 00000018 __PRU_CREG_PRU_DMEM_0_1 +abs 00000019 __PRU_CREG_PRU_DMEM_1_0 +abs 0000001a __PRU_CREG_PRU_IEP +abs 0000001b __PRU_CREG_RSVD27 +abs 0000001c __PRU_CREG_PRU_SHAREDMEM +abs 0000001d __PRU_CREG_TPCC +abs 0000001e __PRU_CREG_L3OCMC +abs 0000001f __PRU_CREG_DDR +abs 00000100 __TI_STACK_SIZE +abs 00002000 __PRU_CREG_BASE_PRU_DMEM_1_0 +abs 00010000 __PRU_CREG_BASE_PRU_SHAREDMEM +abs 00020000 __PRU_CREG_BASE_PRU_INTC +abs 00026000 __PRU_CREG_BASE_PRU_CFG +abs 00028000 __PRU_CREG_BASE_PRU_UART +abs 0002e000 __PRU_CREG_BASE_PRU_IEP +abs 00030000 __PRU_CREG_BASE_PRU_ECAP +abs 00032000 __PRU_CREG_BASE_RSVD27 +abs 00032400 __PRU_CREG_BASE_RSVD21 +abs 40000000 __PRU_CREG_BASE_L3OCMC +abs 46000000 __PRU_CREG_BASE_MCASP0_DMA +abs 48022000 __PRU_CREG_BASE_UART1 +abs 48024000 __PRU_CREG_BASE_UART2 +abs 4802a000 __PRU_CREG_BASE_I2C1 +abs 48030000 __PRU_CREG_BASE_MCSPI0 +abs 48040000 __PRU_CREG_BASE_DMTIMER2 +abs 48060000 __PRU_CREG_BASE_MMCHS0 +abs 480c8000 __PRU_CREG_BASE_MBX0 +abs 480ca000 __PRU_CREG_BASE_SPINLOCK +abs 4819c000 __PRU_CREG_BASE_I2C2 +abs 481a0000 __PRU_CREG_BASE_MCSPI1 +abs 481cc000 __PRU_CREG_BASE_DCAN0 +abs 481d0000 __PRU_CREG_BASE_DCAN1 +abs 48300000 __PRU_CREG_BASE_PWMSS0 +abs 48302000 __PRU_CREG_BASE_PWMSS1 +abs 48304000 __PRU_CREG_BASE_PWMSS2 +abs 48310000 __PRU_CREG_BASE_RSVD13 +abs 48318000 __PRU_CREG_BASE_RSVD10 +abs 49000000 __PRU_CREG_BASE_TPCC +abs 4a100000 __PRU_CREG_BASE_GEMAC +abs 80000000 __PRU_CREG_BASE_DDR +abs ffffffff __binit__ +abs ffffffff __c_args__ +abs ffffffff binit + +[100 symbols] diff --git a/pru-cookbook/05blocks/code/sine.pru0.c b/pru-cookbook/05blocks/code/sine.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..ed6b5778c644f860c8fa2c0f91b7f23cf50e6df7 --- /dev/null +++ b/pru-cookbook/05blocks/code/sine.pru0.c @@ -0,0 +1,58 @@ +// Generate an analog waveform and use a filter to reconstruct it. +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include <math.h> + +#define MAXT 100 // Maximum number of time samples +#define SAWTOOTH // Pick which waveform + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t onCount; // Current count for 1 out + uint32_t offCount; // count for 0 out + uint32_t i; + uint32_t waveform[MAXT]; // Waveform to be produced + + // Generate a periodic wave in an array of MAXT values +#ifdef SAWTOOTH + for(i=0; i<MAXT; i++) { + waveform[i] = i*100/MAXT; + } +#endif +#ifdef TRIANGLE + for(i=0; i<MAXT/2; i++) { + waveform[i] = 2*i*100/MAXT; + waveform[MAXT-i-1] = 2*i*100/MAXT; + } +#endif +#ifdef SINE + float gain = 50.0f; + float bias = 50.0f; + float freq = 2.0f * 3.14159f / MAXT; + for (i=0; i<MAXT; i++){ + waveform[i] = (uint32_t)(bias+gain*sin(i*freq)); + } +#endif + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while (1) { + // Generate a PWM signal whose duty cycle matches + // the amplitude of the signal. + for(i=0; i<MAXT; i++) { + onCount = waveform[i]; + offCount = 100 - onCount; + while(onCount--) { + __R30 |= 0x1; // Set the GPIO pin to 1 + } + while(offCount--) { + __R30 &= ~(0x1); // Clear the GPIO pin + } + } + } +} diff --git a/pru-cookbook/05blocks/code/write_init_pins.sh b/pru-cookbook/05blocks/code/write_init_pins.sh new file mode 100755 index 0000000000000000000000000000000000000000..f52082365a8b2a72a6c30d39938077bd9addf741 --- /dev/null +++ b/pru-cookbook/05blocks/code/write_init_pins.sh @@ -0,0 +1,9 @@ +#!/bin/bash +init_pins=$(readelf -x .init_pins $1 | grep 0x000 | cut -d' ' -f4-7 | xxd -r -p | tr '\0' '\n' | paste - -) +while read -a line; do + if [ ${#line[@]} == 2 ]; then + echo writing \"${line[1]}\" to \"${line[0]}\" + echo ${line[1]} > ${line[0]} + sleep 0.1 + fi +done <<< "$init_pins" diff --git a/pru-cookbook/05blocks/figures/blockDiagram.png b/pru-cookbook/05blocks/figures/blockDiagram.png new file mode 100644 index 0000000000000000000000000000000000000000..901f5b4af178dd896a3a07b5701804a0eaf17fa4 Binary files /dev/null and b/pru-cookbook/05blocks/figures/blockDiagram.png differ diff --git a/pru-cookbook/05blocks/figures/blocks_plots.asv b/pru-cookbook/05blocks/figures/blocks_plots.asv new file mode 100644 index 0000000000000000000000000000000000000000..760326592b877da906a542683a91bb8778433134 --- /dev/null +++ b/pru-cookbook/05blocks/figures/blocks_plots.asv @@ -0,0 +1,27 @@ +% Plots for PRU example +MAXT=100 +waveform = linspace(0, MAXT-1, 100); + +plot(waveform) + +%% +T = 10*(1/50); +Fs = 100; +dt = 1/Fs; +t = 0:dt:T-dt; +x = 50*sawtooth(2*pi*10*t)+50; + +stem(t,x) +title('Sawtooth Waveform') +grid on + +%% Solid sawtooth +T = 10*(1/50); +Fs = 100; +dt = 1/Fs; +t = 0:dt:T-dt; +x = 50*sawtooth(2*pi*20*t)+50; + +stem(t,x) +title('Sampled Sawtooth Waveform') +grid on diff --git a/pru-cookbook/05blocks/figures/blocks_plots.m b/pru-cookbook/05blocks/figures/blocks_plots.m new file mode 100644 index 0000000000000000000000000000000000000000..37ab48a26c799541d51ec0fa43835b9387131a7d --- /dev/null +++ b/pru-cookbook/05blocks/figures/blocks_plots.m @@ -0,0 +1,27 @@ +% Plots for PRU example +MAXT=100 +waveform = linspace(0, MAXT-1, 100); + +plot(waveform) + +%% +T = 20*(1/50); +Fs = 1000; +dt = 1/Fs; +t = 0:dt:T-dt; +x = 50*sawtooth(2*pi*10*t)+50; + +plot(t,x) +title('Sawtooth Waveform') +grid on + +%% Solid sawtooth +T = 5*(1/50); +Fs = 200; +dt = 1/Fs; +t = 0:dt:T-dt; +x = 50*sawtooth(2*pi*10*t)+50; + +stem(t,x) +title('Sampled Sawtooth Waveform') +grid on diff --git a/pru-cookbook/05blocks/figures/filter.fzz b/pru-cookbook/05blocks/figures/filter.fzz new file mode 100644 index 0000000000000000000000000000000000000000..3342d1cc50cd475f668a42b18f1900448ebb1c92 Binary files /dev/null and b/pru-cookbook/05blocks/figures/filter.fzz differ diff --git a/pru-cookbook/05blocks/figures/filter_bb.png b/pru-cookbook/05blocks/figures/filter_bb.png new file mode 100644 index 0000000000000000000000000000000000000000..3fc2955b5c48540261818efd47e01e6c1d268350 Binary files /dev/null and b/pru-cookbook/05blocks/figures/filter_bb.png differ diff --git a/pru-cookbook/05blocks/figures/matrix_j1.jpg b/pru-cookbook/05blocks/figures/matrix_j1.jpg new file mode 100644 index 0000000000000000000000000000000000000000..711c90412e80626e916a14fe0474ca083eb3c642 Binary files /dev/null and b/pru-cookbook/05blocks/figures/matrix_j1.jpg differ diff --git a/pru-cookbook/05blocks/figures/neo.fzz b/pru-cookbook/05blocks/figures/neo.fzz new file mode 100644 index 0000000000000000000000000000000000000000..36191328e10761ac75321186fce2f2955ca8004e Binary files /dev/null and b/pru-cookbook/05blocks/figures/neo.fzz differ diff --git a/pru-cookbook/05blocks/figures/neo_bb.png b/pru-cookbook/05blocks/figures/neo_bb.png new file mode 100644 index 0000000000000000000000000000000000000000..e7cc6b2fa60b62fff23b3abddc4e2e6578408718 Binary files /dev/null and b/pru-cookbook/05blocks/figures/neo_bb.png differ diff --git a/pru-cookbook/05blocks/figures/neo_data_seq.png b/pru-cookbook/05blocks/figures/neo_data_seq.png new file mode 100644 index 0000000000000000000000000000000000000000..bef90bbeb121dee12cd393faceb17e59e2e9a985 Binary files /dev/null and b/pru-cookbook/05blocks/figures/neo_data_seq.png differ diff --git a/pru-cookbook/05blocks/figures/neo_scope.png b/pru-cookbook/05blocks/figures/neo_scope.png new file mode 100644 index 0000000000000000000000000000000000000000..919d0b8d574b64cbf237761befed37b3812f6f3c Binary files /dev/null and b/pru-cookbook/05blocks/figures/neo_scope.png differ diff --git a/pru-cookbook/05blocks/figures/neo_sequence.png b/pru-cookbook/05blocks/figures/neo_sequence.png new file mode 100644 index 0000000000000000000000000000000000000000..be8c48c18e270048b8ae6585b237e6b0973735cc Binary files /dev/null and b/pru-cookbook/05blocks/figures/neo_sequence.png differ diff --git a/pru-cookbook/05blocks/figures/pwm1.png b/pru-cookbook/05blocks/figures/pwm1.png new file mode 100644 index 0000000000000000000000000000000000000000..6b56f035350a75c7459d82d268de4cf58bba6ffd Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm1.png differ diff --git a/pru-cookbook/05blocks/figures/pwm2.png b/pru-cookbook/05blocks/figures/pwm2.png new file mode 100644 index 0000000000000000000000000000000000000000..248a142b345ccd3e425e0bfd596116df8b4ad30b Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm2.png differ diff --git a/pru-cookbook/05blocks/figures/pwm3.png b/pru-cookbook/05blocks/figures/pwm3.png new file mode 100644 index 0000000000000000000000000000000000000000..0b76c32939ab7d9143ed48815b5739caad428e0a Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm3.png differ diff --git a/pru-cookbook/05blocks/figures/pwm4.png b/pru-cookbook/05blocks/figures/pwm4.png new file mode 100644 index 0000000000000000000000000000000000000000..dfc9c75da241ba42dffc326726f143dc2a58a8d0 Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm4.png differ diff --git a/pru-cookbook/05blocks/figures/pwm5_no_loop.png b/pru-cookbook/05blocks/figures/pwm5_no_loop.png new file mode 100644 index 0000000000000000000000000000000000000000..35aff88d2bf00c727c6c6ebf0667fa0464c7c68e Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm5_no_loop.png differ diff --git a/pru-cookbook/05blocks/figures/pwm5_zoomed.png b/pru-cookbook/05blocks/figures/pwm5_zoomed.png new file mode 100644 index 0000000000000000000000000000000000000000..6802cb55ca07d815d665d7aa8f987addd4366c13 Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm5_zoomed.png differ diff --git a/pru-cookbook/05blocks/figures/pwm6_synced.png b/pru-cookbook/05blocks/figures/pwm6_synced.png new file mode 100644 index 0000000000000000000000000000000000000000..1b71ee53d8d974fd0207342ecd37fe27a34b89f5 Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm6_synced.png differ diff --git a/pru-cookbook/05blocks/figures/pwm7_two_prus_running.png b/pru-cookbook/05blocks/figures/pwm7_two_prus_running.png new file mode 100644 index 0000000000000000000000000000000000000000..662706b9203b7734ad9e3146ee7a8aa6386f39f0 Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm7_two_prus_running.png differ diff --git a/pru-cookbook/05blocks/figures/pwm7_two_prus_stopped.png b/pru-cookbook/05blocks/figures/pwm7_two_prus_stopped.png new file mode 100644 index 0000000000000000000000000000000000000000..00cfc20dc65ce2cb5b88cb34a6f60a61266e05bc Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm7_two_prus_stopped.png differ diff --git a/pru-cookbook/05blocks/figures/pwm8_prus_sycned.png b/pru-cookbook/05blocks/figures/pwm8_prus_sycned.png new file mode 100644 index 0000000000000000000000000000000000000000..f27388fa93ec02685296c3aa13ec3a7b3a182144 Binary files /dev/null and b/pru-cookbook/05blocks/figures/pwm8_prus_sycned.png differ diff --git a/pru-cookbook/05blocks/figures/rgb_fpp.png b/pru-cookbook/05blocks/figures/rgb_fpp.png new file mode 100644 index 0000000000000000000000000000000000000000..38adf5a27d49f028e09b75426211e743b35aa97b Binary files /dev/null and b/pru-cookbook/05blocks/figures/rgb_fpp.png differ diff --git a/pru-cookbook/05blocks/figures/rgb_pru.jpg b/pru-cookbook/05blocks/figures/rgb_pru.jpg new file mode 100644 index 0000000000000000000000000000000000000000..a588b42bd7e02958c8e48142921825d79adb8b8e Binary files /dev/null and b/pru-cookbook/05blocks/figures/rgb_pru.jpg differ diff --git a/pru-cookbook/05blocks/figures/rgb_python.jpg b/pru-cookbook/05blocks/figures/rgb_python.jpg new file mode 100644 index 0000000000000000000000000000000000000000..26c83ec2d73310ea64aea2252c1c437a304530f7 Binary files /dev/null and b/pru-cookbook/05blocks/figures/rgb_python.jpg differ diff --git a/pru-cookbook/05blocks/figures/rgb_waveforms.png b/pru-cookbook/05blocks/figures/rgb_waveforms.png new file mode 100644 index 0000000000000000000000000000000000000000..df9d8cbdbe93f2b72c08945246dccec62aba3879 Binary files /dev/null and b/pru-cookbook/05blocks/figures/rgb_waveforms.png differ diff --git a/pru-cookbook/05blocks/figures/ring_around.mp4 b/pru-cookbook/05blocks/figures/ring_around.mp4 new file mode 100644 index 0000000000000000000000000000000000000000..1695cb0532647566c09e966dca41348943962bfb Binary files /dev/null and b/pru-cookbook/05blocks/figures/ring_around.mp4 differ diff --git a/pru-cookbook/05blocks/figures/sawhighercutoff.png b/pru-cookbook/05blocks/figures/sawhighercutoff.png new file mode 100644 index 0000000000000000000000000000000000000000..2ca0f4d4023db196b2e85b17dda9dc915e9b00c4 Binary files /dev/null and b/pru-cookbook/05blocks/figures/sawhighercutoff.png differ diff --git a/pru-cookbook/05blocks/figures/sawlowercutoff.png b/pru-cookbook/05blocks/figures/sawlowercutoff.png new file mode 100644 index 0000000000000000000000000000000000000000..933d7cc2587e9177bbc98572a302d2d6a298d5a0 Binary files /dev/null and b/pru-cookbook/05blocks/figures/sawlowercutoff.png differ diff --git a/pru-cookbook/05blocks/figures/sawscope.png b/pru-cookbook/05blocks/figures/sawscope.png new file mode 100644 index 0000000000000000000000000000000000000000..d9d191d3ebce062b6d44108a17fcf7871a39b775 Binary files /dev/null and b/pru-cookbook/05blocks/figures/sawscope.png differ diff --git a/pru-cookbook/05blocks/figures/sawtoothsampled.png b/pru-cookbook/05blocks/figures/sawtoothsampled.png new file mode 100644 index 0000000000000000000000000000000000000000..c6b09731731306d2e40695e9cc5a8bc2635bce4e Binary files /dev/null and b/pru-cookbook/05blocks/figures/sawtoothsampled.png differ diff --git a/pru-cookbook/05blocks/figures/sawtoothsmooth.png b/pru-cookbook/05blocks/figures/sawtoothsmooth.png new file mode 100644 index 0000000000000000000000000000000000000000..60296663cb33d64b76f7fc099fc5fbb0fbf86a5d Binary files /dev/null and b/pru-cookbook/05blocks/figures/sawtoothsmooth.png differ diff --git a/pru-cookbook/05blocks/figures/sawunfiltered.png b/pru-cookbook/05blocks/figures/sawunfiltered.png new file mode 100644 index 0000000000000000000000000000000000000000..9f77ac535d832cbf159b072d59cdf69819af61a4 Binary files /dev/null and b/pru-cookbook/05blocks/figures/sawunfiltered.png differ diff --git a/pru-cookbook/05blocks/figures/sine.png b/pru-cookbook/05blocks/figures/sine.png new file mode 100644 index 0000000000000000000000000000000000000000..0c6913b16be3ab3694c99fc41c4b68e969c75db6 Binary files /dev/null and b/pru-cookbook/05blocks/figures/sine.png differ diff --git a/pru-cookbook/05blocks/figures/triangle.png b/pru-cookbook/05blocks/figures/triangle.png new file mode 100644 index 0000000000000000000000000000000000000000..988a28810411074efc6de6337040dce50d558513 Binary files /dev/null and b/pru-cookbook/05blocks/figures/triangle.png differ diff --git a/pru-cookbook/06io/code/Makefile b/pru-cookbook/06io/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/06io/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/06io/code/gpio.pru0.c b/pru-cookbook/06io/code/gpio.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..bcab90eff3f115577055dac3781c50bfb3e0efbf --- /dev/null +++ b/pru-cookbook/06io/code/gpio.pru0.c @@ -0,0 +1,23 @@ +// This code accesses GPIO without using R30 and R31 +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define P9_11 (0x1<<30) // Bit position tied to P9_11 on Black +#define P2_05 (0x1<<30) // Bit position tied to P2_05 on Pocket + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t *gpio0 = (uint32_t *)GPIO0; + + while(1) { + gpio0[GPIO_SETDATAOUT] = P9_11; + __delay_cycles(100000000); + gpio0[GPIO_CLEARDATAOUT] = P9_11; + __delay_cycles(100000000); + } +} diff --git a/pru-cookbook/06io/code/setup.sh b/pru-cookbook/06io/code/setup.sh new file mode 100755 index 0000000000000000000000000000000000000000..e676ed7abe2a6beaf62d3ce6c56f5aa151d86700 --- /dev/null +++ b/pru-cookbook/06io/code/setup.sh @@ -0,0 +1,28 @@ +#!/bin/bash + +export TARGET=gpio.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_11" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_05" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin gpio + config-pin -q $pin +done diff --git a/pru-cookbook/06io/figures/gpio0delay.png b/pru-cookbook/06io/figures/gpio0delay.png new file mode 100644 index 0000000000000000000000000000000000000000..ad8568542d5d6b2445cb7ba2e42792bc671e6d6b Binary files /dev/null and b/pru-cookbook/06io/figures/gpio0delay.png differ diff --git a/pru-cookbook/06io/figures/jitter.png b/pru-cookbook/06io/figures/jitter.png new file mode 100644 index 0000000000000000000000000000000000000000..e9b80b4d61809420fa6a8ee6e23cb44ee2ab9c2b Binary files /dev/null and b/pru-cookbook/06io/figures/jitter.png differ diff --git a/pru-cookbook/06io/figures/pruIntegration.png b/pru-cookbook/06io/figures/pruIntegration.png new file mode 100644 index 0000000000000000000000000000000000000000..de7ef9196b3e56e5bd1ff14a9957c2ff23d3fa41 Binary files /dev/null and b/pru-cookbook/06io/figures/pruIntegration.png differ diff --git a/pru-cookbook/06io/io.html b/pru-cookbook/06io/io.html new file mode 100644 index 0000000000000000000000000000000000000000..5bb5425db9ef3699741ac9d74701f36aac6519b4 --- /dev/null +++ b/pru-cookbook/06io/io.html @@ -0,0 +1,1003 @@ +<!DOCTYPE html> +<html lang="en"> +<head> +<meta charset="UTF-8"> +<!--[if IE]><meta http-equiv="X-UA-Compatible" content="IE=edge"><![endif]--> +<meta name="viewport" content="width=device-width, initial-scale=1.0"> +<meta name="generator" content="Asciidoctor 1.5.8"> +<title>Accessing More I/O</title> +<link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Open+Sans:300,300italic,400,400italic,600,600italic%7CNoto+Serif:400,400italic,700,700italic%7CDroid+Sans+Mono:400,700"> +<style> +/* Asciidoctor default stylesheet | MIT License | http://asciidoctor.org */ +/* Uncomment @import statement below to use as custom stylesheet */ +/*@import "https://fonts.googleapis.com/css?family=Open+Sans:300,300italic,400,400italic,600,600italic%7CNoto+Serif:400,400italic,700,700italic%7CDroid+Sans+Mono:400,700";*/ +article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block} +audio,canvas,video{display:inline-block} +audio:not([controls]){display:none;height:0} +script{display:none!important} +html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%} +a{background:transparent} +a:focus{outline:thin dotted} +a:active,a:hover{outline:0} +h1{font-size:2em;margin:.67em 0} +abbr[title]{border-bottom:1px dotted} +b,strong{font-weight:bold} +dfn{font-style:italic} +hr{-moz-box-sizing:content-box;box-sizing:content-box;height:0} +mark{background:#ff0;color:#000} +code,kbd,pre,samp{font-family:monospace;font-size:1em} +pre{white-space:pre-wrap} +q{quotes:"\201C" "\201D" "\2018" "\2019"} +small{font-size:80%} +sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline} +sup{top:-.5em} +sub{bottom:-.25em} +img{border:0} +svg:not(:root){overflow:hidden} +figure{margin:0} +fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em} +legend{border:0;padding:0} +button,input,select,textarea{font-family:inherit;font-size:100%;margin:0} +button,input{line-height:normal} +button,select{text-transform:none} +button,html input[type="button"],input[type="reset"],input[type="submit"]{-webkit-appearance:button;cursor:pointer} +button[disabled],html input[disabled]{cursor:default} +input[type="checkbox"],input[type="radio"]{box-sizing:border-box;padding:0} +button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0} +textarea{overflow:auto;vertical-align:top} +table{border-collapse:collapse;border-spacing:0} +*,*::before,*::after{-moz-box-sizing:border-box;-webkit-box-sizing:border-box;box-sizing:border-box} +html,body{font-size:100%} +body{background:#fff;color:rgba(0,0,0,.8);padding:0;margin:0;font-family:"Noto Serif","DejaVu Serif",serif;font-weight:400;font-style:normal;line-height:1;position:relative;cursor:auto;tab-size:4;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased} +a:hover{cursor:pointer} +img,object,embed{max-width:100%;height:auto} +object,embed{height:100%} +img{-ms-interpolation-mode:bicubic} +.left{float:left!important} +.right{float:right!important} +.text-left{text-align:left!important} +.text-right{text-align:right!important} +.text-center{text-align:center!important} +.text-justify{text-align:justify!important} +.hide{display:none} +img,object,svg{display:inline-block;vertical-align:middle} +textarea{height:auto;min-height:50px} +select{width:100%} +.center{margin-left:auto;margin-right:auto} +.stretch{width:100%} +.subheader,.admonitionblock td.content>.title,.audioblock>.title,.exampleblock>.title,.imageblock>.title,.listingblock>.title,.literalblock>.title,.stemblock>.title,.openblock>.title,.paragraph>.title,.quoteblock>.title,table.tableblock>.title,.verseblock>.title,.videoblock>.title,.dlist>.title,.olist>.title,.ulist>.title,.qlist>.title,.hdlist>.title{line-height:1.45;color:#7a2518;font-weight:400;margin-top:0;margin-bottom:.25em} +div,dl,dt,dd,ul,ol,li,h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6,pre,form,p,blockquote,th,td{margin:0;padding:0;direction:ltr} +a{color:#2156a5;text-decoration:underline;line-height:inherit} +a:hover,a:focus{color:#1d4b8f} +a img{border:none} +p{font-family:inherit;font-weight:400;font-size:1em;line-height:1.6;margin-bottom:1.25em;text-rendering:optimizeLegibility} +p aside{font-size:.875em;line-height:1.35;font-style:italic} +h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6{font-family:"Open Sans","DejaVu Sans",sans-serif;font-weight:300;font-style:normal;color:#ba3925;text-rendering:optimizeLegibility;margin-top:1em;margin-bottom:.5em;line-height:1.0125em} +h1 small,h2 small,h3 small,#toctitle small,.sidebarblock>.content>.title small,h4 small,h5 small,h6 small{font-size:60%;color:#e99b8f;line-height:0} +h1{font-size:2.125em} +h2{font-size:1.6875em} +h3,#toctitle,.sidebarblock>.content>.title{font-size:1.375em} +h4,h5{font-size:1.125em} +h6{font-size:1em} +hr{border:solid #dddddf;border-width:1px 0 0;clear:both;margin:1.25em 0 1.1875em;height:0} +em,i{font-style:italic;line-height:inherit} +strong,b{font-weight:bold;line-height:inherit} +small{font-size:60%;line-height:inherit} +code{font-family:"Droid Sans Mono","DejaVu Sans Mono",monospace;font-weight:400;color:rgba(0,0,0,.9)} +ul,ol,dl{font-size:1em;line-height:1.6;margin-bottom:1.25em;list-style-position:outside;font-family:inherit} +ul,ol{margin-left:1.5em} +ul li ul,ul li ol{margin-left:1.25em;margin-bottom:0;font-size:1em} +ul.square li ul,ul.circle li ul,ul.disc li ul{list-style:inherit} +ul.square{list-style-type:square} +ul.circle{list-style-type:circle} +ul.disc{list-style-type:disc} +ol li ul,ol li ol{margin-left:1.25em;margin-bottom:0} +dl dt{margin-bottom:.3125em;font-weight:bold} +dl dd{margin-bottom:1.25em} +abbr,acronym{text-transform:uppercase;font-size:90%;color:rgba(0,0,0,.8);border-bottom:1px dotted #ddd;cursor:help} +abbr{text-transform:none} +blockquote{margin:0 0 1.25em;padding:.5625em 1.25em 0 1.1875em;border-left:1px solid #ddd} +blockquote cite{display:block;font-size:.9375em;color:rgba(0,0,0,.6)} +blockquote cite::before{content:"\2014 \0020"} +blockquote cite a,blockquote cite a:visited{color:rgba(0,0,0,.6)} +blockquote,blockquote p{line-height:1.6;color:rgba(0,0,0,.85)} +@media screen and (min-width:768px){h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6{line-height:1.2} +h1{font-size:2.75em} +h2{font-size:2.3125em} +h3,#toctitle,.sidebarblock>.content>.title{font-size:1.6875em} +h4{font-size:1.4375em}} +table{background:#fff;margin-bottom:1.25em;border:solid 1px #dedede} +table thead,table tfoot{background:#f7f8f7} +table thead tr th,table thead tr td,table tfoot tr th,table tfoot tr td{padding:.5em .625em .625em;font-size:inherit;color:rgba(0,0,0,.8);text-align:left} +table tr th,table tr td{padding:.5625em .625em;font-size:inherit;color:rgba(0,0,0,.8)} +table tr.even,table tr.alt,table tr:nth-of-type(even){background:#f8f8f7} +table thead tr th,table tfoot tr th,table tbody tr td,table tr td,table tfoot tr td{display:table-cell;line-height:1.6} +h1,h2,h3,#toctitle,.sidebarblock>.content>.title,h4,h5,h6{line-height:1.2;word-spacing:-.05em} +h1 strong,h2 strong,h3 strong,#toctitle strong,.sidebarblock>.content>.title strong,h4 strong,h5 strong,h6 strong{font-weight:400} +.clearfix::before,.clearfix::after,.float-group::before,.float-group::after{content:" ";display:table} +.clearfix::after,.float-group::after{clear:both} +*:not(pre)>code{font-size:.9375em;font-style:normal!important;letter-spacing:0;padding:.1em .5ex;word-spacing:-.15em;background-color:#f7f7f8;-webkit-border-radius:4px;border-radius:4px;line-height:1.45;text-rendering:optimizeSpeed;word-wrap:break-word} +*:not(pre)>code.nobreak{word-wrap:normal} +*:not(pre)>code.nowrap{white-space:nowrap} +pre,pre>code{line-height:1.45;color:rgba(0,0,0,.9);font-family:"Droid Sans Mono","DejaVu Sans Mono",monospace;font-weight:400;text-rendering:optimizeSpeed} +em em{font-style:normal} +strong strong{font-weight:400} +.keyseq{color:rgba(51,51,51,.8)} +kbd{font-family:"Droid Sans Mono","DejaVu Sans Mono",monospace;display:inline-block;color:rgba(0,0,0,.8);font-size:.65em;line-height:1.45;background-color:#f7f7f7;border:1px solid #ccc;-webkit-border-radius:3px;border-radius:3px;-webkit-box-shadow:0 1px 0 rgba(0,0,0,.2),0 0 0 .1em white inset;box-shadow:0 1px 0 rgba(0,0,0,.2),0 0 0 .1em #fff inset;margin:0 .15em;padding:.2em .5em;vertical-align:middle;position:relative;top:-.1em;white-space:nowrap} +.keyseq kbd:first-child{margin-left:0} +.keyseq kbd:last-child{margin-right:0} +.menuseq,.menuref{color:#000} +.menuseq b:not(.caret),.menuref{font-weight:inherit} +.menuseq{word-spacing:-.02em} +.menuseq b.caret{font-size:1.25em;line-height:.8} +.menuseq i.caret{font-weight:bold;text-align:center;width:.45em} +b.button::before,b.button::after{position:relative;top:-1px;font-weight:400} +b.button::before{content:"[";padding:0 3px 0 2px} +b.button::after{content:"]";padding:0 2px 0 3px} +p a>code:hover{color:rgba(0,0,0,.9)} +#header,#content,#footnotes,#footer{width:100%;margin-left:auto;margin-right:auto;margin-top:0;margin-bottom:0;max-width:62.5em;*zoom:1;position:relative;padding-left:.9375em;padding-right:.9375em} +#header::before,#header::after,#content::before,#content::after,#footnotes::before,#footnotes::after,#footer::before,#footer::after{content:" ";display:table} +#header::after,#content::after,#footnotes::after,#footer::after{clear:both} +#content{margin-top:1.25em} +#content::before{content:none} +#header>h1:first-child{color:rgba(0,0,0,.85);margin-top:2.25rem;margin-bottom:0} +#header>h1:first-child+#toc{margin-top:8px;border-top:1px solid #dddddf} +#header>h1:only-child,body.toc2 #header>h1:nth-last-child(2){border-bottom:1px solid #dddddf;padding-bottom:8px} +#header .details{border-bottom:1px solid #dddddf;line-height:1.45;padding-top:.25em;padding-bottom:.25em;padding-left:.25em;color:rgba(0,0,0,.6);display:-ms-flexbox;display:-webkit-flex;display:flex;-ms-flex-flow:row wrap;-webkit-flex-flow:row wrap;flex-flow:row wrap} +#header .details span:first-child{margin-left:-.125em} +#header .details span.email a{color:rgba(0,0,0,.85)} +#header .details br{display:none} +#header .details br+span::before{content:"\00a0\2013\00a0"} +#header .details br+span.author::before{content:"\00a0\22c5\00a0";color:rgba(0,0,0,.85)} +#header .details br+span#revremark::before{content:"\00a0|\00a0"} +#header #revnumber{text-transform:capitalize} +#header #revnumber::after{content:"\00a0"} +#content>h1:first-child:not([class]){color:rgba(0,0,0,.85);border-bottom:1px solid #dddddf;padding-bottom:8px;margin-top:0;padding-top:1rem;margin-bottom:1.25rem} +#toc{border-bottom:1px solid #e7e7e9;padding-bottom:.5em} +#toc>ul{margin-left:.125em} +#toc ul.sectlevel0>li>a{font-style:italic} +#toc ul.sectlevel0 ul.sectlevel1{margin:.5em 0} +#toc ul{font-family:"Open Sans","DejaVu Sans",sans-serif;list-style-type:none} +#toc li{line-height:1.3334;margin-top:.3334em} +#toc a{text-decoration:none} +#toc a:active{text-decoration:underline} +#toctitle{color:#7a2518;font-size:1.2em} +@media screen and (min-width:768px){#toctitle{font-size:1.375em} +body.toc2{padding-left:15em;padding-right:0} +#toc.toc2{margin-top:0!important;background-color:#f8f8f7;position:fixed;width:15em;left:0;top:0;border-right:1px solid #e7e7e9;border-top-width:0!important;border-bottom-width:0!important;z-index:1000;padding:1.25em 1em;height:100%;overflow:auto} +#toc.toc2 #toctitle{margin-top:0;margin-bottom:.8rem;font-size:1.2em} +#toc.toc2>ul{font-size:.9em;margin-bottom:0} +#toc.toc2 ul ul{margin-left:0;padding-left:1em} +#toc.toc2 ul.sectlevel0 ul.sectlevel1{padding-left:0;margin-top:.5em;margin-bottom:.5em} +body.toc2.toc-right{padding-left:0;padding-right:15em} +body.toc2.toc-right #toc.toc2{border-right-width:0;border-left:1px solid #e7e7e9;left:auto;right:0}} +@media screen and (min-width:1280px){body.toc2{padding-left:20em;padding-right:0} +#toc.toc2{width:20em} +#toc.toc2 #toctitle{font-size:1.375em} +#toc.toc2>ul{font-size:.95em} +#toc.toc2 ul ul{padding-left:1.25em} +body.toc2.toc-right{padding-left:0;padding-right:20em}} +#content #toc{border-style:solid;border-width:1px;border-color:#e0e0dc;margin-bottom:1.25em;padding:1.25em;background:#f8f8f7;-webkit-border-radius:4px;border-radius:4px} +#content #toc>:first-child{margin-top:0} +#content #toc>:last-child{margin-bottom:0} +#footer{max-width:100%;background-color:rgba(0,0,0,.8);padding:1.25em} +#footer-text{color:rgba(255,255,255,.8);line-height:1.44} +#content{margin-bottom:.625em} +.sect1{padding-bottom:.625em} +@media screen and (min-width:768px){#content{margin-bottom:1.25em} +.sect1{padding-bottom:1.25em}} +.sect1:last-child{padding-bottom:0} +.sect1+.sect1{border-top:1px solid #e7e7e9} +#content h1>a.anchor,h2>a.anchor,h3>a.anchor,#toctitle>a.anchor,.sidebarblock>.content>.title>a.anchor,h4>a.anchor,h5>a.anchor,h6>a.anchor{position:absolute;z-index:1001;width:1.5ex;margin-left:-1.5ex;display:block;text-decoration:none!important;visibility:hidden;text-align:center;font-weight:400} +#content h1>a.anchor::before,h2>a.anchor::before,h3>a.anchor::before,#toctitle>a.anchor::before,.sidebarblock>.content>.title>a.anchor::before,h4>a.anchor::before,h5>a.anchor::before,h6>a.anchor::before{content:"\00A7";font-size:.85em;display:block;padding-top:.1em} +#content h1:hover>a.anchor,#content h1>a.anchor:hover,h2:hover>a.anchor,h2>a.anchor:hover,h3:hover>a.anchor,#toctitle:hover>a.anchor,.sidebarblock>.content>.title:hover>a.anchor,h3>a.anchor:hover,#toctitle>a.anchor:hover,.sidebarblock>.content>.title>a.anchor:hover,h4:hover>a.anchor,h4>a.anchor:hover,h5:hover>a.anchor,h5>a.anchor:hover,h6:hover>a.anchor,h6>a.anchor:hover{visibility:visible} +#content h1>a.link,h2>a.link,h3>a.link,#toctitle>a.link,.sidebarblock>.content>.title>a.link,h4>a.link,h5>a.link,h6>a.link{color:#ba3925;text-decoration:none} +#content h1>a.link:hover,h2>a.link:hover,h3>a.link:hover,#toctitle>a.link:hover,.sidebarblock>.content>.title>a.link:hover,h4>a.link:hover,h5>a.link:hover,h6>a.link:hover{color:#a53221} +.audioblock,.imageblock,.literalblock,.listingblock,.stemblock,.videoblock{margin-bottom:1.25em} +.admonitionblock td.content>.title,.audioblock>.title,.exampleblock>.title,.imageblock>.title,.listingblock>.title,.literalblock>.title,.stemblock>.title,.openblock>.title,.paragraph>.title,.quoteblock>.title,table.tableblock>.title,.verseblock>.title,.videoblock>.title,.dlist>.title,.olist>.title,.ulist>.title,.qlist>.title,.hdlist>.title{text-rendering:optimizeLegibility;text-align:left;font-family:"Noto Serif","DejaVu Serif",serif;font-size:1rem;font-style:italic} +table.tableblock.fit-content>caption.title{white-space:nowrap;width:0} +.paragraph.lead>p,#preamble>.sectionbody>[class="paragraph"]:first-of-type p{font-size:1.21875em;line-height:1.6;color:rgba(0,0,0,.85)} +table.tableblock #preamble>.sectionbody>[class="paragraph"]:first-of-type p{font-size:inherit} +.admonitionblock>table{border-collapse:separate;border:0;background:none;width:100%} +.admonitionblock>table td.icon{text-align:center;width:80px} +.admonitionblock>table td.icon img{max-width:none} +.admonitionblock>table td.icon .title{font-weight:bold;font-family:"Open Sans","DejaVu Sans",sans-serif;text-transform:uppercase} +.admonitionblock>table td.content{padding-left:1.125em;padding-right:1.25em;border-left:1px solid #dddddf;color:rgba(0,0,0,.6)} +.admonitionblock>table td.content>:last-child>:last-child{margin-bottom:0} +.exampleblock>.content{border-style:solid;border-width:1px;border-color:#e6e6e6;margin-bottom:1.25em;padding:1.25em;background:#fff;-webkit-border-radius:4px;border-radius:4px} +.exampleblock>.content>:first-child{margin-top:0} +.exampleblock>.content>:last-child{margin-bottom:0} +.sidebarblock{border-style:solid;border-width:1px;border-color:#e0e0dc;margin-bottom:1.25em;padding:1.25em;background:#f8f8f7;-webkit-border-radius:4px;border-radius:4px} +.sidebarblock>:first-child{margin-top:0} +.sidebarblock>:last-child{margin-bottom:0} +.sidebarblock>.content>.title{color:#7a2518;margin-top:0;text-align:center} +.exampleblock>.content>:last-child>:last-child,.exampleblock>.content .olist>ol>li:last-child>:last-child,.exampleblock>.content .ulist>ul>li:last-child>:last-child,.exampleblock>.content .qlist>ol>li:last-child>:last-child,.sidebarblock>.content>:last-child>:last-child,.sidebarblock>.content .olist>ol>li:last-child>:last-child,.sidebarblock>.content .ulist>ul>li:last-child>:last-child,.sidebarblock>.content .qlist>ol>li:last-child>:last-child{margin-bottom:0} +.literalblock pre,.listingblock pre:not(.highlight),.listingblock pre[class="highlight"],.listingblock pre[class^="highlight "],.listingblock pre.CodeRay,.listingblock pre.prettyprint{background:#f7f7f8} +.sidebarblock .literalblock pre,.sidebarblock .listingblock pre:not(.highlight),.sidebarblock .listingblock pre[class="highlight"],.sidebarblock .listingblock pre[class^="highlight "],.sidebarblock .listingblock pre.CodeRay,.sidebarblock .listingblock pre.prettyprint{background:#f2f1f1} +.literalblock pre,.literalblock pre[class],.listingblock pre,.listingblock pre[class]{-webkit-border-radius:4px;border-radius:4px;word-wrap:break-word;overflow-x:auto;padding:1em;font-size:.8125em} +@media screen and (min-width:768px){.literalblock pre,.literalblock pre[class],.listingblock pre,.listingblock pre[class]{font-size:.90625em}} +@media screen and (min-width:1280px){.literalblock pre,.literalblock pre[class],.listingblock pre,.listingblock pre[class]{font-size:1em}} +.literalblock pre.nowrap,.literalblock pre.nowrap pre,.listingblock pre.nowrap,.listingblock pre.nowrap pre{white-space:pre;word-wrap:normal} +.literalblock.output pre{color:#f7f7f8;background-color:rgba(0,0,0,.9)} +.listingblock pre.highlightjs{padding:0} +.listingblock pre.highlightjs>code{padding:1em;-webkit-border-radius:4px;border-radius:4px} +.listingblock pre.prettyprint{border-width:0} +.listingblock>.content{position:relative} +.listingblock code[data-lang]::before{display:none;content:attr(data-lang);position:absolute;font-size:.75em;top:.425rem;right:.5rem;line-height:1;text-transform:uppercase;color:#999} +.listingblock:hover code[data-lang]::before{display:block} +.listingblock.terminal pre .command::before{content:attr(data-prompt);padding-right:.5em;color:#999} +.listingblock.terminal pre .command:not([data-prompt])::before{content:"$"} +table.pyhltable{border-collapse:separate;border:0;margin-bottom:0;background:none} +table.pyhltable td{vertical-align:top;padding-top:0;padding-bottom:0;line-height:1.45} +table.pyhltable td.code{padding-left:.75em;padding-right:0} +pre.pygments .lineno,table.pyhltable td:not(.code){color:#999;padding-left:0;padding-right:.5em;border-right:1px solid #dddddf} +pre.pygments .lineno{display:inline-block;margin-right:.25em} +table.pyhltable .linenodiv{background:none!important;padding-right:0!important} +.quoteblock{margin:0 1em 1.25em 1.5em;display:table} +.quoteblock>.title{margin-left:-1.5em;margin-bottom:.75em} +.quoteblock blockquote,.quoteblock p{color:rgba(0,0,0,.85);font-size:1.15rem;line-height:1.75;word-spacing:.1em;letter-spacing:0;font-style:italic;text-align:justify} +.quoteblock blockquote{margin:0;padding:0;border:0} +.quoteblock blockquote::before{content:"\201c";float:left;font-size:2.75em;font-weight:bold;line-height:.6em;margin-left:-.6em;color:#7a2518;text-shadow:0 1px 2px rgba(0,0,0,.1)} +.quoteblock blockquote>.paragraph:last-child p{margin-bottom:0} +.quoteblock .attribution{margin-top:.75em;margin-right:.5ex;text-align:right} +.verseblock{margin:0 1em 1.25em} +.verseblock pre{font-family:"Open Sans","DejaVu Sans",sans;font-size:1.15rem;color:rgba(0,0,0,.85);font-weight:300;text-rendering:optimizeLegibility} +.verseblock pre strong{font-weight:400} +.verseblock .attribution{margin-top:1.25rem;margin-left:.5ex} +.quoteblock .attribution,.verseblock .attribution{font-size:.9375em;line-height:1.45;font-style:italic} +.quoteblock .attribution br,.verseblock .attribution br{display:none} +.quoteblock .attribution cite,.verseblock .attribution cite{display:block;letter-spacing:-.025em;color:rgba(0,0,0,.6)} +.quoteblock.abstract blockquote::before,.quoteblock.excerpt blockquote::before,.quoteblock .quoteblock blockquote::before{display:none} +.quoteblock.abstract blockquote,.quoteblock.abstract p,.quoteblock.excerpt blockquote,.quoteblock.excerpt p,.quoteblock .quoteblock blockquote,.quoteblock .quoteblock p{line-height:1.6;word-spacing:0} +.quoteblock.abstract{margin:0 1em 1.25em;display:block} +.quoteblock.abstract>.title{margin:0 0 .375em;font-size:1.15em;text-align:center} +.quoteblock.excerpt,.quoteblock .quoteblock{margin:0 0 1.25em;padding:0 0 .25em 1em;border-left:.25em solid #dddddf} +.quoteblock.excerpt blockquote,.quoteblock.excerpt p,.quoteblock .quoteblock blockquote,.quoteblock .quoteblock p{color:inherit;font-size:1.0625rem} +.quoteblock.excerpt .attribution,.quoteblock .quoteblock .attribution{color:inherit;text-align:left;margin-right:0} +table.tableblock{max-width:100%;border-collapse:separate} +p.tableblock:last-child{margin-bottom:0} +td.tableblock>.content{margin-bottom:-1.25em} +table.tableblock,th.tableblock,td.tableblock{border:0 solid #dedede} +table.grid-all>thead>tr>.tableblock,table.grid-all>tbody>tr>.tableblock{border-width:0 1px 1px 0} +table.grid-all>tfoot>tr>.tableblock{border-width:1px 1px 0 0} +table.grid-cols>*>tr>.tableblock{border-width:0 1px 0 0} +table.grid-rows>thead>tr>.tableblock,table.grid-rows>tbody>tr>.tableblock{border-width:0 0 1px} +table.grid-rows>tfoot>tr>.tableblock{border-width:1px 0 0} +table.grid-all>*>tr>.tableblock:last-child,table.grid-cols>*>tr>.tableblock:last-child{border-right-width:0} +table.grid-all>tbody>tr:last-child>.tableblock,table.grid-all>thead:last-child>tr>.tableblock,table.grid-rows>tbody>tr:last-child>.tableblock,table.grid-rows>thead:last-child>tr>.tableblock{border-bottom-width:0} +table.frame-all{border-width:1px} +table.frame-sides{border-width:0 1px} +table.frame-topbot,table.frame-ends{border-width:1px 0} +table.stripes-all tr,table.stripes-odd tr:nth-of-type(odd){background:#f8f8f7} +table.stripes-none tr,table.stripes-odd tr:nth-of-type(even){background:none} +th.halign-left,td.halign-left{text-align:left} +th.halign-right,td.halign-right{text-align:right} +th.halign-center,td.halign-center{text-align:center} +th.valign-top,td.valign-top{vertical-align:top} +th.valign-bottom,td.valign-bottom{vertical-align:bottom} +th.valign-middle,td.valign-middle{vertical-align:middle} +table thead th,table tfoot th{font-weight:bold} +tbody tr th{display:table-cell;line-height:1.6;background:#f7f8f7} +tbody tr th,tbody tr th p,tfoot tr th,tfoot tr th p{color:rgba(0,0,0,.8);font-weight:bold} +p.tableblock>code:only-child{background:none;padding:0} +p.tableblock{font-size:1em} +td>div.verse{white-space:pre} +ol{margin-left:1.75em} +ul li ol{margin-left:1.5em} +dl dd{margin-left:1.125em} +dl dd:last-child,dl dd:last-child>:last-child{margin-bottom:0} +ol>li p,ul>li p,ul dd,ol dd,.olist .olist,.ulist .ulist,.ulist .olist,.olist .ulist{margin-bottom:.625em} +ul.checklist,ul.none,ol.none,ul.no-bullet,ol.no-bullet,ol.unnumbered,ul.unstyled,ol.unstyled{list-style-type:none} +ul.no-bullet,ol.no-bullet,ol.unnumbered{margin-left:.625em} +ul.unstyled,ol.unstyled{margin-left:0} +ul.checklist{margin-left:.625em} +ul.checklist li>p:first-child>.fa-square-o:first-child,ul.checklist li>p:first-child>.fa-check-square-o:first-child{width:1.25em;font-size:.8em;position:relative;bottom:.125em} +ul.checklist li>p:first-child>input[type="checkbox"]:first-child{margin-right:.25em} +ul.inline{display:-ms-flexbox;display:-webkit-box;display:flex;-ms-flex-flow:row wrap;-webkit-flex-flow:row wrap;flex-flow:row wrap;list-style:none;margin:0 0 .625em -1.25em} +ul.inline>li{margin-left:1.25em} +.unstyled dl dt{font-weight:400;font-style:normal} +ol.arabic{list-style-type:decimal} +ol.decimal{list-style-type:decimal-leading-zero} +ol.loweralpha{list-style-type:lower-alpha} +ol.upperalpha{list-style-type:upper-alpha} +ol.lowerroman{list-style-type:lower-roman} +ol.upperroman{list-style-type:upper-roman} +ol.lowergreek{list-style-type:lower-greek} +.hdlist>table,.colist>table{border:0;background:none} +.hdlist>table>tbody>tr,.colist>table>tbody>tr{background:none} +td.hdlist1,td.hdlist2{vertical-align:top;padding:0 .625em} +td.hdlist1{font-weight:bold;padding-bottom:1.25em} +.literalblock+.colist,.listingblock+.colist{margin-top:-.5em} +.colist td:not([class]):first-child{padding:.4em .75em 0;line-height:1;vertical-align:top} +.colist td:not([class]):first-child img{max-width:none} +.colist td:not([class]):last-child{padding:.25em 0} +.thumb,.th{line-height:0;display:inline-block;border:solid 4px #fff;-webkit-box-shadow:0 0 0 1px #ddd;box-shadow:0 0 0 1px #ddd} +.imageblock.left{margin:.25em .625em 1.25em 0} +.imageblock.right{margin:.25em 0 1.25em .625em} +.imageblock>.title{margin-bottom:0} +.imageblock.thumb,.imageblock.th{border-width:6px} +.imageblock.thumb>.title,.imageblock.th>.title{padding:0 .125em} +.image.left,.image.right{margin-top:.25em;margin-bottom:.25em;display:inline-block;line-height:0} +.image.left{margin-right:.625em} +.image.right{margin-left:.625em} +a.image{text-decoration:none;display:inline-block} +a.image object{pointer-events:none} +sup.footnote,sup.footnoteref{font-size:.875em;position:static;vertical-align:super} +sup.footnote a,sup.footnoteref a{text-decoration:none} +sup.footnote a:active,sup.footnoteref a:active{text-decoration:underline} +#footnotes{padding-top:.75em;padding-bottom:.75em;margin-bottom:.625em} +#footnotes hr{width:20%;min-width:6.25em;margin:-.25em 0 .75em;border-width:1px 0 0} +#footnotes .footnote{padding:0 .375em 0 .225em;line-height:1.3334;font-size:.875em;margin-left:1.2em;margin-bottom:.2em} +#footnotes .footnote a:first-of-type{font-weight:bold;text-decoration:none;margin-left:-1.05em} +#footnotes .footnote:last-of-type{margin-bottom:0} +#content #footnotes{margin-top:-.625em;margin-bottom:0;padding:.75em 0} +.gist .file-data>table{border:0;background:#fff;width:100%;margin-bottom:0} +.gist .file-data>table td.line-data{width:99%} +div.unbreakable{page-break-inside:avoid} +.big{font-size:larger} +.small{font-size:smaller} +.underline{text-decoration:underline} +.overline{text-decoration:overline} +.line-through{text-decoration:line-through} +.aqua{color:#00bfbf} +.aqua-background{background-color:#00fafa} +.black{color:#000} +.black-background{background-color:#000} +.blue{color:#0000bf} +.blue-background{background-color:#0000fa} +.fuchsia{color:#bf00bf} +.fuchsia-background{background-color:#fa00fa} +.gray{color:#606060} +.gray-background{background-color:#7d7d7d} +.green{color:#006000} +.green-background{background-color:#007d00} +.lime{color:#00bf00} +.lime-background{background-color:#00fa00} +.maroon{color:#600000} +.maroon-background{background-color:#7d0000} +.navy{color:#000060} +.navy-background{background-color:#00007d} +.olive{color:#606000} +.olive-background{background-color:#7d7d00} +.purple{color:#600060} +.purple-background{background-color:#7d007d} +.red{color:#bf0000} +.red-background{background-color:#fa0000} +.silver{color:#909090} +.silver-background{background-color:#bcbcbc} +.teal{color:#006060} +.teal-background{background-color:#007d7d} +.white{color:#bfbfbf} +.white-background{background-color:#fafafa} +.yellow{color:#bfbf00} +.yellow-background{background-color:#fafa00} +span.icon>.fa{cursor:default} +a span.icon>.fa{cursor:inherit} +.admonitionblock td.icon [class^="fa icon-"]{font-size:2.5em;text-shadow:1px 1px 2px rgba(0,0,0,.5);cursor:default} +.admonitionblock td.icon .icon-note::before{content:"\f05a";color:#19407c} +.admonitionblock td.icon .icon-tip::before{content:"\f0eb";text-shadow:1px 1px 2px rgba(155,155,0,.8);color:#111} +.admonitionblock td.icon .icon-warning::before{content:"\f071";color:#bf6900} +.admonitionblock td.icon .icon-caution::before{content:"\f06d";color:#bf3400} +.admonitionblock td.icon .icon-important::before{content:"\f06a";color:#bf0000} +.conum[data-value]{display:inline-block;color:#fff!important;background-color:rgba(0,0,0,.8);-webkit-border-radius:100px;border-radius:100px;text-align:center;font-size:.75em;width:1.67em;height:1.67em;line-height:1.67em;font-family:"Open Sans","DejaVu Sans",sans-serif;font-style:normal;font-weight:bold} +.conum[data-value] *{color:#fff!important} +.conum[data-value]+b{display:none} +.conum[data-value]::after{content:attr(data-value)} +pre .conum[data-value]{position:relative;top:-.125em} +b.conum *{color:inherit!important} +.conum:not([data-value]):empty{display:none} +dt,th.tableblock,td.content,div.footnote{text-rendering:optimizeLegibility} +h1,h2,p,td.content,span.alt{letter-spacing:-.01em} +p strong,td.content strong,div.footnote strong{letter-spacing:-.005em} +p,blockquote,dt,td.content,span.alt{font-size:1.0625rem} +p{margin-bottom:1.25rem} +.sidebarblock p,.sidebarblock dt,.sidebarblock td.content,p.tableblock{font-size:1em} +.exampleblock>.content{background-color:#fffef7;border-color:#e0e0dc;-webkit-box-shadow:0 1px 4px #e0e0dc;box-shadow:0 1px 4px #e0e0dc} +.print-only{display:none!important} +@page{margin:1.25cm .75cm} +@media print{*{-webkit-box-shadow:none!important;box-shadow:none!important;text-shadow:none!important} +html{font-size:80%} +a{color:inherit!important;text-decoration:underline!important} +a.bare,a[href^="#"],a[href^="mailto:"]{text-decoration:none!important} +a[href^="http:"]:not(.bare)::after,a[href^="https:"]:not(.bare)::after{content:"(" attr(href) ")";display:inline-block;font-size:.875em;padding-left:.25em} +abbr[title]::after{content:" (" attr(title) ")"} +pre,blockquote,tr,img,object,svg{page-break-inside:avoid} +thead{display:table-header-group} +svg{max-width:100%} +p,blockquote,dt,td.content{font-size:1em;orphans:3;widows:3} +h2,h3,#toctitle,.sidebarblock>.content>.title{page-break-after:avoid} +#toc,.sidebarblock,.exampleblock>.content{background:none!important} +#toc{border-bottom:1px solid #dddddf!important;padding-bottom:0!important} +body.book #header{text-align:center} +body.book #header>h1:first-child{border:0!important;margin:2.5em 0 1em} +body.book #header .details{border:0!important;display:block;padding:0!important} +body.book #header .details span:first-child{margin-left:0!important} +body.book #header .details br{display:block} +body.book #header .details br+span::before{content:none!important} +body.book #toc{border:0!important;text-align:left!important;padding:0!important;margin:0!important} +body.book #toc,body.book #preamble,body.book h1.sect0,body.book .sect1>h2{page-break-before:always} +.listingblock code[data-lang]::before{display:block} +#footer{padding:0 .9375em} +.hide-on-print{display:none!important} +.print-only{display:block!important} +.hide-for-print{display:none!important} +.show-for-print{display:inherit!important}} +@media print,amzn-kf8{#header>h1:first-child{margin-top:1.25rem} +.sect1{padding:0!important} +.sect1+.sect1{border:0} +#footer{background:none} +#footer-text{color:rgba(0,0,0,.6);font-size:.9em}} +@media amzn-kf8{#header,#content,#footnotes,#footer{padding:0}} +</style> +<style> +/* Stylesheet for CodeRay to match GitHub theme | MIT License | http://foundation.zurb.com */ +/*pre.CodeRay {background-color:#f7f7f8;}*/ +.CodeRay .line-numbers{border-right:1px solid #d8d8d8;padding:0 0.5em 0 .25em} +.CodeRay span.line-numbers{display:inline-block;margin-right:.5em;color:rgba(0,0,0,.3)} +.CodeRay .line-numbers strong{color:rgba(0,0,0,.4)} +table.CodeRay{border-collapse:separate;border-spacing:0;margin-bottom:0;border:0;background:none} +table.CodeRay td{vertical-align: top;line-height:1.45} +table.CodeRay td.line-numbers{text-align:right} +table.CodeRay td.line-numbers>pre{padding:0;color:rgba(0,0,0,.3)} +table.CodeRay td.code{padding:0 0 0 .5em} +table.CodeRay td.code>pre{padding:0} +.CodeRay .debug{color:#fff !important;background:#000080 !important} +.CodeRay .annotation{color:#007} +.CodeRay .attribute-name{color:#000080} +.CodeRay .attribute-value{color:#700} +.CodeRay .binary{color:#509} +.CodeRay .comment{color:#998;font-style:italic} +.CodeRay .char{color:#04d} +.CodeRay .char .content{color:#04d} +.CodeRay .char .delimiter{color:#039} +.CodeRay .class{color:#458;font-weight:bold} +.CodeRay .complex{color:#a08} +.CodeRay .constant,.CodeRay .predefined-constant{color:#008080} +.CodeRay .color{color:#099} +.CodeRay .class-variable{color:#369} +.CodeRay .decorator{color:#b0b} +.CodeRay .definition{color:#099} +.CodeRay .delimiter{color:#000} +.CodeRay .doc{color:#970} +.CodeRay .doctype{color:#34b} +.CodeRay .doc-string{color:#d42} +.CodeRay .escape{color:#666} +.CodeRay .entity{color:#800} +.CodeRay .error{color:#808} +.CodeRay .exception{color:inherit} +.CodeRay .filename{color:#099} +.CodeRay .function{color:#900;font-weight:bold} +.CodeRay .global-variable{color:#008080} +.CodeRay .hex{color:#058} +.CodeRay .integer,.CodeRay .float{color:#099} +.CodeRay .include{color:#555} +.CodeRay .inline{color:#000} +.CodeRay .inline .inline{background:#ccc} +.CodeRay .inline .inline .inline{background:#bbb} +.CodeRay .inline .inline-delimiter{color:#d14} +.CodeRay .inline-delimiter{color:#d14} +.CodeRay .important{color:#555;font-weight:bold} +.CodeRay .interpreted{color:#b2b} +.CodeRay .instance-variable{color:#008080} +.CodeRay .label{color:#970} +.CodeRay .local-variable{color:#963} +.CodeRay .octal{color:#40e} +.CodeRay .predefined{color:#369} +.CodeRay .preprocessor{color:#579} +.CodeRay .pseudo-class{color:#555} +.CodeRay .directive{font-weight:bold} +.CodeRay .type{font-weight:bold} +.CodeRay .predefined-type{color:inherit} +.CodeRay .reserved,.CodeRay .keyword {color:#000;font-weight:bold} +.CodeRay .key{color:#808} +.CodeRay .key .delimiter{color:#606} +.CodeRay .key .char{color:#80f} +.CodeRay .value{color:#088} +.CodeRay .regexp .delimiter{color:#808} +.CodeRay .regexp .content{color:#808} +.CodeRay .regexp .modifier{color:#808} +.CodeRay .regexp .char{color:#d14} +.CodeRay .regexp .function{color:#404;font-weight:bold} +.CodeRay .string{color:#d20} +.CodeRay .string .string .string{background:#ffd0d0} +.CodeRay .string .content{color:#d14} +.CodeRay .string .char{color:#d14} +.CodeRay .string .delimiter{color:#d14} +.CodeRay .shell{color:#d14} +.CodeRay .shell .delimiter{color:#d14} +.CodeRay .symbol{color:#990073} +.CodeRay .symbol .content{color:#a60} +.CodeRay .symbol .delimiter{color:#630} +.CodeRay .tag{color:#008080} +.CodeRay .tag-special{color:#d70} +.CodeRay .variable{color:#036} +.CodeRay .insert{background:#afa} +.CodeRay .delete{background:#faa} +.CodeRay .change{color:#aaf;background:#007} +.CodeRay .head{color:#f8f;background:#505} +.CodeRay .insert .insert{color:#080} +.CodeRay .delete .delete{color:#800} +.CodeRay .change .change{color:#66f} +.CodeRay .head .head{color:#f4f} +</style> +</head> +<body class="article"> +<div id="header"> +<div id="toc" class="toc"> +<div id="toctitle">Table of Contents</div> +<ul class="sectlevel1"> +<li><a href="#_accessing_more_io">1. Accessing More I/O</a> +<ul class="sectlevel2"> +<li><a href="#_editing_bootuenv_txt_to_access_the_p8_header_on_the_black">1.1. Editing /boot/uEnv.txt to Access the P8 Header on the Black</a></li> +<li><a href="#_accessing_gpio">1.2. Accessing gpio</a></li> +<li><a href="#io_uio">1.3. Configuring for UIO Instead of RemoteProc</a></li> +<li><a href="#_converting_pasm_assembly_code_to_clpru">1.4. Converting pasm Assembly Code to clpru</a></li> +</ul> +</li> +</ul> +</div> +</div> +<div id="content"> +<div class="paragraph"> +<p><a href="../index.html">Outline</a></p> +</div> +<div class="sect1"> +<h2 id="_accessing_more_io"><a class="link" href="#_accessing_more_io">1. Accessing More I/O</a></h2> +<div class="sectionbody"> +<div class="paragraph"> +<p>So far the examples have shown how to access the GPIO pins on the BeagleBone Black’s +<code>P9</code> header and through the <code>__R30</code> register. Below shows how more GPIO pins +can be accessed.</p> +</div> +<div class="paragraph"> +<p>The following are resources used in this chapter.</p> +</div> +<div class="ulist"> +<div class="title">Resources</div> +<ul> +<li> +<p><a href="https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP8HeaderTable.pdf">P8 Header Table</a></p> +</li> +<li> +<p><a href="https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP9HeaderTable.pdf">P9 Header Table</a></p> +</li> +<li> +<p><a href="http://www.ti.com/lit/pdf/spruhz6l">AM572x Technical Reference Manual</a> (AI)</p> +</li> +<li> +<p><a href="http://www.ti.com/lit/pdf/spruh73">AM335x Technical Reference Manual</a> (All others)</p> +</li> +<li> +<p><a href="http://www.ti.com/lit/ug/spruhv6a/spruhv6a.pdf">PRU Assembly Language Tools</a></p> +</li> +</ul> +</div> +<div class="sect2"> +<h3 id="_editing_bootuenv_txt_to_access_the_p8_header_on_the_black"><a class="link" href="#_editing_bootuenv_txt_to_access_the_p8_header_on_the_black">1.1. Editing /boot/uEnv.txt to Access the P8 Header on the Black</a></h3> +<div class="sect3"> +<h4 id="_problem"><a class="link" href="#_problem">Problem</a></h4> +<div class="paragraph"> +<p>When I try to configure some pins on the <code>P8</code> header of the Black I get an error.</p> +</div> +<div class="listingblock"> +<div class="title">config-pin</div> +<div class="content"> +<pre class="CodeRay highlight"><code data-lang="bash">bone$ <strong>config-pin P8_28 pruout</strong> +ERROR: open() for /sys/devices/platform/ocp/ocp:P8_28_pinmux/state failed, No such file or directory</code></pre> +</div> +</div> +</div> +<div class="sect3"> +<h4 id="_solution"><a class="link" href="#_solution">Solution</a></h4> +<div class="paragraph"> +<p>On the images for the BeagleBone Black, the HDMI display driver is enabled by +default and uses many of the <code>P8</code> pins. If you are not using +HDMI video (or the HDI audio, or even the eMMC) you can disable it by editing +<code>/boot/uEnv.txt</code></p> +</div> +<div class="paragraph"> +<p>Open <code>/boot/uEnv.txt</code> and scroll down aways until you see:</p> +</div> +<div class="listingblock"> +<div class="title">/boot/uEnv.txt</div> +<div class="content"> +<pre class="CodeRay highlight"><code data-lang="bash">###Disable auto loading of virtual capes (emmc/video/wireless/adc) +#disable_uboot_overlay_emmc=1 +disable_uboot_overlay_video=1 +#disable_uboot_overlay_audio=1</code></pre> +</div> +</div> +<div class="paragraph"> +<p>Uncomment the lines that correspond to the devices you want to disable and +free up their pins.</p> +</div> +<div class="admonitionblock tip"> +<table> +<tr> +<td class="icon"> +<div class="title">Tip</div> +</td> +<td class="content"> +<div class="paragraph"> +<p><a href="https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP8HeaderTable.pdf">P8 Header Table</a> +shows what pins are allocated for what.</p> +</div> +</td> +</tr> +</table> +</div> +<div class="paragraph"> +<p>Save the file and reboot. You now have access to the <code>P8</code> pins.</p> +</div> +</div> +</div> +<div class="sect2"> +<h3 id="_accessing_gpio"><a class="link" href="#_accessing_gpio">1.2. Accessing gpio</a></h3> +<div class="sect3"> +<h4 id="_problem_2"><a class="link" href="#_problem_2">Problem</a></h4> +<div class="paragraph"> +<p>I’ve used up all the GPIO in <code>__R30</code>, where can I get more?</p> +</div> +</div> +<div class="sect3"> +<h4 id="_solution_2"><a class="link" href="#_solution_2">Solution</a></h4> +<div class="paragraph"> +<p>So far we have focused on using PRU 0. +<a href="../05blocks/blocks.html#blocks_mapping_bits">Mapping bit positions to pin names +PRU</a> shows +that PRU 0 can access ten GPIO pins on the BeagleBone Black. If you use +PRU 1 you can get to an additional 14 pins (if they aren’t in use for other things.)</p> +</div> +<div class="paragraph"> +<p>What if you need even more GPIO pins? You can access <em>any</em> GPIO pin by going +through the <strong>O</strong>pen-<strong>C</strong>ore <strong>P</strong>rotocol (OCP) port.</p> +</div> +<div class="paragraph"> +<div class="title">PRU Integration</div> +<p><span class="image"><img src="figures/pruIntegration.png" alt="PRU Integration"></span></p> +</div> +<div class="paragraph"> +<p>The figure above shows we’ve been using the <em>Enhanced GPIO</em> interface when using +<code>__R30</code>, but it also shows you can use the OCP. You get access to many more +GPIO pins, but it’s a slower access.</p> +</div> +<div class="listingblock"> +<div class="title">gpio.pru0.c</div> +<div class="content"> +<pre class="CodeRay highlight"><code data-lang="c"><table class="CodeRay"><tr> + <td class="line-numbers"><pre>1 +2 +3 +4 +5 +6 +7 +8 +9 +10 +11 +12 +13 +14 +15 +16 +17 +18 +19 +20 +21 +22 +23 +</pre></td> + <td class="code"><pre><span class="comment">// This code accesses GPIO without using R30 and R31</span> +<span class="preprocessor">#include</span> <span class="include"><stdint.h></span> +<span class="preprocessor">#include</span> <span class="include"><pru_cfg.h></span> +<span class="preprocessor">#include</span> <span class="include">"resource_table_empty.h"</span> +<span class="preprocessor">#include</span> <span class="include">"prugpio.h"</span> + +<span class="preprocessor">#define</span> P9_11 (<span class="hex">0x1</span><<<span class="integer">30</span>) <span class="comment">// Bit position tied to P9_11 on Black</span> +<span class="preprocessor">#define</span> P2_05 (<span class="hex">0x1</span><<<span class="integer">30</span>) <span class="comment">// Bit position tied to P2_05 on Pocket</span> + +<span class="directive">volatile</span> <span class="directive">register</span> uint32_t __R30; +<span class="directive">volatile</span> <span class="directive">register</span> uint32_t __R31; + +<span class="directive">void</span> main(<span class="directive">void</span>) +{ + uint32_t *gpio0 = (uint32_t *)GPIO0; + + <span class="keyword">while</span>(<span class="integer">1</span>) { + gpio0[GPIO_SETDATAOUT] = P9_11; + __delay_cycles(<span class="integer">100000000</span>); + gpio0[GPIO_CLEARDATAOUT] = P9_11; + __delay_cycles(<span class="integer">100000000</span>); + } +}</pre></td> +</tr></table></code></pre> +</div> +</div> +<div class="paragraph"> +<p>This code will toggle <code>P9_11</code> on and off. Here’s the setup file.</p> +</div> +<div class="listingblock"> +<div class="title">setup.sh</div> +<div class="content"> +<pre class="CodeRay highlight"><code data-lang="bash"><table class="CodeRay"><tr> + <td class="line-numbers"><pre>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 +</pre></td> + <td class="code"><pre>#!/bin/bash + +export TARGET=gpio.pru0 +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_11" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P2_05" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin gpio + config-pin -q $pin +done</pre></td> +</tr></table></code></pre> +</div> +</div> +<div class="paragraph"> +<p>Notice in the code <code>config-pin</code> set <code>P9_11</code> to <code>gpio</code>, not <code>pruout</code>. This is because +are are using the OCP interface to the pin, not the usual PRU interface.</p> +</div> +<div class="paragraph"> +<p>Set your exports and make.</p> +</div> +<div class="listingblock"> +<div class="content"> +<pre class="CodeRay highlight"><code data-lang="bash">bone$ <strong>source setup.sh</strong> +TARGET=gpio.pru0 +... +bone$ <strong>make</strong> +/var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=gpio.pru0 +- Stopping PRU 0 +- copying firmware file /tmp/cloud9-examples/gpio.pru0.out to /lib/firmware/am335x-pru0-fw +write_init_pins.sh +- Starting PRU 0 +MODEL = TI_AM335x_BeagleBone_Black +PROC = pru +PRUN = 0 +PRU_DIR = /sys/class/remoteproc/remoteproc1</code></pre> +</div> +</div> +</div> +<div class="sect3"> +<h4 id="_discussion"><a class="link" href="#_discussion">Discussion</a></h4> +<div class="paragraph"> +<p>When you run the code you see <code>P9_11</code> toggling on and off. Let’s go through +the code line-by-line to see what’s happening.</p> +</div> +<table class="tableblock frame-all grid-all stretch"> +<caption class="title">Table 1. gpio.pru0.c line-by-line</caption> +<colgroup> +<col style="width: 10%;"> +<col style="width: 90%;"> +</colgroup> +<thead> +<tr> +<th class="tableblock halign-left valign-top">Line</th> +<th class="tableblock halign-left valign-top">Explanation</th> +</tr> +</thead> +<tbody> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">2-5</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">Standard includes</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">5</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">The AM335x has four 32-bit GPIO ports. Lines 55-58 of <code>prugpio.h</code> define the addresses +for each of the ports. You can find these in Table 2-2 page 180 of the +<a href="https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf">AM335x Technical Reference Manual</a>. +Look up <code>P9_11</code> in the <a href="https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP9HeaderTable.pdf">P9 Header Table</a>. +Under the <em>Mode7</em> column you see <code>gpio0[30]</code>. This means <code>P9_11</code> is bit 30 +on GPIO port 0. Therefore we will use <code>GPIO0</code> in this code.</p> +<p class="tableblock">You can also run +<code>gpioinfo</code> and look for P9_11.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">5</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">Line 103 of <code>prugpio.h</code> defines the address offset from <code>GIO0</code> that will +allow us to <em>clear</em> +any (or all) bits in GPIO port 0. Other architectures require you to read a port, +then change some bit, then write it out again, three steps. Here we can do the same by writing to one location, just one step.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">5</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">Line 104 of <code>prugpio.h</code> is like above, but for <em>setting</em> bits.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">5</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">Using this offset of line 105 of <code>prugpio.h</code> lets us just read the bits +without changing them.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">7,8</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">This shifts <code>0x1</code> to the 30<sup>th</sup> bit position, which is the one corresponding +to <code>P9_11</code>.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">15</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">Here we initialize <code>gpio0</code> to point to the start of GPIO port 0’s control +registers.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">18</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock"><code>gpio0[GPIO_SETDATAOUT]</code> refers to the <code>SETDATAOUT</code> register of port 0. +Writing to this register turns on the bits where 1’s are written, +but leaves alone the bits where 0’s are.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">19</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">Wait 100,000,000 cycles, which is 0.5 seconds.</p></td> +</tr> +<tr> +<td class="tableblock halign-left valign-top"><p class="tableblock">20</p></td> +<td class="tableblock halign-left valign-top"><p class="tableblock">This is line 18, but the output bit is set to 0 where 1’s are written.</p></td> +</tr> +</tbody> +</table> +<div class="sect4"> +<h5 id="_how_fast_can_it_go"><a class="link" href="#_how_fast_can_it_go">How fast can it go?</a></h5> +<div class="paragraph"> +<p>This approach to GPIO goes through the slower OCP interface. If you set <code>__delay_cycles(0)</code> you can see how fast it is.</p> +</div> +<div class="paragraph"> +<div class="title">gpio.pru0.c with __delay_cycles(0)</div> +<p><span class="image"><img src="figures/gpio0delay.png" alt="gpio.pru0.c with __delay_cycles(0)"></span></p> +</div> +<div class="paragraph"> +<p>The period is 80ns which is 12.MHz. That’s about one forth the speed of the +<code>__R30</code> method, but still not bad.</p> +</div> +<div class="paragraph"> +<p>If you are using an oscilloscope, look closely and you’ll see the following.</p> +</div> +<div class="paragraph"> +<div class="title">PWM with jitter</div> +<p><span class="image"><img src="figures/jitter.png" alt="PWM with jitter"></span></p> +</div> +<div class="paragraph"> +<p>The PRU is still as solid as before in it’s timing, but now it’s going through +the OCP interface. This interface is shared with other parts of the system, +therefore the sometimes the PRU must wait for the other parts to finish. +When this happens the pulse width is a bit longer than usual thus adding +jitter to the output.</p> +</div> +<div class="paragraph"> +<p>For many applications a few nanoseconds of jitter is unimportant and this +GPIO interface can be used. If your application needs better timing, +use the <code>__R30</code> interface.</p> +</div> +</div> +</div> +</div> +<div class="sect2"> +<h3 id="io_uio"><a class="link" href="#io_uio">1.3. Configuring for UIO Instead of RemoteProc</a></h3> +<div class="sect3"> +<h4 id="_problem_3"><a class="link" href="#_problem_3">Problem</a></h4> +<div class="paragraph"> +<p>You have some legacy PRU code that uses UIO instead of remoteproc and +you want to switch to UIO.</p> +</div> +</div> +<div class="sect3"> +<h4 id="_solution_3"><a class="link" href="#_solution_3">Solution</a></h4> +<div class="paragraph"> +<p>Edit <code>/boot/uEnt.txt</code> and search for <code>uio</code>. I find</p> +</div> +<div class="listingblock"> +<div class="content"> +<pre>###pru_uio (4.4.x-ti, 4.9.x-ti, 4.14.x-ti & mainline/bone kernel) +uboot_overlay_pru=/lib/firmware/AM335X-PRU-UIO-00A0.dtbo</pre> +</div> +</div> +<div class="paragraph"> +<p>Uncomment the <code>uboot</code> line. Look for other lines with +<code>uboot_overlay_pru=</code> and be sure they are commented out.</p> +</div> +<div class="paragraph"> +<p>Reboot your Bone.</p> +</div> +<div class="listingblock"> +<div class="content"> +<pre>bone$ <strong>sudo reboot</strong></pre> +</div> +</div> +<div class="paragraph"> +<p>Check that UIO is running.</p> +</div> +<div class="listingblock"> +<div class="content"> +<pre>bone$ <strong>lsmod | grep uio</strong> +uio_pruss 16384 0 +uio_pdrv_genirq 16384 0 +uio 20480 2 uio_pruss,uio_pdrv_genirq</pre> +</div> +</div> +<div class="paragraph"> +<p>You are now ready to run the legacy PRU code.</p> +</div> +</div> +</div> +<div class="sect2"> +<h3 id="_converting_pasm_assembly_code_to_clpru"><a class="link" href="#_converting_pasm_assembly_code_to_clpru">1.4. Converting pasm Assembly Code to clpru</a></h3> +<div class="sect3"> +<h4 id="_problem_4"><a class="link" href="#_problem_4">Problem</a></h4> +<div class="paragraph"> +<p>You have some legacy assembly code written in pasm and it won’t assemble +with clpru.</p> +</div> +</div> +<div class="sect3"> +<h4 id="_solution_4"><a class="link" href="#_solution_4">Solution</a></h4> +<div class="paragraph"> +<p>Generally there is a simple mapping from pasm to clpru. +<a href="http://processors.wiki.ti.com/index.php/PRU_Assembly_Instructions#pasm_vs._clpru">pasm vs. clpru</a> +notes what needs to be changed. I have a less complete version on my +<a href="https://elinux.org/EBC_Exercise_30_PRU_porting_pasm_to_clpru">eLinux.org site</a>.</p> +</div> +</div> +<div class="sect3"> +<h4 id="_discussion_2"><a class="link" href="#_discussion_2">Discussion</a></h4> +<div class="paragraph"> +<p>The clpru assembly can be found in <a href="http://www.ti.com/lit/ug/spruhv6a/spruhv6a.pdf">PRU Assembly Language Tools</a>.</p> +</div> +</div> +</div> +</div> +</div> +</div> +<div id="footer"> +<div id="footer-text"> +Last updated 2021-08-04 14:38:24 -0400 +</div> +</div> +</body> +</html> \ No newline at end of file diff --git a/pru-cookbook/06io/io.rst b/pru-cookbook/06io/io.rst new file mode 100644 index 0000000000000000000000000000000000000000..3548d5e0a4330f10a7a06d4f94856bac5ba1a89c --- /dev/null +++ b/pru-cookbook/06io/io.rst @@ -0,0 +1,271 @@ +.. _pru-cookbook-io: + +Accessing More I/O +#################### + +So far the examples have shown how to access the GPIO pins on the BeagleBone Black's +``P9`` header and through the ``pass:[__]R30`` register. Below shows how more GPIO pins +can be accessed. + +The following are resources used in this chapter. + +Resources +~~~~~~~~~~ + +* `P8 Header Table <https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP8HeaderTable.pdf>`_ +* `P9 Header Table <https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP9HeaderTable.pdf>`_ +* `AM572x Technical Reference Manual<http://www.ti.com/lit/pdf/spruhz6l>`_ (AI) +* `AM335x Technical Reference Manual <http://www.ti.com/lit/pdf/spruh73>`_ (All others) +* `PRU Assembly Language Tools<http://www.ti.com/lit/ug/spruhv6a/spruhv6a.pdf>`_ + +Editing /boot/uEnv.txt to Access the P8 Header on the Black +============================================================ + +Problem +-------- + +When I try to configure some pins on the `P8` header of the Black I get an error. + +config-pin +~~~~~~~~~~~ + +.. code-block:: bash + :linenos: + + bone$ *config-pin P8_28 pruout* + ERROR: open() for /sys/devices/platform/ocp/ocp:P8_28_pinmux/state failed, No such file or directory + +Solution +-------- + +On the images for the BeagleBone Black, the HDMI display driver is enabled by default +and uses many of the ``P8`` pins. If you are not using HDMI video (or the HDI audio, +or even the eMMC) you can disable it by editing ``/boot/uEnv.txt`` + +Open ``/boot/uEnv.txt`` and scroll down aways until you see: + +/boot/uEnv.txt +~~~~~~~~~~~~~~~~ + +.. code-block:: bash + :linenos: + + ###Disable auto loading of virtual capes (emmc/video/wireless/adc) + #disable_uboot_overlay_emmc=1 + disable_uboot_overlay_video=1 + #disable_uboot_overlay_audio=1 + +Uncomment the lines that correspond to the devices you want to disable and free up their pins. + +.. tip:: + + `P8 Header Table <https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP8HeaderTable.pdf>`_ + shows what pins are allocated for what. + +Save the file and reboot. You now have access to the ``P8`` pins. + +Accessing gpio +================ + +Problem +-------- + +I've used up all the GPIO in ``pass:[__]R30``, where can I get more? + +Solution +-------- + +So far we have focused on using PRU 0. +:ref:`../05blocks/blocks.html#blocks_mapping_bits,Mapping bit positions to pin names PRU` shows +that PRU 0 can access ten GPIO pins on the BeagleBone Black. If you use +PRU 1 you can get to an additional 14 pins (if they aren't in use for other things.) + +What if you need even more GPIO pins? You can access **any** GPIO pin by going +through the **O**pen-**C**ore **P**rotocol (OCP) port. + +PRU Integration +~~~~~~~~~~~~~~~~~ + +.. figure:: figures/pruIntegration.png + :align: center + :alt: PRU Integration + +The figure above shows we've been using the _Enhanced **GPIO** interface when using +``pass:[__]R30``, but it also shows you can use the OCP. You get access to many more +GPIO pins, but it's a slower access. + +gpio.pru0.c +~~~~~~~~~~~~ + +:download:`gpio.pru0.c <code/gpio.pru0.c>` + +This code will toggle ``P9_11`` on and off. Here's the setup file. + +setup.sh +~~~~~~~~~ + +:download:`setup.sh <code/setup.sh>` + +Notice in the code ``config-pin`` set ``P9_11`` to ``gpio``, not ``pruout``. This is because +are are using the OCP interface to the pin, not the usual PRU interface. + +Set your exports and make. + +.. code-block:: bash + :linenos: + + bone$ *source setup.sh* + TARGET=gpio.pru0 + ... + bone$ *make* + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=gpio.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/gpio.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + +Discussion +----------- + +When you run the code you see ``P9_11`` toggling on and off. Let's go through +the code line-by-line to see what's happening. + +gpio.pru0.c line-by-line +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |Line | Explanation | + +=======+===============================================================================================================================================+ + |2-5 | Standard includes | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |5 | The AM335x has four 32-bit GPIO ports. Lines 55-58 of `prugpio.h` define the addresses | + | | for each of the ports. You can find these in Table 2-2 page 180 of the | + | | https://www.ti.com/lit/ug/spruh73p/spruh73p.pdf[AM335x Technical Reference Manual]. | + | | Look up `P9_11` in the https://github.com/derekmolloy/exploringBB/blob/master/chp06/docs/BeagleboneBlackP9HeaderTable.pdf[P9 Header Table]. | + | | Under the _Mode7_ column you see `gpio0[30]`. This means `P9_11` is bit 30 | + | | on GPIO port 0. Therefore we will use `GPIO0` in this code. You can also run ``gpioinfo`` and look for P9_11. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |5 | Line 103 of `prugpio.h` defines the address offset from `GIO0` that will allow us to _clear_ any (or all) bits in GPIO port 0. | + | | Other architectures require you to read a port, then change some bit, then write it out again, three steps. Here we can do | + | | the same by writing to one location, just one step. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |5 |Line 104 of `prugpio.h` is like above, but for _setting_ bits. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |5 |Using this offset of line 105 of `prugpio.h` lets us just read the bits without changing them. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |7,8 |This shifts `0x1` to the 30^th^ bit position, which is the one corresponding to `P9_11`. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |15 |Here we initialize `gpio0` to point to the start of GPIO port 0's control registers. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |18 |`gpio0[GPIO_SETDATAOUT]` refers to the `SETDATAOUT` register of port 0. Writing to this register turns on the bits | + | | where 1's are written, but leaves alone the bits where 0's are. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |19 |Wait 100,000,000 cycles, which is 0.5 seconds. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + |20 |This is line 18, but the output bit is set to 0 where 1's are written. | + +-------+-----------------------------------------------------------------------------------------------------------------------------------------------+ + +How fast can it go? +~~~~~~~~~~~~~~~~~~~~ + +This approach to GPIO goes through the slower OCP interface. If you set +``pass:[__]delay_cycles(0)`` you can see how fast it is. + +gpio.pru0.c with pass:[__]delay_cycles(0) +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/gpio0delay.png + :align: center + :alt: gpio.pru0.c with pass:[__]delay_cycles(0) + +The period is 80ns which is 12.MHz. That's about one forth the speed of the +``pass:[__]R30`` method, but still not bad. + +If you are using an oscilloscope, look closely and you'll see the following. + +PWM with jitter +~~~~~~~~~~~~~~~~ + +.. figure:: figures/jitter.png + :align: center + :alt: PWM with jitter + +The PRU is still as solid as before in it's timing, but now it's going through +the OCP interface. This interface is shared with other parts of the system, +therefore the sometimes the PRU must wait for the other parts to finish. +When this happens the pulse width is a bit longer than usual thus adding +jitter to the output. + +For many applications a few nanoseconds of jitter is unimportant and this +GPIO interface can be used. If your application needs better timing, +use the ``pass:[__]R30`` interface. + + +.. _io_uio: + +Configuring for UIO Instead of RemoteProc +======================================== + +Problem +-------- + +You have some legacy PRU code that uses UIO instead of remoteproc and +you want to switch to UIO. + +Solution +-------- + +Edit ``/boot/uEnt.txt`` and search for ``uio``. I find + +.. code-block:: bash + + ###pru_uio (4.4.x-ti, 4.9.x-ti, 4.14.x-ti & mainline/bone kernel) + uboot_overlay_pru=/lib/firmware/AM335X-PRU-UIO-00A0.dtbo + +Uncomment the ``uboot`` line. Look for other lines with +``uboot_overlay_pru=`` and be sure they are commented out. + +Reboot your Bone. + +.. code-block:: bash + + bone$ sudo reboot + +Check that UIO is running. + +.. code-block:: bash + + bone$ lsmod | grep uio + uio_pruss 16384 0 + uio_pdrv_genirq 16384 0 + uio 20480 2 uio_pruss,uio_pdrv_genirq + +You are now ready to run the legacy PRU code. + +Converting pasm Assembly Code to clpru +======================================== + +Problem +-------- + +You have some legacy assembly code written in pasm and it won't assemble with clpru. + +Solution +-------- + +Generally there is a simple mapping from pasm to clpru. +`pasm vs. clpru <http://processors.wiki.ti.com/index.php/PRU_Assembly_Instructions#pasm_vs._clpru>`_ +notes what needs to be changed. I have a less complete version on my +`eLinux.org site <https://elinux.org/EBC_Exercise_30_PRU_porting_pasm_to_clpru>`_. + +Discussion +-------- + +The clpru assembly can be found in +`PRU Assembly Language Tools <http://www.ti.com/lit/ug/spruhv6a/spruhv6a.pdf>`_. diff --git a/pru-cookbook/07more/code/Makefile b/pru-cookbook/07more/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/07more/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/07more/code/copyright.c b/pru-cookbook/07more/code/copyright.c new file mode 100644 index 0000000000000000000000000000000000000000..63b6a419e4ee3a2767de3fcec0c29e74e3754700 --- /dev/null +++ b/pru-cookbook/07more/code/copyright.c @@ -0,0 +1,33 @@ +/* + * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + \ No newline at end of file diff --git a/pru-cookbook/07more/code/cycle.pru0.c b/pru-cookbook/07more/code/cycle.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..73055d9f0662483f6e9cc03ef266b670f571e1f0 --- /dev/null +++ b/pru-cookbook/07more/code/cycle.pru0.c @@ -0,0 +1,32 @@ +// Access the CYCLE and STALL registers +#include <stdint.h> +#include <pru_cfg.h> +#include <pru_ctrl.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_31; // Select which pin to toggle.; + + // These will be kept in registers and never witten to DRAM + uint32_t cycle, stall; + + // Clear SYSCFG[STANDBY_INIT] to enable OCP master port + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + PRU0_CTRL.CTRL_bit.CTR_EN = 1; // Enable cycle counter + + __R30 |= gpio; // Set the GPIO pin to 1 + // Reset cycle counter, cycle is on the right side to force the compiler + // to put it in it's own register + PRU0_CTRL.CYCLE = cycle; + __R30 &= ~gpio; // Clear the GPIO pin + cycle = PRU0_CTRL.CYCLE; // Read cycle and store in a register + stall = PRU0_CTRL.STALL; // Ditto for stall + + __halt(); +} diff --git a/pru-cookbook/07more/code/cycle.pru0.lst b/pru-cookbook/07more/code/cycle.pru0.lst new file mode 100644 index 0000000000000000000000000000000000000000..f2e131765eaf0e5bd13c719287eeb04b82b8d2cd --- /dev/null +++ b/pru-cookbook/07more/code/cycle.pru0.lst @@ -0,0 +1,3082 @@ +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 1 + + 1;****************************************************************************** + 2;* PRU C/C++ Codegen Unix v2.1.5 * + 3;* Date/Time created: Fri Jun 5 14:14:07 2020 * + 4;****************************************************************************** + 5 .compiler_opts --abi=eabi --endian=little --hll_source=on --object_format=elf --silicon_versio + 6 + 7$C$DW$CU .dwtag DW_TAG_compile_unit + 8 .dwattr $C$DW$CU, DW_AT_name("cycle.pru0.c") + 9 .dwattr $C$DW$CU, DW_AT_producer("TI PRU C/C++ Codegen Unix v2.1.5 Copyright (c) 2012-2017 Tex + 10 .dwattr $C$DW$CU, DW_AT_TI_version(0x01) + 11 .dwattr $C$DW$CU, DW_AT_comp_dir("/home/debian/PRUCookbook/docs/07more/code") + 12 .global __PRU_CREG_PRU_CFG + 13 + 14$C$DW$1 .dwtag DW_TAG_subprogram, DW_AT_name("__halt") + 15 .dwattr $C$DW$1, DW_AT_TI_symbol_name("__halt") + 16 .dwattr $C$DW$1, DW_AT_declaration + 17 .dwattr $C$DW$1, DW_AT_external + 18 .weak ||CT_CFG|| + 19 00000000 ||CT_CFG||: .usect ".creg.PRU_CFG.noload.near",68,1 + 20$C$DW$2 .dwtag DW_TAG_variable, DW_AT_name("CT_CFG") + 21 .dwattr $C$DW$2, DW_AT_TI_symbol_name("CT_CFG") + 22 .dwattr $C$DW$2, DW_AT_location[DW_OP_addr ||CT_CFG||] + 23 .dwattr $C$DW$2, DW_AT_type(*$C$DW$T$98) + 24 .dwattr $C$DW$2, DW_AT_external + 25 .dwattr $C$DW$2, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru_ + 26 .dwattr $C$DW$2, DW_AT_decl_line(0xf2) + 27 .dwattr $C$DW$2, DW_AT_decl_column(0x17) + 28 .global ||pru_remoteproc_ResourceTable|| + 29 00000000 .sect ".resource_table:retain", RW + 30 .retain + 31 .align 1 + 32 .elfsym ||pru_remoteproc_ResourceTable||,SYM_SIZE(20) + 33 00000000 ||pru_remoteproc_ResourceTable||: + 34 00000000 00000000000001 .bits 1,32 ; pru_remoteproc_ResourceTable.base.ver @ 0 + 35 00000004 00000000000000 .bits 0,32 ; pru_remoteproc_ResourceTable.base.num @ 32 + 36 00000008 00000000000000 .bits 0,32 ; pru_remoteproc_ResourceTable.base.reserved[0] @ 64 + 37 0000000c 00000000000000 .bits 0,32 ; pru_remoteproc_ResourceTable.base.reserved[1] @ 96 + 38 00000010 00000000000000 .bits 0,32 ; pru_remoteproc_ResourceTable.offset[0] @ 128 + 39 + 40$C$DW$3 .dwtag DW_TAG_variable, DW_AT_name("pru_remoteproc_ResourceTable") + 41 .dwattr $C$DW$3, DW_AT_TI_symbol_name("pru_remoteproc_ResourceTable") + 42 .dwattr $C$DW$3, DW_AT_location[DW_OP_addr ||pru_remoteproc_ResourceTable||] + 43 .dwattr $C$DW$3, DW_AT_type(*$C$DW$T$92) + 44 .dwattr $C$DW$3, DW_AT_external + 45 .dwattr $C$DW$3, DW_AT_decl_file("/var/lib/cloud9/common/resource_table_empty.h") + 46 .dwattr $C$DW$3, DW_AT_decl_line(0x1f) + 47 .dwattr $C$DW$3, DW_AT_decl_column(0x1a) + 48; optpru /tmp/TI18yQKu8QO /tmp/TI18y6Ruqzt + 49; acpiapru -@/tmp/TI18yH9tQJL + 50 00000000 .sect ".text:main" + 51 .clink + 52 .global ||main|| + 53 + 54$C$DW$4 .dwtag DW_TAG_subprogram, DW_AT_name("main") + 55 .dwattr $C$DW$4, DW_AT_low_pc(||main||) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 2 + + 56 .dwattr $C$DW$4, DW_AT_high_pc(0x00) + 57 .dwattr $C$DW$4, DW_AT_TI_symbol_name("main") + 58 .dwattr $C$DW$4, DW_AT_external + 59 .dwattr $C$DW$4, DW_AT_TI_begin_file("cycle.pru0.c") + 60 .dwattr $C$DW$4, DW_AT_TI_begin_line(0x0d) + 61 .dwattr $C$DW$4, DW_AT_TI_begin_column(0x06) + 62 .dwattr $C$DW$4, DW_AT_decl_file("cycle.pru0.c") + 63 .dwattr $C$DW$4, DW_AT_decl_line(0x0d) + 64 .dwattr $C$DW$4, DW_AT_decl_column(0x06) + 65 .dwattr $C$DW$4, DW_AT_TI_max_frame_size(0x00) + 66 .dwpsn file "cycle.pru0.c",line 14,column 1,is_stmt,address ||main||,isa 0 + 67 + 68 .dwfde $C$DW$CIE, ||main|| + 69;---------------------------------------------------------------------- + 70; 13 | void main(void) + 71; 15 | uint32_t gpio = P9_31; // Select which pin to toggle.; + 72; 17 | // These will be kept in registers and never witten to DRAM + 73; 18 | uint32_t cycle, stall; + 74; 20 | // Clear SYSCFG[STANDBY_INIT] to enable OCP master port + 75;---------------------------------------------------------------------- + 76 + 77;*************************************************************** + 78;* FNAME: main FR SIZE: 0 * + 79;* * + 80;* FUNCTION ENVIRONMENT * + 81;* * + 82;* FUNCTION PROPERTIES * + 83;* 0 Auto, 0 SOE * + 84;*************************************************************** + 85 + 86||main||: + 87;* --------------------------------------------------------------------------* + 88;* r0_0 assigned to $O$C1 + 89;* r14_0 assigned to cycle + 90$C$DW$5 .dwtag DW_TAG_variable, DW_AT_name("cycle") + 91 .dwattr $C$DW$5, DW_AT_TI_symbol_name("cycle") + 92 .dwattr $C$DW$5, DW_AT_type(*$C$DW$T$32) + 93 .dwattr $C$DW$5, DW_AT_location[DW_OP_regx 0x38] + 94 .dwcfi cfa_offset, 0 + 95 .dwpsn file "cycle.pru0.c",line 21,column 2,is_stmt,isa 0 + 96;---------------------------------------------------------------------- + 97; 21 | CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + 98;---------------------------------------------------------------------- + 99 00000000 00000091042080! LBCO &r0, __PRU_CREG_PRU_CFG, $CSBREL(||CT_CFG||+4), 4 ; [ALU_PRU] |21| CT_CFG + 100 00000004 0000001D04E0E0 CLR r0, r0, 0x00000004 ; [ALU_PRU] |21| + 101 00000008 00000081042080! SBCO &r0, __PRU_CREG_PRU_CFG, $CSBREL(||CT_CFG||+4), 4 ; [ALU_PRU] |21| CT_CFG + 102 .dwpsn file "cycle.pru0.c",line 23,column 2,is_stmt,isa 0 + 103;---------------------------------------------------------------------- + 104; 23 | PRU0_CTRL.CTRL_bit.CTR_EN = 1; // Enable cycle counter + 105;---------------------------------------------------------------------- + 106 0000000c 200080240002C0 LDI32 r0, 0x00022000 ; [ALU_PRU] |23| $O$C1 + 107 00000014 000000F1002081 LBBO &r1, r0, 0, 4 ; [ALU_PRU] |23| + 108 00000018 0000001F03E1E1 SET r1, r1, 0x00000003 ; [ALU_PRU] |23| + 109 0000001c 000000E1002081 SBBO &r1, r0, 0, 4 ; [ALU_PRU] |23| + 110 .dwpsn file "cycle.pru0.c",line 25,column 2,is_stmt,isa 0 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 3 + + 111;---------------------------------------------------------------------- + 112; 25 | __R30 |= gpio; // Set the GPIO pin to 1 + 113; 26 | // Reset cycle counter, cycle is on the right side to force the compile + 114; | r + 115; 27 | // to put it in it's own register + 116;---------------------------------------------------------------------- + 117 00000020 0000001F00FEFE SET r30, r30, 0x00000000 ; [ALU_PRU] |25| + 118 .dwpsn file "cycle.pru0.c",line 28,column 2,is_stmt,isa 0 + 119;---------------------------------------------------------------------- + 120; 28 | PRU0_CTRL.CYCLE = cycle; + 121;---------------------------------------------------------------------- + 122 00000024 000000E10C208E SBBO &r14, r0, 12, 4 ; [ALU_PRU] |28| $O$C1,cycle + 123 .dwpsn file "cycle.pru0.c",line 29,column 2,is_stmt,isa 0 + 124;---------------------------------------------------------------------- + 125; 29 | __R30 &= ~gpio; // Clear the GPIO pin + 126;---------------------------------------------------------------------- + 127 00000028 0000001D00FEFE CLR r30, r30, 0x00000000 ; [ALU_PRU] |29| + 128 .dwpsn file "cycle.pru0.c",line 30,column 2,is_stmt,isa 0 + 129;---------------------------------------------------------------------- + 130; 30 | cycle = PRU0_CTRL.CYCLE; // Read cycle and store in a register + 131;---------------------------------------------------------------------- + 132 0000002c 000000F10C2081 LBBO &r1, r0, 12, 4 ; [ALU_PRU] |30| $O$C1 + 133 .dwpsn file "cycle.pru0.c",line 31,column 2,is_stmt,isa 0 + 134;---------------------------------------------------------------------- + 135; 31 | stall = PRU0_CTRL.STALL; // Ditto for stall + 136;---------------------------------------------------------------------- + 137 00000030 000000F1102080 LBBO &r0, r0, 16, 4 ; [ALU_PRU] |31| $O$C1 + 138 .dwpsn file "cycle.pru0.c",line 33,column 2,is_stmt,isa 0 + 139;---------------------------------------------------------------------- + 140; 33 | __halt(); + 141;---------------------------------------------------------------------- + 142 00000034 0000002A000000 HALT ; [ALU_PRU] |33| + 143$C$DW$6 .dwtag DW_TAG_TI_branch + 144 .dwattr $C$DW$6, DW_AT_low_pc(0x00) + 145 .dwattr $C$DW$6, DW_AT_TI_return + 146 00000038 00000020C30000 JMP r3.w2 ; [ALU_PRU] + 147 .dwattr $C$DW$4, DW_AT_TI_end_file("cycle.pru0.c") + 148 .dwattr $C$DW$4, DW_AT_TI_end_line(0x22) + 149 .dwattr $C$DW$4, DW_AT_TI_end_column(0x01) + 150 .dwendentry + 151 .dwendtag $C$DW$4 + 152 + 153 + 154;****************************************************************************** + 155;* TYPE INFORMATION * + 156;****************************************************************************** + 157 + 158$C$DW$T$19 .dwtag DW_TAG_structure_type + 159 .dwattr $C$DW$T$19, DW_AT_byte_size(0x04) + 160$C$DW$7 .dwtag DW_TAG_member + 161 .dwattr $C$DW$7, DW_AT_type(*$C$DW$T$11) + 162 .dwattr $C$DW$7, DW_AT_name("REVID") + 163 .dwattr $C$DW$7, DW_AT_TI_symbol_name("REVID") + 164 .dwattr $C$DW$7, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x20) + 165 .dwattr $C$DW$7, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 4 + + 166 .dwattr $C$DW$7, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 167 .dwattr $C$DW$7, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru_ + 168 .dwattr $C$DW$7, DW_AT_decl_line(0x2d) + 169 .dwattr $C$DW$7, DW_AT_decl_column(0x0d) + 170 .dwendtag $C$DW$T$19 + 171 + 172 .dwattr $C$DW$T$19, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 173 .dwattr $C$DW$T$19, DW_AT_decl_line(0x2c) + 174 .dwattr $C$DW$T$19, DW_AT_decl_column(0x13) + 175$C$DW$T$48 .dwtag DW_TAG_volatile_type + 176 .dwattr $C$DW$T$48, DW_AT_type(*$C$DW$T$19) + 177 + 178$C$DW$T$20 .dwtag DW_TAG_structure_type + 179 .dwattr $C$DW$T$20, DW_AT_byte_size(0x04) + 180$C$DW$8 .dwtag DW_TAG_member + 181 .dwattr $C$DW$8, DW_AT_type(*$C$DW$T$11) + 182 .dwattr $C$DW$8, DW_AT_name("IDLE_MODE") + 183 .dwattr $C$DW$8, DW_AT_TI_symbol_name("IDLE_MODE") + 184 .dwattr $C$DW$8, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x02) + 185 .dwattr $C$DW$8, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 186 .dwattr $C$DW$8, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 187 .dwattr $C$DW$8, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru_ + 188 .dwattr $C$DW$8, DW_AT_decl_line(0x37) + 189 .dwattr $C$DW$8, DW_AT_decl_column(0x0d) + 190$C$DW$9 .dwtag DW_TAG_member + 191 .dwattr $C$DW$9, DW_AT_type(*$C$DW$T$11) + 192 .dwattr $C$DW$9, DW_AT_name("STANDBY_MODE") + 193 .dwattr $C$DW$9, DW_AT_TI_symbol_name("STANDBY_MODE") + 194 .dwattr $C$DW$9, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x02) + 195 .dwattr $C$DW$9, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 196 .dwattr $C$DW$9, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 197 .dwattr $C$DW$9, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru_ + 198 .dwattr $C$DW$9, DW_AT_decl_line(0x38) + 199 .dwattr $C$DW$9, DW_AT_decl_column(0x0d) + 200$C$DW$10 .dwtag DW_TAG_member + 201 .dwattr $C$DW$10, DW_AT_type(*$C$DW$T$11) + 202 .dwattr $C$DW$10, DW_AT_name("STANDBY_INIT") + 203 .dwattr $C$DW$10, DW_AT_TI_symbol_name("STANDBY_INIT") + 204 .dwattr $C$DW$10, DW_AT_bit_offset(0x1b), DW_AT_bit_size(0x01) + 205 .dwattr $C$DW$10, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 206 .dwattr $C$DW$10, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 207 .dwattr $C$DW$10, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 208 .dwattr $C$DW$10, DW_AT_decl_line(0x39) + 209 .dwattr $C$DW$10, DW_AT_decl_column(0x0d) + 210$C$DW$11 .dwtag DW_TAG_member + 211 .dwattr $C$DW$11, DW_AT_type(*$C$DW$T$11) + 212 .dwattr $C$DW$11, DW_AT_name("SUB_MWAIT") + 213 .dwattr $C$DW$11, DW_AT_TI_symbol_name("SUB_MWAIT") + 214 .dwattr $C$DW$11, DW_AT_bit_offset(0x1a), DW_AT_bit_size(0x01) + 215 .dwattr $C$DW$11, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 216 .dwattr $C$DW$11, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 217 .dwattr $C$DW$11, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 218 .dwattr $C$DW$11, DW_AT_decl_line(0x3a) + 219 .dwattr $C$DW$11, DW_AT_decl_column(0x0d) + 220$C$DW$12 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 5 + + 221 .dwattr $C$DW$12, DW_AT_type(*$C$DW$T$11) + 222 .dwattr $C$DW$12, DW_AT_name("rsvd6") + 223 .dwattr $C$DW$12, DW_AT_TI_symbol_name("rsvd6") + 224 .dwattr $C$DW$12, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x1a) + 225 .dwattr $C$DW$12, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 226 .dwattr $C$DW$12, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 227 .dwattr $C$DW$12, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 228 .dwattr $C$DW$12, DW_AT_decl_line(0x3b) + 229 .dwattr $C$DW$12, DW_AT_decl_column(0x0d) + 230 .dwendtag $C$DW$T$20 + 231 + 232 .dwattr $C$DW$T$20, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 233 .dwattr $C$DW$T$20, DW_AT_decl_line(0x36) + 234 .dwattr $C$DW$T$20, DW_AT_decl_column(0x13) + 235$C$DW$T$50 .dwtag DW_TAG_volatile_type + 236 .dwattr $C$DW$T$50, DW_AT_type(*$C$DW$T$20) + 237 + 238$C$DW$T$21 .dwtag DW_TAG_structure_type + 239 .dwattr $C$DW$T$21, DW_AT_byte_size(0x04) + 240$C$DW$13 .dwtag DW_TAG_member + 241 .dwattr $C$DW$13, DW_AT_type(*$C$DW$T$11) + 242 .dwattr $C$DW$13, DW_AT_name("PRU0_GPI_MODE") + 243 .dwattr $C$DW$13, DW_AT_TI_symbol_name("PRU0_GPI_MODE") + 244 .dwattr $C$DW$13, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x02) + 245 .dwattr $C$DW$13, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 246 .dwattr $C$DW$13, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 247 .dwattr $C$DW$13, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 248 .dwattr $C$DW$13, DW_AT_decl_line(0x45) + 249 .dwattr $C$DW$13, DW_AT_decl_column(0x0d) + 250$C$DW$14 .dwtag DW_TAG_member + 251 .dwattr $C$DW$14, DW_AT_type(*$C$DW$T$11) + 252 .dwattr $C$DW$14, DW_AT_name("PRU0_GPI_CLK_MODE") + 253 .dwattr $C$DW$14, DW_AT_TI_symbol_name("PRU0_GPI_CLK_MODE") + 254 .dwattr $C$DW$14, DW_AT_bit_offset(0x1d), DW_AT_bit_size(0x01) + 255 .dwattr $C$DW$14, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 256 .dwattr $C$DW$14, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 257 .dwattr $C$DW$14, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 258 .dwattr $C$DW$14, DW_AT_decl_line(0x46) + 259 .dwattr $C$DW$14, DW_AT_decl_column(0x0d) + 260$C$DW$15 .dwtag DW_TAG_member + 261 .dwattr $C$DW$15, DW_AT_type(*$C$DW$T$11) + 262 .dwattr $C$DW$15, DW_AT_name("PRU0_GPI_DIV0") + 263 .dwattr $C$DW$15, DW_AT_TI_symbol_name("PRU0_GPI_DIV0") + 264 .dwattr $C$DW$15, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x05) + 265 .dwattr $C$DW$15, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 266 .dwattr $C$DW$15, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 267 .dwattr $C$DW$15, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 268 .dwattr $C$DW$15, DW_AT_decl_line(0x47) + 269 .dwattr $C$DW$15, DW_AT_decl_column(0x0d) + 270$C$DW$16 .dwtag DW_TAG_member + 271 .dwattr $C$DW$16, DW_AT_type(*$C$DW$T$11) + 272 .dwattr $C$DW$16, DW_AT_name("PRU0_GPI_DIV1") + 273 .dwattr $C$DW$16, DW_AT_TI_symbol_name("PRU0_GPI_DIV1") + 274 .dwattr $C$DW$16, DW_AT_bit_offset(0x13), DW_AT_bit_size(0x05) + 275 .dwattr $C$DW$16, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 6 + + 276 .dwattr $C$DW$16, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 277 .dwattr $C$DW$16, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 278 .dwattr $C$DW$16, DW_AT_decl_line(0x48) + 279 .dwattr $C$DW$16, DW_AT_decl_column(0x0d) + 280$C$DW$17 .dwtag DW_TAG_member + 281 .dwattr $C$DW$17, DW_AT_type(*$C$DW$T$11) + 282 .dwattr $C$DW$17, DW_AT_name("PRU0_GPI_SB") + 283 .dwattr $C$DW$17, DW_AT_TI_symbol_name("PRU0_GPI_SB") + 284 .dwattr $C$DW$17, DW_AT_bit_offset(0x12), DW_AT_bit_size(0x01) + 285 .dwattr $C$DW$17, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 286 .dwattr $C$DW$17, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 287 .dwattr $C$DW$17, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 288 .dwattr $C$DW$17, DW_AT_decl_line(0x49) + 289 .dwattr $C$DW$17, DW_AT_decl_column(0x0d) + 290$C$DW$18 .dwtag DW_TAG_member + 291 .dwattr $C$DW$18, DW_AT_type(*$C$DW$T$11) + 292 .dwattr $C$DW$18, DW_AT_name("PRU0_GPO_MODE") + 293 .dwattr $C$DW$18, DW_AT_TI_symbol_name("PRU0_GPO_MODE") + 294 .dwattr $C$DW$18, DW_AT_bit_offset(0x11), DW_AT_bit_size(0x01) + 295 .dwattr $C$DW$18, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 296 .dwattr $C$DW$18, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 297 .dwattr $C$DW$18, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 298 .dwattr $C$DW$18, DW_AT_decl_line(0x4a) + 299 .dwattr $C$DW$18, DW_AT_decl_column(0x0d) + 300$C$DW$19 .dwtag DW_TAG_member + 301 .dwattr $C$DW$19, DW_AT_type(*$C$DW$T$11) + 302 .dwattr $C$DW$19, DW_AT_name("PRU0_GPO_DIV0") + 303 .dwattr $C$DW$19, DW_AT_TI_symbol_name("PRU0_GPO_DIV0") + 304 .dwattr $C$DW$19, DW_AT_bit_offset(0x0c), DW_AT_bit_size(0x05) + 305 .dwattr $C$DW$19, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 306 .dwattr $C$DW$19, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 307 .dwattr $C$DW$19, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 308 .dwattr $C$DW$19, DW_AT_decl_line(0x4b) + 309 .dwattr $C$DW$19, DW_AT_decl_column(0x0d) + 310$C$DW$20 .dwtag DW_TAG_member + 311 .dwattr $C$DW$20, DW_AT_type(*$C$DW$T$11) + 312 .dwattr $C$DW$20, DW_AT_name("PRU0_GPO_DIV1") + 313 .dwattr $C$DW$20, DW_AT_TI_symbol_name("PRU0_GPO_DIV1") + 314 .dwattr $C$DW$20, DW_AT_bit_offset(0x07), DW_AT_bit_size(0x05) + 315 .dwattr $C$DW$20, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 316 .dwattr $C$DW$20, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 317 .dwattr $C$DW$20, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 318 .dwattr $C$DW$20, DW_AT_decl_line(0x4c) + 319 .dwattr $C$DW$20, DW_AT_decl_column(0x0d) + 320$C$DW$21 .dwtag DW_TAG_member + 321 .dwattr $C$DW$21, DW_AT_type(*$C$DW$T$11) + 322 .dwattr $C$DW$21, DW_AT_name("PRU0_GPO_SH_SEL") + 323 .dwattr $C$DW$21, DW_AT_TI_symbol_name("PRU0_GPO_SH_SEL") + 324 .dwattr $C$DW$21, DW_AT_bit_offset(0x06), DW_AT_bit_size(0x01) + 325 .dwattr $C$DW$21, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 326 .dwattr $C$DW$21, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 327 .dwattr $C$DW$21, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 328 .dwattr $C$DW$21, DW_AT_decl_line(0x4d) + 329 .dwattr $C$DW$21, DW_AT_decl_column(0x0d) + 330$C$DW$22 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 7 + + 331 .dwattr $C$DW$22, DW_AT_type(*$C$DW$T$11) + 332 .dwattr $C$DW$22, DW_AT_name("rsvd26") + 333 .dwattr $C$DW$22, DW_AT_TI_symbol_name("rsvd26") + 334 .dwattr $C$DW$22, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x06) + 335 .dwattr $C$DW$22, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 336 .dwattr $C$DW$22, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 337 .dwattr $C$DW$22, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 338 .dwattr $C$DW$22, DW_AT_decl_line(0x4e) + 339 .dwattr $C$DW$22, DW_AT_decl_column(0x0d) + 340 .dwendtag $C$DW$T$21 + 341 + 342 .dwattr $C$DW$T$21, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 343 .dwattr $C$DW$T$21, DW_AT_decl_line(0x44) + 344 .dwattr $C$DW$T$21, DW_AT_decl_column(0x13) + 345$C$DW$T$52 .dwtag DW_TAG_volatile_type + 346 .dwattr $C$DW$T$52, DW_AT_type(*$C$DW$T$21) + 347 + 348$C$DW$T$22 .dwtag DW_TAG_structure_type + 349 .dwattr $C$DW$T$22, DW_AT_byte_size(0x04) + 350$C$DW$23 .dwtag DW_TAG_member + 351 .dwattr $C$DW$23, DW_AT_type(*$C$DW$T$11) + 352 .dwattr $C$DW$23, DW_AT_name("PRU1_GPI_MODE") + 353 .dwattr $C$DW$23, DW_AT_TI_symbol_name("PRU1_GPI_MODE") + 354 .dwattr $C$DW$23, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x02) + 355 .dwattr $C$DW$23, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 356 .dwattr $C$DW$23, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 357 .dwattr $C$DW$23, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 358 .dwattr $C$DW$23, DW_AT_decl_line(0x58) + 359 .dwattr $C$DW$23, DW_AT_decl_column(0x0d) + 360$C$DW$24 .dwtag DW_TAG_member + 361 .dwattr $C$DW$24, DW_AT_type(*$C$DW$T$11) + 362 .dwattr $C$DW$24, DW_AT_name("PRU1_GPI_CLK_MODE") + 363 .dwattr $C$DW$24, DW_AT_TI_symbol_name("PRU1_GPI_CLK_MODE") + 364 .dwattr $C$DW$24, DW_AT_bit_offset(0x1d), DW_AT_bit_size(0x01) + 365 .dwattr $C$DW$24, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 366 .dwattr $C$DW$24, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 367 .dwattr $C$DW$24, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 368 .dwattr $C$DW$24, DW_AT_decl_line(0x59) + 369 .dwattr $C$DW$24, DW_AT_decl_column(0x0d) + 370$C$DW$25 .dwtag DW_TAG_member + 371 .dwattr $C$DW$25, DW_AT_type(*$C$DW$T$11) + 372 .dwattr $C$DW$25, DW_AT_name("PRU1_GPI_DIV0") + 373 .dwattr $C$DW$25, DW_AT_TI_symbol_name("PRU1_GPI_DIV0") + 374 .dwattr $C$DW$25, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x05) + 375 .dwattr $C$DW$25, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 376 .dwattr $C$DW$25, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 377 .dwattr $C$DW$25, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 378 .dwattr $C$DW$25, DW_AT_decl_line(0x5a) + 379 .dwattr $C$DW$25, DW_AT_decl_column(0x0d) + 380$C$DW$26 .dwtag DW_TAG_member + 381 .dwattr $C$DW$26, DW_AT_type(*$C$DW$T$11) + 382 .dwattr $C$DW$26, DW_AT_name("PRU1_GPI_DIV1") + 383 .dwattr $C$DW$26, DW_AT_TI_symbol_name("PRU1_GPI_DIV1") + 384 .dwattr $C$DW$26, DW_AT_bit_offset(0x13), DW_AT_bit_size(0x05) + 385 .dwattr $C$DW$26, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 8 + + 386 .dwattr $C$DW$26, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 387 .dwattr $C$DW$26, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 388 .dwattr $C$DW$26, DW_AT_decl_line(0x5b) + 389 .dwattr $C$DW$26, DW_AT_decl_column(0x0d) + 390$C$DW$27 .dwtag DW_TAG_member + 391 .dwattr $C$DW$27, DW_AT_type(*$C$DW$T$11) + 392 .dwattr $C$DW$27, DW_AT_name("PRU1_GPI_SB") + 393 .dwattr $C$DW$27, DW_AT_TI_symbol_name("PRU1_GPI_SB") + 394 .dwattr $C$DW$27, DW_AT_bit_offset(0x12), DW_AT_bit_size(0x01) + 395 .dwattr $C$DW$27, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 396 .dwattr $C$DW$27, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 397 .dwattr $C$DW$27, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 398 .dwattr $C$DW$27, DW_AT_decl_line(0x5c) + 399 .dwattr $C$DW$27, DW_AT_decl_column(0x0d) + 400$C$DW$28 .dwtag DW_TAG_member + 401 .dwattr $C$DW$28, DW_AT_type(*$C$DW$T$11) + 402 .dwattr $C$DW$28, DW_AT_name("PRU1_GPO_MODE") + 403 .dwattr $C$DW$28, DW_AT_TI_symbol_name("PRU1_GPO_MODE") + 404 .dwattr $C$DW$28, DW_AT_bit_offset(0x11), DW_AT_bit_size(0x01) + 405 .dwattr $C$DW$28, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 406 .dwattr $C$DW$28, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 407 .dwattr $C$DW$28, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 408 .dwattr $C$DW$28, DW_AT_decl_line(0x5d) + 409 .dwattr $C$DW$28, DW_AT_decl_column(0x0d) + 410$C$DW$29 .dwtag DW_TAG_member + 411 .dwattr $C$DW$29, DW_AT_type(*$C$DW$T$11) + 412 .dwattr $C$DW$29, DW_AT_name("PRU1_GPO_DIV0") + 413 .dwattr $C$DW$29, DW_AT_TI_symbol_name("PRU1_GPO_DIV0") + 414 .dwattr $C$DW$29, DW_AT_bit_offset(0x0c), DW_AT_bit_size(0x05) + 415 .dwattr $C$DW$29, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 416 .dwattr $C$DW$29, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 417 .dwattr $C$DW$29, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 418 .dwattr $C$DW$29, DW_AT_decl_line(0x5e) + 419 .dwattr $C$DW$29, DW_AT_decl_column(0x0d) + 420$C$DW$30 .dwtag DW_TAG_member + 421 .dwattr $C$DW$30, DW_AT_type(*$C$DW$T$11) + 422 .dwattr $C$DW$30, DW_AT_name("PRU1_GPO_DIV1") + 423 .dwattr $C$DW$30, DW_AT_TI_symbol_name("PRU1_GPO_DIV1") + 424 .dwattr $C$DW$30, DW_AT_bit_offset(0x07), DW_AT_bit_size(0x05) + 425 .dwattr $C$DW$30, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 426 .dwattr $C$DW$30, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 427 .dwattr $C$DW$30, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 428 .dwattr $C$DW$30, DW_AT_decl_line(0x5f) + 429 .dwattr $C$DW$30, DW_AT_decl_column(0x0d) + 430$C$DW$31 .dwtag DW_TAG_member + 431 .dwattr $C$DW$31, DW_AT_type(*$C$DW$T$11) + 432 .dwattr $C$DW$31, DW_AT_name("PRU1_GPO_SH_SEL") + 433 .dwattr $C$DW$31, DW_AT_TI_symbol_name("PRU1_GPO_SH_SEL") + 434 .dwattr $C$DW$31, DW_AT_bit_offset(0x06), DW_AT_bit_size(0x01) + 435 .dwattr $C$DW$31, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 436 .dwattr $C$DW$31, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 437 .dwattr $C$DW$31, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 438 .dwattr $C$DW$31, DW_AT_decl_line(0x60) + 439 .dwattr $C$DW$31, DW_AT_decl_column(0x0d) + 440$C$DW$32 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 9 + + 441 .dwattr $C$DW$32, DW_AT_type(*$C$DW$T$11) + 442 .dwattr $C$DW$32, DW_AT_name("rsvd26") + 443 .dwattr $C$DW$32, DW_AT_TI_symbol_name("rsvd26") + 444 .dwattr $C$DW$32, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x06) + 445 .dwattr $C$DW$32, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 446 .dwattr $C$DW$32, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 447 .dwattr $C$DW$32, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 448 .dwattr $C$DW$32, DW_AT_decl_line(0x61) + 449 .dwattr $C$DW$32, DW_AT_decl_column(0x0d) + 450 .dwendtag $C$DW$T$22 + 451 + 452 .dwattr $C$DW$T$22, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 453 .dwattr $C$DW$T$22, DW_AT_decl_line(0x57) + 454 .dwattr $C$DW$T$22, DW_AT_decl_column(0x13) + 455$C$DW$T$54 .dwtag DW_TAG_volatile_type + 456 .dwattr $C$DW$T$54, DW_AT_type(*$C$DW$T$22) + 457 + 458$C$DW$T$23 .dwtag DW_TAG_structure_type + 459 .dwattr $C$DW$T$23, DW_AT_byte_size(0x04) + 460$C$DW$33 .dwtag DW_TAG_member + 461 .dwattr $C$DW$33, DW_AT_type(*$C$DW$T$11) + 462 .dwattr $C$DW$33, DW_AT_name("PRU0_CLK_STOP_REQ") + 463 .dwattr $C$DW$33, DW_AT_TI_symbol_name("PRU0_CLK_STOP_REQ") + 464 .dwattr $C$DW$33, DW_AT_bit_offset(0x1f), DW_AT_bit_size(0x01) + 465 .dwattr $C$DW$33, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 466 .dwattr $C$DW$33, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 467 .dwattr $C$DW$33, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 468 .dwattr $C$DW$33, DW_AT_decl_line(0x6b) + 469 .dwattr $C$DW$33, DW_AT_decl_column(0x0d) + 470$C$DW$34 .dwtag DW_TAG_member + 471 .dwattr $C$DW$34, DW_AT_type(*$C$DW$T$11) + 472 .dwattr $C$DW$34, DW_AT_name("PRU0_CLK_STOP_ACK") + 473 .dwattr $C$DW$34, DW_AT_TI_symbol_name("PRU0_CLK_STOP_ACK") + 474 .dwattr $C$DW$34, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x01) + 475 .dwattr $C$DW$34, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 476 .dwattr $C$DW$34, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 477 .dwattr $C$DW$34, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 478 .dwattr $C$DW$34, DW_AT_decl_line(0x6c) + 479 .dwattr $C$DW$34, DW_AT_decl_column(0x0d) + 480$C$DW$35 .dwtag DW_TAG_member + 481 .dwattr $C$DW$35, DW_AT_type(*$C$DW$T$11) + 482 .dwattr $C$DW$35, DW_AT_name("PRU0_CLK_EN") + 483 .dwattr $C$DW$35, DW_AT_TI_symbol_name("PRU0_CLK_EN") + 484 .dwattr $C$DW$35, DW_AT_bit_offset(0x1d), DW_AT_bit_size(0x01) + 485 .dwattr $C$DW$35, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 486 .dwattr $C$DW$35, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 487 .dwattr $C$DW$35, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 488 .dwattr $C$DW$35, DW_AT_decl_line(0x6d) + 489 .dwattr $C$DW$35, DW_AT_decl_column(0x0d) + 490$C$DW$36 .dwtag DW_TAG_member + 491 .dwattr $C$DW$36, DW_AT_type(*$C$DW$T$11) + 492 .dwattr $C$DW$36, DW_AT_name("PRU1_CLK_STOP_REQ") + 493 .dwattr $C$DW$36, DW_AT_TI_symbol_name("PRU1_CLK_STOP_REQ") + 494 .dwattr $C$DW$36, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x01) + 495 .dwattr $C$DW$36, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 10 + + 496 .dwattr $C$DW$36, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 497 .dwattr $C$DW$36, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 498 .dwattr $C$DW$36, DW_AT_decl_line(0x6e) + 499 .dwattr $C$DW$36, DW_AT_decl_column(0x0d) + 500$C$DW$37 .dwtag DW_TAG_member + 501 .dwattr $C$DW$37, DW_AT_type(*$C$DW$T$11) + 502 .dwattr $C$DW$37, DW_AT_name("PRU1_CLK_STOP_ACK") + 503 .dwattr $C$DW$37, DW_AT_TI_symbol_name("PRU1_CLK_STOP_ACK") + 504 .dwattr $C$DW$37, DW_AT_bit_offset(0x1b), DW_AT_bit_size(0x01) + 505 .dwattr $C$DW$37, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 506 .dwattr $C$DW$37, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 507 .dwattr $C$DW$37, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 508 .dwattr $C$DW$37, DW_AT_decl_line(0x6f) + 509 .dwattr $C$DW$37, DW_AT_decl_column(0x0d) + 510$C$DW$38 .dwtag DW_TAG_member + 511 .dwattr $C$DW$38, DW_AT_type(*$C$DW$T$11) + 512 .dwattr $C$DW$38, DW_AT_name("PRU1_CLK_EN") + 513 .dwattr $C$DW$38, DW_AT_TI_symbol_name("PRU1_CLK_EN") + 514 .dwattr $C$DW$38, DW_AT_bit_offset(0x1a), DW_AT_bit_size(0x01) + 515 .dwattr $C$DW$38, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 516 .dwattr $C$DW$38, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 517 .dwattr $C$DW$38, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 518 .dwattr $C$DW$38, DW_AT_decl_line(0x70) + 519 .dwattr $C$DW$38, DW_AT_decl_column(0x0d) + 520$C$DW$39 .dwtag DW_TAG_member + 521 .dwattr $C$DW$39, DW_AT_type(*$C$DW$T$11) + 522 .dwattr $C$DW$39, DW_AT_name("INTC_CLK_STOP_REQ") + 523 .dwattr $C$DW$39, DW_AT_TI_symbol_name("INTC_CLK_STOP_REQ") + 524 .dwattr $C$DW$39, DW_AT_bit_offset(0x19), DW_AT_bit_size(0x01) + 525 .dwattr $C$DW$39, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 526 .dwattr $C$DW$39, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 527 .dwattr $C$DW$39, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 528 .dwattr $C$DW$39, DW_AT_decl_line(0x71) + 529 .dwattr $C$DW$39, DW_AT_decl_column(0x0d) + 530$C$DW$40 .dwtag DW_TAG_member + 531 .dwattr $C$DW$40, DW_AT_type(*$C$DW$T$11) + 532 .dwattr $C$DW$40, DW_AT_name("INTC_CLK_STOP_ACK") + 533 .dwattr $C$DW$40, DW_AT_TI_symbol_name("INTC_CLK_STOP_ACK") + 534 .dwattr $C$DW$40, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x01) + 535 .dwattr $C$DW$40, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 536 .dwattr $C$DW$40, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 537 .dwattr $C$DW$40, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 538 .dwattr $C$DW$40, DW_AT_decl_line(0x72) + 539 .dwattr $C$DW$40, DW_AT_decl_column(0x0d) + 540$C$DW$41 .dwtag DW_TAG_member + 541 .dwattr $C$DW$41, DW_AT_type(*$C$DW$T$11) + 542 .dwattr $C$DW$41, DW_AT_name("INTC_CLK_EN") + 543 .dwattr $C$DW$41, DW_AT_TI_symbol_name("INTC_CLK_EN") + 544 .dwattr $C$DW$41, DW_AT_bit_offset(0x17), DW_AT_bit_size(0x01) + 545 .dwattr $C$DW$41, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 546 .dwattr $C$DW$41, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 547 .dwattr $C$DW$41, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 548 .dwattr $C$DW$41, DW_AT_decl_line(0x73) + 549 .dwattr $C$DW$41, DW_AT_decl_column(0x0d) + 550$C$DW$42 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 11 + + 551 .dwattr $C$DW$42, DW_AT_type(*$C$DW$T$11) + 552 .dwattr $C$DW$42, DW_AT_name("UART_CLK_STOP_REQ") + 553 .dwattr $C$DW$42, DW_AT_TI_symbol_name("UART_CLK_STOP_REQ") + 554 .dwattr $C$DW$42, DW_AT_bit_offset(0x16), DW_AT_bit_size(0x01) + 555 .dwattr $C$DW$42, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 556 .dwattr $C$DW$42, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 557 .dwattr $C$DW$42, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 558 .dwattr $C$DW$42, DW_AT_decl_line(0x74) + 559 .dwattr $C$DW$42, DW_AT_decl_column(0x0d) + 560$C$DW$43 .dwtag DW_TAG_member + 561 .dwattr $C$DW$43, DW_AT_type(*$C$DW$T$11) + 562 .dwattr $C$DW$43, DW_AT_name("UART_CLK_STOP_ACK") + 563 .dwattr $C$DW$43, DW_AT_TI_symbol_name("UART_CLK_STOP_ACK") + 564 .dwattr $C$DW$43, DW_AT_bit_offset(0x15), DW_AT_bit_size(0x01) + 565 .dwattr $C$DW$43, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 566 .dwattr $C$DW$43, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 567 .dwattr $C$DW$43, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 568 .dwattr $C$DW$43, DW_AT_decl_line(0x75) + 569 .dwattr $C$DW$43, DW_AT_decl_column(0x0d) + 570$C$DW$44 .dwtag DW_TAG_member + 571 .dwattr $C$DW$44, DW_AT_type(*$C$DW$T$11) + 572 .dwattr $C$DW$44, DW_AT_name("UART_CLK_EN") + 573 .dwattr $C$DW$44, DW_AT_TI_symbol_name("UART_CLK_EN") + 574 .dwattr $C$DW$44, DW_AT_bit_offset(0x14), DW_AT_bit_size(0x01) + 575 .dwattr $C$DW$44, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 576 .dwattr $C$DW$44, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 577 .dwattr $C$DW$44, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 578 .dwattr $C$DW$44, DW_AT_decl_line(0x76) + 579 .dwattr $C$DW$44, DW_AT_decl_column(0x0d) + 580$C$DW$45 .dwtag DW_TAG_member + 581 .dwattr $C$DW$45, DW_AT_type(*$C$DW$T$11) + 582 .dwattr $C$DW$45, DW_AT_name("ECAP_CLK_STOP_REQ") + 583 .dwattr $C$DW$45, DW_AT_TI_symbol_name("ECAP_CLK_STOP_REQ") + 584 .dwattr $C$DW$45, DW_AT_bit_offset(0x13), DW_AT_bit_size(0x01) + 585 .dwattr $C$DW$45, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 586 .dwattr $C$DW$45, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 587 .dwattr $C$DW$45, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 588 .dwattr $C$DW$45, DW_AT_decl_line(0x77) + 589 .dwattr $C$DW$45, DW_AT_decl_column(0x0d) + 590$C$DW$46 .dwtag DW_TAG_member + 591 .dwattr $C$DW$46, DW_AT_type(*$C$DW$T$11) + 592 .dwattr $C$DW$46, DW_AT_name("ECAP_CLK_STOP_ACK") + 593 .dwattr $C$DW$46, DW_AT_TI_symbol_name("ECAP_CLK_STOP_ACK") + 594 .dwattr $C$DW$46, DW_AT_bit_offset(0x12), DW_AT_bit_size(0x01) + 595 .dwattr $C$DW$46, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 596 .dwattr $C$DW$46, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 597 .dwattr $C$DW$46, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 598 .dwattr $C$DW$46, DW_AT_decl_line(0x78) + 599 .dwattr $C$DW$46, DW_AT_decl_column(0x0d) + 600$C$DW$47 .dwtag DW_TAG_member + 601 .dwattr $C$DW$47, DW_AT_type(*$C$DW$T$11) + 602 .dwattr $C$DW$47, DW_AT_name("ECAP_CLK_EN") + 603 .dwattr $C$DW$47, DW_AT_TI_symbol_name("ECAP_CLK_EN") + 604 .dwattr $C$DW$47, DW_AT_bit_offset(0x11), DW_AT_bit_size(0x01) + 605 .dwattr $C$DW$47, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 12 + + 606 .dwattr $C$DW$47, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 607 .dwattr $C$DW$47, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 608 .dwattr $C$DW$47, DW_AT_decl_line(0x79) + 609 .dwattr $C$DW$47, DW_AT_decl_column(0x0d) + 610$C$DW$48 .dwtag DW_TAG_member + 611 .dwattr $C$DW$48, DW_AT_type(*$C$DW$T$11) + 612 .dwattr $C$DW$48, DW_AT_name("IEP_CLK_STOP_REQ") + 613 .dwattr $C$DW$48, DW_AT_TI_symbol_name("IEP_CLK_STOP_REQ") + 614 .dwattr $C$DW$48, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x01) + 615 .dwattr $C$DW$48, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 616 .dwattr $C$DW$48, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 617 .dwattr $C$DW$48, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 618 .dwattr $C$DW$48, DW_AT_decl_line(0x7a) + 619 .dwattr $C$DW$48, DW_AT_decl_column(0x0d) + 620$C$DW$49 .dwtag DW_TAG_member + 621 .dwattr $C$DW$49, DW_AT_type(*$C$DW$T$11) + 622 .dwattr $C$DW$49, DW_AT_name("IEP_CLK_STOP_ACK") + 623 .dwattr $C$DW$49, DW_AT_TI_symbol_name("IEP_CLK_STOP_ACK") + 624 .dwattr $C$DW$49, DW_AT_bit_offset(0x0f), DW_AT_bit_size(0x01) + 625 .dwattr $C$DW$49, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 626 .dwattr $C$DW$49, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 627 .dwattr $C$DW$49, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 628 .dwattr $C$DW$49, DW_AT_decl_line(0x7b) + 629 .dwattr $C$DW$49, DW_AT_decl_column(0x0d) + 630$C$DW$50 .dwtag DW_TAG_member + 631 .dwattr $C$DW$50, DW_AT_type(*$C$DW$T$11) + 632 .dwattr $C$DW$50, DW_AT_name("IEP_CLK_EN") + 633 .dwattr $C$DW$50, DW_AT_TI_symbol_name("IEP_CLK_EN") + 634 .dwattr $C$DW$50, DW_AT_bit_offset(0x0e), DW_AT_bit_size(0x01) + 635 .dwattr $C$DW$50, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 636 .dwattr $C$DW$50, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 637 .dwattr $C$DW$50, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 638 .dwattr $C$DW$50, DW_AT_decl_line(0x7c) + 639 .dwattr $C$DW$50, DW_AT_decl_column(0x0d) + 640$C$DW$51 .dwtag DW_TAG_member + 641 .dwattr $C$DW$51, DW_AT_type(*$C$DW$T$11) + 642 .dwattr $C$DW$51, DW_AT_name("rsvd18") + 643 .dwattr $C$DW$51, DW_AT_TI_symbol_name("rsvd18") + 644 .dwattr $C$DW$51, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x0e) + 645 .dwattr $C$DW$51, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 646 .dwattr $C$DW$51, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 647 .dwattr $C$DW$51, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 648 .dwattr $C$DW$51, DW_AT_decl_line(0x7d) + 649 .dwattr $C$DW$51, DW_AT_decl_column(0x0d) + 650 .dwendtag $C$DW$T$23 + 651 + 652 .dwattr $C$DW$T$23, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 653 .dwattr $C$DW$T$23, DW_AT_decl_line(0x6a) + 654 .dwattr $C$DW$T$23, DW_AT_decl_column(0x13) + 655$C$DW$T$56 .dwtag DW_TAG_volatile_type + 656 .dwattr $C$DW$T$56, DW_AT_type(*$C$DW$T$23) + 657 + 658$C$DW$T$24 .dwtag DW_TAG_structure_type + 659 .dwattr $C$DW$T$24, DW_AT_byte_size(0x04) + 660$C$DW$52 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 13 + + 661 .dwattr $C$DW$52, DW_AT_type(*$C$DW$T$11) + 662 .dwattr $C$DW$52, DW_AT_name("PRU0_IMEM_PE_RAW") + 663 .dwattr $C$DW$52, DW_AT_TI_symbol_name("PRU0_IMEM_PE_RAW") + 664 .dwattr $C$DW$52, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x04) + 665 .dwattr $C$DW$52, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 666 .dwattr $C$DW$52, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 667 .dwattr $C$DW$52, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 668 .dwattr $C$DW$52, DW_AT_decl_line(0x87) + 669 .dwattr $C$DW$52, DW_AT_decl_column(0x0d) + 670$C$DW$53 .dwtag DW_TAG_member + 671 .dwattr $C$DW$53, DW_AT_type(*$C$DW$T$11) + 672 .dwattr $C$DW$53, DW_AT_name("PRU0_DMEM_PE_RAW") + 673 .dwattr $C$DW$53, DW_AT_TI_symbol_name("PRU0_DMEM_PE_RAW") + 674 .dwattr $C$DW$53, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x04) + 675 .dwattr $C$DW$53, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 676 .dwattr $C$DW$53, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 677 .dwattr $C$DW$53, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 678 .dwattr $C$DW$53, DW_AT_decl_line(0x88) + 679 .dwattr $C$DW$53, DW_AT_decl_column(0x0d) + 680$C$DW$54 .dwtag DW_TAG_member + 681 .dwattr $C$DW$54, DW_AT_type(*$C$DW$T$11) + 682 .dwattr $C$DW$54, DW_AT_name("PRU1_IMEM_PE_RAW") + 683 .dwattr $C$DW$54, DW_AT_TI_symbol_name("PRU1_IMEM_PE_RAW") + 684 .dwattr $C$DW$54, DW_AT_bit_offset(0x14), DW_AT_bit_size(0x04) + 685 .dwattr $C$DW$54, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 686 .dwattr $C$DW$54, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 687 .dwattr $C$DW$54, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 688 .dwattr $C$DW$54, DW_AT_decl_line(0x89) + 689 .dwattr $C$DW$54, DW_AT_decl_column(0x0d) + 690$C$DW$55 .dwtag DW_TAG_member + 691 .dwattr $C$DW$55, DW_AT_type(*$C$DW$T$11) + 692 .dwattr $C$DW$55, DW_AT_name("PRU1_DMEM_PE_RAW") + 693 .dwattr $C$DW$55, DW_AT_TI_symbol_name("PRU1_DMEM_PE_RAW") + 694 .dwattr $C$DW$55, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x04) + 695 .dwattr $C$DW$55, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 696 .dwattr $C$DW$55, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 697 .dwattr $C$DW$55, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 698 .dwattr $C$DW$55, DW_AT_decl_line(0x8a) + 699 .dwattr $C$DW$55, DW_AT_decl_column(0x0d) + 700$C$DW$56 .dwtag DW_TAG_member + 701 .dwattr $C$DW$56, DW_AT_type(*$C$DW$T$11) + 702 .dwattr $C$DW$56, DW_AT_name("RAM_PE_RAW") + 703 .dwattr $C$DW$56, DW_AT_TI_symbol_name("RAM_PE_RAW") + 704 .dwattr $C$DW$56, DW_AT_bit_offset(0x0c), DW_AT_bit_size(0x04) + 705 .dwattr $C$DW$56, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 706 .dwattr $C$DW$56, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 707 .dwattr $C$DW$56, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 708 .dwattr $C$DW$56, DW_AT_decl_line(0x8b) + 709 .dwattr $C$DW$56, DW_AT_decl_column(0x0d) + 710$C$DW$57 .dwtag DW_TAG_member + 711 .dwattr $C$DW$57, DW_AT_type(*$C$DW$T$11) + 712 .dwattr $C$DW$57, DW_AT_name("rsvd20") + 713 .dwattr $C$DW$57, DW_AT_TI_symbol_name("rsvd20") + 714 .dwattr $C$DW$57, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x0c) + 715 .dwattr $C$DW$57, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 14 + + 716 .dwattr $C$DW$57, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 717 .dwattr $C$DW$57, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 718 .dwattr $C$DW$57, DW_AT_decl_line(0x8c) + 719 .dwattr $C$DW$57, DW_AT_decl_column(0x0d) + 720 .dwendtag $C$DW$T$24 + 721 + 722 .dwattr $C$DW$T$24, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 723 .dwattr $C$DW$T$24, DW_AT_decl_line(0x86) + 724 .dwattr $C$DW$T$24, DW_AT_decl_column(0x14) + 725$C$DW$T$58 .dwtag DW_TAG_volatile_type + 726 .dwattr $C$DW$T$58, DW_AT_type(*$C$DW$T$24) + 727 + 728$C$DW$T$25 .dwtag DW_TAG_structure_type + 729 .dwattr $C$DW$T$25, DW_AT_byte_size(0x04) + 730$C$DW$58 .dwtag DW_TAG_member + 731 .dwattr $C$DW$58, DW_AT_type(*$C$DW$T$11) + 732 .dwattr $C$DW$58, DW_AT_name("PRU0_IMEM_PE") + 733 .dwattr $C$DW$58, DW_AT_TI_symbol_name("PRU0_IMEM_PE") + 734 .dwattr $C$DW$58, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x04) + 735 .dwattr $C$DW$58, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 736 .dwattr $C$DW$58, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 737 .dwattr $C$DW$58, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 738 .dwattr $C$DW$58, DW_AT_decl_line(0x96) + 739 .dwattr $C$DW$58, DW_AT_decl_column(0x0d) + 740$C$DW$59 .dwtag DW_TAG_member + 741 .dwattr $C$DW$59, DW_AT_type(*$C$DW$T$11) + 742 .dwattr $C$DW$59, DW_AT_name("PRU0_DMEM_PE") + 743 .dwattr $C$DW$59, DW_AT_TI_symbol_name("PRU0_DMEM_PE") + 744 .dwattr $C$DW$59, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x04) + 745 .dwattr $C$DW$59, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 746 .dwattr $C$DW$59, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 747 .dwattr $C$DW$59, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 748 .dwattr $C$DW$59, DW_AT_decl_line(0x97) + 749 .dwattr $C$DW$59, DW_AT_decl_column(0x0d) + 750$C$DW$60 .dwtag DW_TAG_member + 751 .dwattr $C$DW$60, DW_AT_type(*$C$DW$T$11) + 752 .dwattr $C$DW$60, DW_AT_name("PRU1_IMEM_PE") + 753 .dwattr $C$DW$60, DW_AT_TI_symbol_name("PRU1_IMEM_PE") + 754 .dwattr $C$DW$60, DW_AT_bit_offset(0x14), DW_AT_bit_size(0x04) + 755 .dwattr $C$DW$60, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 756 .dwattr $C$DW$60, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 757 .dwattr $C$DW$60, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 758 .dwattr $C$DW$60, DW_AT_decl_line(0x98) + 759 .dwattr $C$DW$60, DW_AT_decl_column(0x0d) + 760$C$DW$61 .dwtag DW_TAG_member + 761 .dwattr $C$DW$61, DW_AT_type(*$C$DW$T$11) + 762 .dwattr $C$DW$61, DW_AT_name("PRU1_DMEM_PE") + 763 .dwattr $C$DW$61, DW_AT_TI_symbol_name("PRU1_DMEM_PE") + 764 .dwattr $C$DW$61, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x04) + 765 .dwattr $C$DW$61, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 766 .dwattr $C$DW$61, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 767 .dwattr $C$DW$61, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 768 .dwattr $C$DW$61, DW_AT_decl_line(0x99) + 769 .dwattr $C$DW$61, DW_AT_decl_column(0x0d) + 770$C$DW$62 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 15 + + 771 .dwattr $C$DW$62, DW_AT_type(*$C$DW$T$11) + 772 .dwattr $C$DW$62, DW_AT_name("RAM_PE") + 773 .dwattr $C$DW$62, DW_AT_TI_symbol_name("RAM_PE") + 774 .dwattr $C$DW$62, DW_AT_bit_offset(0x0c), DW_AT_bit_size(0x04) + 775 .dwattr $C$DW$62, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 776 .dwattr $C$DW$62, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 777 .dwattr $C$DW$62, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 778 .dwattr $C$DW$62, DW_AT_decl_line(0x9a) + 779 .dwattr $C$DW$62, DW_AT_decl_column(0x0d) + 780$C$DW$63 .dwtag DW_TAG_member + 781 .dwattr $C$DW$63, DW_AT_type(*$C$DW$T$11) + 782 .dwattr $C$DW$63, DW_AT_name("rsvd20") + 783 .dwattr $C$DW$63, DW_AT_TI_symbol_name("rsvd20") + 784 .dwattr $C$DW$63, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x0c) + 785 .dwattr $C$DW$63, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 786 .dwattr $C$DW$63, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 787 .dwattr $C$DW$63, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 788 .dwattr $C$DW$63, DW_AT_decl_line(0x9b) + 789 .dwattr $C$DW$63, DW_AT_decl_column(0x0d) + 790 .dwendtag $C$DW$T$25 + 791 + 792 .dwattr $C$DW$T$25, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 793 .dwattr $C$DW$T$25, DW_AT_decl_line(0x95) + 794 .dwattr $C$DW$T$25, DW_AT_decl_column(0x14) + 795$C$DW$T$60 .dwtag DW_TAG_volatile_type + 796 .dwattr $C$DW$T$60, DW_AT_type(*$C$DW$T$25) + 797 + 798$C$DW$T$26 .dwtag DW_TAG_structure_type + 799 .dwattr $C$DW$T$26, DW_AT_byte_size(0x04) + 800$C$DW$64 .dwtag DW_TAG_member + 801 .dwattr $C$DW$64, DW_AT_type(*$C$DW$T$11) + 802 .dwattr $C$DW$64, DW_AT_name("PRU0_IMEM_PE_SET") + 803 .dwattr $C$DW$64, DW_AT_TI_symbol_name("PRU0_IMEM_PE_SET") + 804 .dwattr $C$DW$64, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x04) + 805 .dwattr $C$DW$64, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 806 .dwattr $C$DW$64, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 807 .dwattr $C$DW$64, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 808 .dwattr $C$DW$64, DW_AT_decl_line(0xa4) + 809 .dwattr $C$DW$64, DW_AT_decl_column(0x0d) + 810$C$DW$65 .dwtag DW_TAG_member + 811 .dwattr $C$DW$65, DW_AT_type(*$C$DW$T$11) + 812 .dwattr $C$DW$65, DW_AT_name("PRU0_DMEM_PE_SET") + 813 .dwattr $C$DW$65, DW_AT_TI_symbol_name("PRU0_DMEM_PE_SET") + 814 .dwattr $C$DW$65, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x04) + 815 .dwattr $C$DW$65, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 816 .dwattr $C$DW$65, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 817 .dwattr $C$DW$65, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 818 .dwattr $C$DW$65, DW_AT_decl_line(0xa5) + 819 .dwattr $C$DW$65, DW_AT_decl_column(0x0d) + 820$C$DW$66 .dwtag DW_TAG_member + 821 .dwattr $C$DW$66, DW_AT_type(*$C$DW$T$11) + 822 .dwattr $C$DW$66, DW_AT_name("PRU1_IMEM_PE_SET") + 823 .dwattr $C$DW$66, DW_AT_TI_symbol_name("PRU1_IMEM_PE_SET") + 824 .dwattr $C$DW$66, DW_AT_bit_offset(0x14), DW_AT_bit_size(0x04) + 825 .dwattr $C$DW$66, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 16 + + 826 .dwattr $C$DW$66, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 827 .dwattr $C$DW$66, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 828 .dwattr $C$DW$66, DW_AT_decl_line(0xa6) + 829 .dwattr $C$DW$66, DW_AT_decl_column(0x0d) + 830$C$DW$67 .dwtag DW_TAG_member + 831 .dwattr $C$DW$67, DW_AT_type(*$C$DW$T$11) + 832 .dwattr $C$DW$67, DW_AT_name("PRU1_DMEM_PE_SET") + 833 .dwattr $C$DW$67, DW_AT_TI_symbol_name("PRU1_DMEM_PE_SET") + 834 .dwattr $C$DW$67, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x04) + 835 .dwattr $C$DW$67, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 836 .dwattr $C$DW$67, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 837 .dwattr $C$DW$67, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 838 .dwattr $C$DW$67, DW_AT_decl_line(0xa7) + 839 .dwattr $C$DW$67, DW_AT_decl_column(0x0d) + 840$C$DW$68 .dwtag DW_TAG_member + 841 .dwattr $C$DW$68, DW_AT_type(*$C$DW$T$11) + 842 .dwattr $C$DW$68, DW_AT_name("RAM_PE_SET") + 843 .dwattr $C$DW$68, DW_AT_TI_symbol_name("RAM_PE_SET") + 844 .dwattr $C$DW$68, DW_AT_bit_offset(0x0c), DW_AT_bit_size(0x04) + 845 .dwattr $C$DW$68, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 846 .dwattr $C$DW$68, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 847 .dwattr $C$DW$68, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 848 .dwattr $C$DW$68, DW_AT_decl_line(0xa8) + 849 .dwattr $C$DW$68, DW_AT_decl_column(0x0d) + 850$C$DW$69 .dwtag DW_TAG_member + 851 .dwattr $C$DW$69, DW_AT_type(*$C$DW$T$11) + 852 .dwattr $C$DW$69, DW_AT_name("rsvd20") + 853 .dwattr $C$DW$69, DW_AT_TI_symbol_name("rsvd20") + 854 .dwattr $C$DW$69, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x0c) + 855 .dwattr $C$DW$69, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 856 .dwattr $C$DW$69, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 857 .dwattr $C$DW$69, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 858 .dwattr $C$DW$69, DW_AT_decl_line(0xa9) + 859 .dwattr $C$DW$69, DW_AT_decl_column(0x0d) + 860 .dwendtag $C$DW$T$26 + 861 + 862 .dwattr $C$DW$T$26, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 863 .dwattr $C$DW$T$26, DW_AT_decl_line(0xa3) + 864 .dwattr $C$DW$T$26, DW_AT_decl_column(0x13) + 865$C$DW$T$62 .dwtag DW_TAG_volatile_type + 866 .dwattr $C$DW$T$62, DW_AT_type(*$C$DW$T$26) + 867 + 868$C$DW$T$27 .dwtag DW_TAG_structure_type + 869 .dwattr $C$DW$T$27, DW_AT_byte_size(0x04) + 870$C$DW$70 .dwtag DW_TAG_member + 871 .dwattr $C$DW$70, DW_AT_type(*$C$DW$T$11) + 872 .dwattr $C$DW$70, DW_AT_name("PRU0_IMEM_PE_CLR") + 873 .dwattr $C$DW$70, DW_AT_TI_symbol_name("PRU0_IMEM_PE_CLR") + 874 .dwattr $C$DW$70, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x04) + 875 .dwattr $C$DW$70, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 876 .dwattr $C$DW$70, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 877 .dwattr $C$DW$70, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 878 .dwattr $C$DW$70, DW_AT_decl_line(0xb3) + 879 .dwattr $C$DW$70, DW_AT_decl_column(0x0d) + 880$C$DW$71 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 17 + + 881 .dwattr $C$DW$71, DW_AT_type(*$C$DW$T$11) + 882 .dwattr $C$DW$71, DW_AT_name("PRU0_DMEM_PE_CLR") + 883 .dwattr $C$DW$71, DW_AT_TI_symbol_name("PRU0_DMEM_PE_CLR") + 884 .dwattr $C$DW$71, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x04) + 885 .dwattr $C$DW$71, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 886 .dwattr $C$DW$71, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 887 .dwattr $C$DW$71, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 888 .dwattr $C$DW$71, DW_AT_decl_line(0xb4) + 889 .dwattr $C$DW$71, DW_AT_decl_column(0x0d) + 890$C$DW$72 .dwtag DW_TAG_member + 891 .dwattr $C$DW$72, DW_AT_type(*$C$DW$T$11) + 892 .dwattr $C$DW$72, DW_AT_name("PRU1_IMEM_PE_CLR") + 893 .dwattr $C$DW$72, DW_AT_TI_symbol_name("PRU1_IMEM_PE_CLR") + 894 .dwattr $C$DW$72, DW_AT_bit_offset(0x14), DW_AT_bit_size(0x04) + 895 .dwattr $C$DW$72, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 896 .dwattr $C$DW$72, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 897 .dwattr $C$DW$72, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 898 .dwattr $C$DW$72, DW_AT_decl_line(0xb5) + 899 .dwattr $C$DW$72, DW_AT_decl_column(0x0d) + 900$C$DW$73 .dwtag DW_TAG_member + 901 .dwattr $C$DW$73, DW_AT_type(*$C$DW$T$11) + 902 .dwattr $C$DW$73, DW_AT_name("PRU1_DMEM_PE_CLR") + 903 .dwattr $C$DW$73, DW_AT_TI_symbol_name("PRU1_DMEM_PE_CLR") + 904 .dwattr $C$DW$73, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x04) + 905 .dwattr $C$DW$73, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 906 .dwattr $C$DW$73, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 907 .dwattr $C$DW$73, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 908 .dwattr $C$DW$73, DW_AT_decl_line(0xb6) + 909 .dwattr $C$DW$73, DW_AT_decl_column(0x0d) + 910$C$DW$74 .dwtag DW_TAG_member + 911 .dwattr $C$DW$74, DW_AT_type(*$C$DW$T$11) + 912 .dwattr $C$DW$74, DW_AT_name("rsvd16") + 913 .dwattr $C$DW$74, DW_AT_TI_symbol_name("rsvd16") + 914 .dwattr $C$DW$74, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x10) + 915 .dwattr $C$DW$74, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 916 .dwattr $C$DW$74, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 917 .dwattr $C$DW$74, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 918 .dwattr $C$DW$74, DW_AT_decl_line(0xb7) + 919 .dwattr $C$DW$74, DW_AT_decl_column(0x0d) + 920 .dwendtag $C$DW$T$27 + 921 + 922 .dwattr $C$DW$T$27, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 923 .dwattr $C$DW$T$27, DW_AT_decl_line(0xb2) + 924 .dwattr $C$DW$T$27, DW_AT_decl_column(0x13) + 925$C$DW$T$64 .dwtag DW_TAG_volatile_type + 926 .dwattr $C$DW$T$64, DW_AT_type(*$C$DW$T$27) + 927 + 928$C$DW$T$28 .dwtag DW_TAG_structure_type + 929 .dwattr $C$DW$T$28, DW_AT_byte_size(0x04) + 930$C$DW$75 .dwtag DW_TAG_member + 931 .dwattr $C$DW$75, DW_AT_type(*$C$DW$T$11) + 932 .dwattr $C$DW$75, DW_AT_name("PMAO_PRU0") + 933 .dwattr $C$DW$75, DW_AT_TI_symbol_name("PMAO_PRU0") + 934 .dwattr $C$DW$75, DW_AT_bit_offset(0x1f), DW_AT_bit_size(0x01) + 935 .dwattr $C$DW$75, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 18 + + 936 .dwattr $C$DW$75, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 937 .dwattr $C$DW$75, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 938 .dwattr $C$DW$75, DW_AT_decl_line(0xc4) + 939 .dwattr $C$DW$75, DW_AT_decl_column(0x0d) + 940$C$DW$76 .dwtag DW_TAG_member + 941 .dwattr $C$DW$76, DW_AT_type(*$C$DW$T$11) + 942 .dwattr $C$DW$76, DW_AT_name("PMAO_PRU1") + 943 .dwattr $C$DW$76, DW_AT_TI_symbol_name("PMAO_PRU1") + 944 .dwattr $C$DW$76, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x01) + 945 .dwattr $C$DW$76, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 946 .dwattr $C$DW$76, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 947 .dwattr $C$DW$76, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 948 .dwattr $C$DW$76, DW_AT_decl_line(0xc5) + 949 .dwattr $C$DW$76, DW_AT_decl_column(0x0d) + 950$C$DW$77 .dwtag DW_TAG_member + 951 .dwattr $C$DW$77, DW_AT_type(*$C$DW$T$11) + 952 .dwattr $C$DW$77, DW_AT_name("rsvd2") + 953 .dwattr $C$DW$77, DW_AT_TI_symbol_name("rsvd2") + 954 .dwattr $C$DW$77, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x1e) + 955 .dwattr $C$DW$77, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 956 .dwattr $C$DW$77, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 957 .dwattr $C$DW$77, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 958 .dwattr $C$DW$77, DW_AT_decl_line(0xc6) + 959 .dwattr $C$DW$77, DW_AT_decl_column(0x0d) + 960 .dwendtag $C$DW$T$28 + 961 + 962 .dwattr $C$DW$T$28, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 963 .dwattr $C$DW$T$28, DW_AT_decl_line(0xc3) + 964 .dwattr $C$DW$T$28, DW_AT_decl_column(0x13) + 965$C$DW$T$66 .dwtag DW_TAG_volatile_type + 966 .dwattr $C$DW$T$66, DW_AT_type(*$C$DW$T$28) + 967 + 968$C$DW$T$29 .dwtag DW_TAG_structure_type + 969 .dwattr $C$DW$T$29, DW_AT_byte_size(0x04) + 970$C$DW$78 .dwtag DW_TAG_member + 971 .dwattr $C$DW$78, DW_AT_type(*$C$DW$T$11) + 972 .dwattr $C$DW$78, DW_AT_name("OCP_EN") + 973 .dwattr $C$DW$78, DW_AT_TI_symbol_name("OCP_EN") + 974 .dwattr $C$DW$78, DW_AT_bit_offset(0x1f), DW_AT_bit_size(0x01) + 975 .dwattr $C$DW$78, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 976 .dwattr $C$DW$78, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 977 .dwattr $C$DW$78, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 978 .dwattr $C$DW$78, DW_AT_decl_line(0xd3) + 979 .dwattr $C$DW$78, DW_AT_decl_column(0x0d) + 980$C$DW$79 .dwtag DW_TAG_member + 981 .dwattr $C$DW$79, DW_AT_type(*$C$DW$T$11) + 982 .dwattr $C$DW$79, DW_AT_name("rsvd1") + 983 .dwattr $C$DW$79, DW_AT_TI_symbol_name("rsvd1") + 984 .dwattr $C$DW$79, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x1f) + 985 .dwattr $C$DW$79, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 986 .dwattr $C$DW$79, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 987 .dwattr $C$DW$79, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 988 .dwattr $C$DW$79, DW_AT_decl_line(0xd4) + 989 .dwattr $C$DW$79, DW_AT_decl_column(0x0d) + 990 .dwendtag $C$DW$T$29 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 19 + + 991 + 992 .dwattr $C$DW$T$29, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 993 .dwattr $C$DW$T$29, DW_AT_decl_line(0xd2) + 994 .dwattr $C$DW$T$29, DW_AT_decl_column(0x13) + 995$C$DW$T$68 .dwtag DW_TAG_volatile_type + 996 .dwattr $C$DW$T$68, DW_AT_type(*$C$DW$T$29) + 997 + 998$C$DW$T$30 .dwtag DW_TAG_structure_type + 999 .dwattr $C$DW$T$30, DW_AT_byte_size(0x04) + 1000$C$DW$80 .dwtag DW_TAG_member + 1001 .dwattr $C$DW$80, DW_AT_type(*$C$DW$T$11) + 1002 .dwattr $C$DW$80, DW_AT_name("PRU1_PAD_HP_EN") + 1003 .dwattr $C$DW$80, DW_AT_TI_symbol_name("PRU1_PAD_HP_EN") + 1004 .dwattr $C$DW$80, DW_AT_bit_offset(0x1f), DW_AT_bit_size(0x01) + 1005 .dwattr $C$DW$80, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1006 .dwattr $C$DW$80, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1007 .dwattr $C$DW$80, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1008 .dwattr $C$DW$80, DW_AT_decl_line(0xde) + 1009 .dwattr $C$DW$80, DW_AT_decl_column(0x0d) + 1010$C$DW$81 .dwtag DW_TAG_member + 1011 .dwattr $C$DW$81, DW_AT_type(*$C$DW$T$11) + 1012 .dwattr $C$DW$81, DW_AT_name("XFR_SHIFT_EN") + 1013 .dwattr $C$DW$81, DW_AT_TI_symbol_name("XFR_SHIFT_EN") + 1014 .dwattr $C$DW$81, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x01) + 1015 .dwattr $C$DW$81, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1016 .dwattr $C$DW$81, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1017 .dwattr $C$DW$81, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1018 .dwattr $C$DW$81, DW_AT_decl_line(0xdf) + 1019 .dwattr $C$DW$81, DW_AT_decl_column(0x0d) + 1020$C$DW$82 .dwtag DW_TAG_member + 1021 .dwattr $C$DW$82, DW_AT_type(*$C$DW$T$11) + 1022 .dwattr $C$DW$82, DW_AT_name("rsvd2") + 1023 .dwattr $C$DW$82, DW_AT_TI_symbol_name("rsvd2") + 1024 .dwattr $C$DW$82, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x1e) + 1025 .dwattr $C$DW$82, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1026 .dwattr $C$DW$82, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1027 .dwattr $C$DW$82, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1028 .dwattr $C$DW$82, DW_AT_decl_line(0xe0) + 1029 .dwattr $C$DW$82, DW_AT_decl_column(0x0d) + 1030 .dwendtag $C$DW$T$30 + 1031 + 1032 .dwattr $C$DW$T$30, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1033 .dwattr $C$DW$T$30, DW_AT_decl_line(0xdd) + 1034 .dwattr $C$DW$T$30, DW_AT_decl_column(0x13) + 1035$C$DW$T$70 .dwtag DW_TAG_volatile_type + 1036 .dwattr $C$DW$T$70, DW_AT_type(*$C$DW$T$30) + 1037 + 1038$C$DW$T$31 .dwtag DW_TAG_structure_type + 1039 .dwattr $C$DW$T$31, DW_AT_byte_size(0x04) + 1040$C$DW$83 .dwtag DW_TAG_member + 1041 .dwattr $C$DW$83, DW_AT_type(*$C$DW$T$11) + 1042 .dwattr $C$DW$83, DW_AT_name("PIN_MUX_SEL") + 1043 .dwattr $C$DW$83, DW_AT_TI_symbol_name("PIN_MUX_SEL") + 1044 .dwattr $C$DW$83, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x08) + 1045 .dwattr $C$DW$83, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 20 + + 1046 .dwattr $C$DW$83, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1047 .dwattr $C$DW$83, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1048 .dwattr $C$DW$83, DW_AT_decl_line(0xec) + 1049 .dwattr $C$DW$83, DW_AT_decl_column(0x0d) + 1050$C$DW$84 .dwtag DW_TAG_member + 1051 .dwattr $C$DW$84, DW_AT_type(*$C$DW$T$11) + 1052 .dwattr $C$DW$84, DW_AT_name("rsvd2") + 1053 .dwattr $C$DW$84, DW_AT_TI_symbol_name("rsvd2") + 1054 .dwattr $C$DW$84, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x18) + 1055 .dwattr $C$DW$84, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1056 .dwattr $C$DW$84, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1057 .dwattr $C$DW$84, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1058 .dwattr $C$DW$84, DW_AT_decl_line(0xed) + 1059 .dwattr $C$DW$84, DW_AT_decl_column(0x0d) + 1060 .dwendtag $C$DW$T$31 + 1061 + 1062 .dwattr $C$DW$T$31, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1063 .dwattr $C$DW$T$31, DW_AT_decl_line(0xeb) + 1064 .dwattr $C$DW$T$31, DW_AT_decl_column(0x13) + 1065$C$DW$T$72 .dwtag DW_TAG_volatile_type + 1066 .dwattr $C$DW$T$72, DW_AT_type(*$C$DW$T$31) + 1067 + 1068$C$DW$T$35 .dwtag DW_TAG_structure_type + 1069 .dwattr $C$DW$T$35, DW_AT_byte_size(0x44) + 1070$C$DW$85 .dwtag DW_TAG_member + 1071 .dwattr $C$DW$85, DW_AT_type(*$C$DW$T$49) + 1072 .dwattr $C$DW$85, DW_AT_name("$P$T0") + 1073 .dwattr $C$DW$85, DW_AT_TI_symbol_name("$P$T0") + 1074 .dwattr $C$DW$85, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1075 .dwattr $C$DW$85, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1076 .dwattr $C$DW$85, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1077 .dwattr $C$DW$85, DW_AT_decl_line(0x29) + 1078 .dwattr $C$DW$85, DW_AT_decl_column(0x02) + 1079$C$DW$86 .dwtag DW_TAG_member + 1080 .dwattr $C$DW$86, DW_AT_type(*$C$DW$T$51) + 1081 .dwattr $C$DW$86, DW_AT_name("$P$T1") + 1082 .dwattr $C$DW$86, DW_AT_TI_symbol_name("$P$T1") + 1083 .dwattr $C$DW$86, DW_AT_data_member_location[DW_OP_plus_uconst 0x4] + 1084 .dwattr $C$DW$86, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1085 .dwattr $C$DW$86, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1086 .dwattr $C$DW$86, DW_AT_decl_line(0x33) + 1087 .dwattr $C$DW$86, DW_AT_decl_column(0x02) + 1088$C$DW$87 .dwtag DW_TAG_member + 1089 .dwattr $C$DW$87, DW_AT_type(*$C$DW$T$53) + 1090 .dwattr $C$DW$87, DW_AT_name("$P$T2") + 1091 .dwattr $C$DW$87, DW_AT_TI_symbol_name("$P$T2") + 1092 .dwattr $C$DW$87, DW_AT_data_member_location[DW_OP_plus_uconst 0x8] + 1093 .dwattr $C$DW$87, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1094 .dwattr $C$DW$87, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1095 .dwattr $C$DW$87, DW_AT_decl_line(0x41) + 1096 .dwattr $C$DW$87, DW_AT_decl_column(0x02) + 1097$C$DW$88 .dwtag DW_TAG_member + 1098 .dwattr $C$DW$88, DW_AT_type(*$C$DW$T$55) + 1099 .dwattr $C$DW$88, DW_AT_name("$P$T3") + 1100 .dwattr $C$DW$88, DW_AT_TI_symbol_name("$P$T3") +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 21 + + 1101 .dwattr $C$DW$88, DW_AT_data_member_location[DW_OP_plus_uconst 0xc] + 1102 .dwattr $C$DW$88, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1103 .dwattr $C$DW$88, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1104 .dwattr $C$DW$88, DW_AT_decl_line(0x54) + 1105 .dwattr $C$DW$88, DW_AT_decl_column(0x02) + 1106$C$DW$89 .dwtag DW_TAG_member + 1107 .dwattr $C$DW$89, DW_AT_type(*$C$DW$T$57) + 1108 .dwattr $C$DW$89, DW_AT_name("$P$T4") + 1109 .dwattr $C$DW$89, DW_AT_TI_symbol_name("$P$T4") + 1110 .dwattr $C$DW$89, DW_AT_data_member_location[DW_OP_plus_uconst 0x10] + 1111 .dwattr $C$DW$89, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1112 .dwattr $C$DW$89, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1113 .dwattr $C$DW$89, DW_AT_decl_line(0x67) + 1114 .dwattr $C$DW$89, DW_AT_decl_column(0x02) + 1115$C$DW$90 .dwtag DW_TAG_member + 1116 .dwattr $C$DW$90, DW_AT_type(*$C$DW$T$59) + 1117 .dwattr $C$DW$90, DW_AT_name("$P$T5") + 1118 .dwattr $C$DW$90, DW_AT_TI_symbol_name("$P$T5") + 1119 .dwattr $C$DW$90, DW_AT_data_member_location[DW_OP_plus_uconst 0x14] + 1120 .dwattr $C$DW$90, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1121 .dwattr $C$DW$90, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1122 .dwattr $C$DW$90, DW_AT_decl_line(0x83) + 1123 .dwattr $C$DW$90, DW_AT_decl_column(0x02) + 1124$C$DW$91 .dwtag DW_TAG_member + 1125 .dwattr $C$DW$91, DW_AT_type(*$C$DW$T$61) + 1126 .dwattr $C$DW$91, DW_AT_name("$P$T6") + 1127 .dwattr $C$DW$91, DW_AT_TI_symbol_name("$P$T6") + 1128 .dwattr $C$DW$91, DW_AT_data_member_location[DW_OP_plus_uconst 0x18] + 1129 .dwattr $C$DW$91, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1130 .dwattr $C$DW$91, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1131 .dwattr $C$DW$91, DW_AT_decl_line(0x92) + 1132 .dwattr $C$DW$91, DW_AT_decl_column(0x02) + 1133$C$DW$92 .dwtag DW_TAG_member + 1134 .dwattr $C$DW$92, DW_AT_type(*$C$DW$T$63) + 1135 .dwattr $C$DW$92, DW_AT_name("$P$T7") + 1136 .dwattr $C$DW$92, DW_AT_TI_symbol_name("$P$T7") + 1137 .dwattr $C$DW$92, DW_AT_data_member_location[DW_OP_plus_uconst 0x1c] + 1138 .dwattr $C$DW$92, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1139 .dwattr $C$DW$92, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1140 .dwattr $C$DW$92, DW_AT_decl_line(0xa0) + 1141 .dwattr $C$DW$92, DW_AT_decl_column(0x02) + 1142$C$DW$93 .dwtag DW_TAG_member + 1143 .dwattr $C$DW$93, DW_AT_type(*$C$DW$T$65) + 1144 .dwattr $C$DW$93, DW_AT_name("$P$T8") + 1145 .dwattr $C$DW$93, DW_AT_TI_symbol_name("$P$T8") + 1146 .dwattr $C$DW$93, DW_AT_data_member_location[DW_OP_plus_uconst 0x20] + 1147 .dwattr $C$DW$93, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1148 .dwattr $C$DW$93, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1149 .dwattr $C$DW$93, DW_AT_decl_line(0xaf) + 1150 .dwattr $C$DW$93, DW_AT_decl_column(0x02) + 1151$C$DW$94 .dwtag DW_TAG_member + 1152 .dwattr $C$DW$94, DW_AT_type(*$C$DW$T$32) + 1153 .dwattr $C$DW$94, DW_AT_name("rsvd24") + 1154 .dwattr $C$DW$94, DW_AT_TI_symbol_name("rsvd24") + 1155 .dwattr $C$DW$94, DW_AT_data_member_location[DW_OP_plus_uconst 0x24] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 22 + + 1156 .dwattr $C$DW$94, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1157 .dwattr $C$DW$94, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1158 .dwattr $C$DW$94, DW_AT_decl_line(0xbc) + 1159 .dwattr $C$DW$94, DW_AT_decl_column(0x0b) + 1160$C$DW$95 .dwtag DW_TAG_member + 1161 .dwattr $C$DW$95, DW_AT_type(*$C$DW$T$67) + 1162 .dwattr $C$DW$95, DW_AT_name("$P$T9") + 1163 .dwattr $C$DW$95, DW_AT_TI_symbol_name("$P$T9") + 1164 .dwattr $C$DW$95, DW_AT_data_member_location[DW_OP_plus_uconst 0x28] + 1165 .dwattr $C$DW$95, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1166 .dwattr $C$DW$95, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1167 .dwattr $C$DW$95, DW_AT_decl_line(0xc0) + 1168 .dwattr $C$DW$95, DW_AT_decl_column(0x02) + 1169$C$DW$96 .dwtag DW_TAG_member + 1170 .dwattr $C$DW$96, DW_AT_type(*$C$DW$T$33) + 1171 .dwattr $C$DW$96, DW_AT_name("rsvd2c") + 1172 .dwattr $C$DW$96, DW_AT_TI_symbol_name("rsvd2c") + 1173 .dwattr $C$DW$96, DW_AT_data_member_location[DW_OP_plus_uconst 0x2c] + 1174 .dwattr $C$DW$96, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1175 .dwattr $C$DW$96, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1176 .dwattr $C$DW$96, DW_AT_decl_line(0xcb) + 1177 .dwattr $C$DW$96, DW_AT_decl_column(0x0b) + 1178$C$DW$97 .dwtag DW_TAG_member + 1179 .dwattr $C$DW$97, DW_AT_type(*$C$DW$T$69) + 1180 .dwattr $C$DW$97, DW_AT_name("$P$T10") + 1181 .dwattr $C$DW$97, DW_AT_TI_symbol_name("$P$T10") + 1182 .dwattr $C$DW$97, DW_AT_data_member_location[DW_OP_plus_uconst 0x30] + 1183 .dwattr $C$DW$97, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1184 .dwattr $C$DW$97, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1185 .dwattr $C$DW$97, DW_AT_decl_line(0xcf) + 1186 .dwattr $C$DW$97, DW_AT_decl_column(0x02) + 1187$C$DW$98 .dwtag DW_TAG_member + 1188 .dwattr $C$DW$98, DW_AT_type(*$C$DW$T$71) + 1189 .dwattr $C$DW$98, DW_AT_name("$P$T11") + 1190 .dwattr $C$DW$98, DW_AT_TI_symbol_name("$P$T11") + 1191 .dwattr $C$DW$98, DW_AT_data_member_location[DW_OP_plus_uconst 0x34] + 1192 .dwattr $C$DW$98, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1193 .dwattr $C$DW$98, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1194 .dwattr $C$DW$98, DW_AT_decl_line(0xda) + 1195 .dwattr $C$DW$98, DW_AT_decl_column(0x02) + 1196$C$DW$99 .dwtag DW_TAG_member + 1197 .dwattr $C$DW$99, DW_AT_type(*$C$DW$T$34) + 1198 .dwattr $C$DW$99, DW_AT_name("rsvd38") + 1199 .dwattr $C$DW$99, DW_AT_TI_symbol_name("rsvd38") + 1200 .dwattr $C$DW$99, DW_AT_data_member_location[DW_OP_plus_uconst 0x38] + 1201 .dwattr $C$DW$99, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1202 .dwattr $C$DW$99, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pru + 1203 .dwattr $C$DW$99, DW_AT_decl_line(0xe5) + 1204 .dwattr $C$DW$99, DW_AT_decl_column(0x0b) + 1205$C$DW$100 .dwtag DW_TAG_member + 1206 .dwattr $C$DW$100, DW_AT_type(*$C$DW$T$73) + 1207 .dwattr $C$DW$100, DW_AT_name("$P$T12") + 1208 .dwattr $C$DW$100, DW_AT_TI_symbol_name("$P$T12") + 1209 .dwattr $C$DW$100, DW_AT_data_member_location[DW_OP_plus_uconst 0x40] + 1210 .dwattr $C$DW$100, DW_AT_accessibility(DW_ACCESS_ai_64_lic) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 23 + + 1211 .dwattr $C$DW$100, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1212 .dwattr $C$DW$100, DW_AT_decl_line(0xe8) + 1213 .dwattr $C$DW$100, DW_AT_decl_column(0x02) + 1214 .dwendtag $C$DW$T$35 + 1215 + 1216 .dwattr $C$DW$T$35, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1217 .dwattr $C$DW$T$35, DW_AT_decl_line(0x26) + 1218 .dwattr $C$DW$T$35, DW_AT_decl_column(0x10) + 1219$C$DW$T$97 .dwtag DW_TAG_typedef, DW_AT_name("pruCfg") + 1220 .dwattr $C$DW$T$97, DW_AT_type(*$C$DW$T$35) + 1221 .dwattr $C$DW$T$97, DW_AT_language(DW_LANG_C) + 1222 .dwattr $C$DW$T$97, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1223 .dwattr $C$DW$T$97, DW_AT_decl_line(0xf0) + 1224 .dwattr $C$DW$T$97, DW_AT_decl_column(0x03) + 1225$C$DW$T$98 .dwtag DW_TAG_volatile_type + 1226 .dwattr $C$DW$T$98, DW_AT_type(*$C$DW$T$97) + 1227 + 1228$C$DW$T$36 .dwtag DW_TAG_structure_type + 1229 .dwattr $C$DW$T$36, DW_AT_byte_size(0x04) + 1230$C$DW$101 .dwtag DW_TAG_member + 1231 .dwattr $C$DW$101, DW_AT_type(*$C$DW$T$11) + 1232 .dwattr $C$DW$101, DW_AT_name("SOFT_RST_N") + 1233 .dwattr $C$DW$101, DW_AT_TI_symbol_name("SOFT_RST_N") + 1234 .dwattr $C$DW$101, DW_AT_bit_offset(0x1f), DW_AT_bit_size(0x01) + 1235 .dwattr $C$DW$101, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1236 .dwattr $C$DW$101, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1237 .dwattr $C$DW$101, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1238 .dwattr $C$DW$101, DW_AT_decl_line(0x2d) + 1239 .dwattr $C$DW$101, DW_AT_decl_column(0x0d) + 1240$C$DW$102 .dwtag DW_TAG_member + 1241 .dwattr $C$DW$102, DW_AT_type(*$C$DW$T$11) + 1242 .dwattr $C$DW$102, DW_AT_name("EN") + 1243 .dwattr $C$DW$102, DW_AT_TI_symbol_name("EN") + 1244 .dwattr $C$DW$102, DW_AT_bit_offset(0x1e), DW_AT_bit_size(0x01) + 1245 .dwattr $C$DW$102, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1246 .dwattr $C$DW$102, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1247 .dwattr $C$DW$102, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1248 .dwattr $C$DW$102, DW_AT_decl_line(0x2e) + 1249 .dwattr $C$DW$102, DW_AT_decl_column(0x0d) + 1250$C$DW$103 .dwtag DW_TAG_member + 1251 .dwattr $C$DW$103, DW_AT_type(*$C$DW$T$11) + 1252 .dwattr $C$DW$103, DW_AT_name("SLEEPING") + 1253 .dwattr $C$DW$103, DW_AT_TI_symbol_name("SLEEPING") + 1254 .dwattr $C$DW$103, DW_AT_bit_offset(0x1d), DW_AT_bit_size(0x01) + 1255 .dwattr $C$DW$103, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1256 .dwattr $C$DW$103, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1257 .dwattr $C$DW$103, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1258 .dwattr $C$DW$103, DW_AT_decl_line(0x2f) + 1259 .dwattr $C$DW$103, DW_AT_decl_column(0x0d) + 1260$C$DW$104 .dwtag DW_TAG_member + 1261 .dwattr $C$DW$104, DW_AT_type(*$C$DW$T$11) + 1262 .dwattr $C$DW$104, DW_AT_name("CTR_EN") + 1263 .dwattr $C$DW$104, DW_AT_TI_symbol_name("CTR_EN") + 1264 .dwattr $C$DW$104, DW_AT_bit_offset(0x1c), DW_AT_bit_size(0x01) + 1265 .dwattr $C$DW$104, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 24 + + 1266 .dwattr $C$DW$104, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1267 .dwattr $C$DW$104, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1268 .dwattr $C$DW$104, DW_AT_decl_line(0x30) + 1269 .dwattr $C$DW$104, DW_AT_decl_column(0x0d) + 1270$C$DW$105 .dwtag DW_TAG_member + 1271 .dwattr $C$DW$105, DW_AT_type(*$C$DW$T$11) + 1272 .dwattr $C$DW$105, DW_AT_name("rsvd4") + 1273 .dwattr $C$DW$105, DW_AT_TI_symbol_name("rsvd4") + 1274 .dwattr $C$DW$105, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x04) + 1275 .dwattr $C$DW$105, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1276 .dwattr $C$DW$105, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1277 .dwattr $C$DW$105, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1278 .dwattr $C$DW$105, DW_AT_decl_line(0x31) + 1279 .dwattr $C$DW$105, DW_AT_decl_column(0x0d) + 1280$C$DW$106 .dwtag DW_TAG_member + 1281 .dwattr $C$DW$106, DW_AT_type(*$C$DW$T$11) + 1282 .dwattr $C$DW$106, DW_AT_name("SINGLE_STEP") + 1283 .dwattr $C$DW$106, DW_AT_TI_symbol_name("SINGLE_STEP") + 1284 .dwattr $C$DW$106, DW_AT_bit_offset(0x17), DW_AT_bit_size(0x01) + 1285 .dwattr $C$DW$106, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1286 .dwattr $C$DW$106, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1287 .dwattr $C$DW$106, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1288 .dwattr $C$DW$106, DW_AT_decl_line(0x32) + 1289 .dwattr $C$DW$106, DW_AT_decl_column(0x0d) + 1290$C$DW$107 .dwtag DW_TAG_member + 1291 .dwattr $C$DW$107, DW_AT_type(*$C$DW$T$11) + 1292 .dwattr $C$DW$107, DW_AT_name("rsvd9") + 1293 .dwattr $C$DW$107, DW_AT_TI_symbol_name("rsvd9") + 1294 .dwattr $C$DW$107, DW_AT_bit_offset(0x11), DW_AT_bit_size(0x06) + 1295 .dwattr $C$DW$107, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1296 .dwattr $C$DW$107, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1297 .dwattr $C$DW$107, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1298 .dwattr $C$DW$107, DW_AT_decl_line(0x33) + 1299 .dwattr $C$DW$107, DW_AT_decl_column(0x0d) + 1300$C$DW$108 .dwtag DW_TAG_member + 1301 .dwattr $C$DW$108, DW_AT_type(*$C$DW$T$11) + 1302 .dwattr $C$DW$108, DW_AT_name("RUNSTATE") + 1303 .dwattr $C$DW$108, DW_AT_TI_symbol_name("RUNSTATE") + 1304 .dwattr $C$DW$108, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x01) + 1305 .dwattr $C$DW$108, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1306 .dwattr $C$DW$108, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1307 .dwattr $C$DW$108, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1308 .dwattr $C$DW$108, DW_AT_decl_line(0x34) + 1309 .dwattr $C$DW$108, DW_AT_decl_column(0x0d) + 1310$C$DW$109 .dwtag DW_TAG_member + 1311 .dwattr $C$DW$109, DW_AT_type(*$C$DW$T$11) + 1312 .dwattr $C$DW$109, DW_AT_name("PCTR_RST_VAL") + 1313 .dwattr $C$DW$109, DW_AT_TI_symbol_name("PCTR_RST_VAL") + 1314 .dwattr $C$DW$109, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x10) + 1315 .dwattr $C$DW$109, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1316 .dwattr $C$DW$109, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1317 .dwattr $C$DW$109, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1318 .dwattr $C$DW$109, DW_AT_decl_line(0x35) + 1319 .dwattr $C$DW$109, DW_AT_decl_column(0x0d) + 1320 .dwendtag $C$DW$T$36 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 25 + + 1321 + 1322 .dwattr $C$DW$T$36, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1323 .dwattr $C$DW$T$36, DW_AT_decl_line(0x2c) + 1324 .dwattr $C$DW$T$36, DW_AT_decl_column(0x13) + 1325$C$DW$T$74 .dwtag DW_TAG_volatile_type + 1326 .dwattr $C$DW$T$74, DW_AT_type(*$C$DW$T$36) + 1327 + 1328$C$DW$T$37 .dwtag DW_TAG_structure_type + 1329 .dwattr $C$DW$T$37, DW_AT_byte_size(0x04) + 1330$C$DW$110 .dwtag DW_TAG_member + 1331 .dwattr $C$DW$110, DW_AT_type(*$C$DW$T$11) + 1332 .dwattr $C$DW$110, DW_AT_name("PCTR") + 1333 .dwattr $C$DW$110, DW_AT_TI_symbol_name("PCTR") + 1334 .dwattr $C$DW$110, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x10) + 1335 .dwattr $C$DW$110, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1336 .dwattr $C$DW$110, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1337 .dwattr $C$DW$110, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1338 .dwattr $C$DW$110, DW_AT_decl_line(0x3f) + 1339 .dwattr $C$DW$110, DW_AT_decl_column(0x0d) + 1340$C$DW$111 .dwtag DW_TAG_member + 1341 .dwattr $C$DW$111, DW_AT_type(*$C$DW$T$11) + 1342 .dwattr $C$DW$111, DW_AT_name("rsvd16") + 1343 .dwattr $C$DW$111, DW_AT_TI_symbol_name("rsvd16") + 1344 .dwattr $C$DW$111, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x10) + 1345 .dwattr $C$DW$111, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1346 .dwattr $C$DW$111, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1347 .dwattr $C$DW$111, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1348 .dwattr $C$DW$111, DW_AT_decl_line(0x40) + 1349 .dwattr $C$DW$111, DW_AT_decl_column(0x0d) + 1350 .dwendtag $C$DW$T$37 + 1351 + 1352 .dwattr $C$DW$T$37, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1353 .dwattr $C$DW$T$37, DW_AT_decl_line(0x3e) + 1354 .dwattr $C$DW$T$37, DW_AT_decl_column(0x13) + 1355$C$DW$T$76 .dwtag DW_TAG_volatile_type + 1356 .dwattr $C$DW$T$76, DW_AT_type(*$C$DW$T$37) + 1357 + 1358$C$DW$T$38 .dwtag DW_TAG_structure_type + 1359 .dwattr $C$DW$T$38, DW_AT_byte_size(0x04) + 1360$C$DW$112 .dwtag DW_TAG_member + 1361 .dwattr $C$DW$112, DW_AT_type(*$C$DW$T$11) + 1362 .dwattr $C$DW$112, DW_AT_name("BITWISE_ENS") + 1363 .dwattr $C$DW$112, DW_AT_TI_symbol_name("BITWISE_ENS") + 1364 .dwattr $C$DW$112, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x20) + 1365 .dwattr $C$DW$112, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1366 .dwattr $C$DW$112, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1367 .dwattr $C$DW$112, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1368 .dwattr $C$DW$112, DW_AT_decl_line(0x4a) + 1369 .dwattr $C$DW$112, DW_AT_decl_column(0x0d) + 1370 .dwendtag $C$DW$T$38 + 1371 + 1372 .dwattr $C$DW$T$38, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1373 .dwattr $C$DW$T$38, DW_AT_decl_line(0x49) + 1374 .dwattr $C$DW$T$38, DW_AT_decl_column(0x13) + 1375$C$DW$T$78 .dwtag DW_TAG_volatile_type +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 26 + + 1376 .dwattr $C$DW$T$78, DW_AT_type(*$C$DW$T$38) + 1377 + 1378$C$DW$T$39 .dwtag DW_TAG_structure_type + 1379 .dwattr $C$DW$T$39, DW_AT_byte_size(0x04) + 1380$C$DW$113 .dwtag DW_TAG_member + 1381 .dwattr $C$DW$113, DW_AT_type(*$C$DW$T$11) + 1382 .dwattr $C$DW$113, DW_AT_name("CYCLECOUNT") + 1383 .dwattr $C$DW$113, DW_AT_TI_symbol_name("CYCLECOUNT") + 1384 .dwattr $C$DW$113, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x20) + 1385 .dwattr $C$DW$113, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1386 .dwattr $C$DW$113, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1387 .dwattr $C$DW$113, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1388 .dwattr $C$DW$113, DW_AT_decl_line(0x54) + 1389 .dwattr $C$DW$113, DW_AT_decl_column(0x0d) + 1390 .dwendtag $C$DW$T$39 + 1391 + 1392 .dwattr $C$DW$T$39, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1393 .dwattr $C$DW$T$39, DW_AT_decl_line(0x53) + 1394 .dwattr $C$DW$T$39, DW_AT_decl_column(0x13) + 1395$C$DW$T$80 .dwtag DW_TAG_volatile_type + 1396 .dwattr $C$DW$T$80, DW_AT_type(*$C$DW$T$39) + 1397 + 1398$C$DW$T$40 .dwtag DW_TAG_structure_type + 1399 .dwattr $C$DW$T$40, DW_AT_byte_size(0x04) + 1400$C$DW$114 .dwtag DW_TAG_member + 1401 .dwattr $C$DW$114, DW_AT_type(*$C$DW$T$11) + 1402 .dwattr $C$DW$114, DW_AT_name("STALLCOUNT") + 1403 .dwattr $C$DW$114, DW_AT_TI_symbol_name("STALLCOUNT") + 1404 .dwattr $C$DW$114, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x20) + 1405 .dwattr $C$DW$114, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1406 .dwattr $C$DW$114, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1407 .dwattr $C$DW$114, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1408 .dwattr $C$DW$114, DW_AT_decl_line(0x5e) + 1409 .dwattr $C$DW$114, DW_AT_decl_column(0x0d) + 1410 .dwendtag $C$DW$T$40 + 1411 + 1412 .dwattr $C$DW$T$40, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1413 .dwattr $C$DW$T$40, DW_AT_decl_line(0x5d) + 1414 .dwattr $C$DW$T$40, DW_AT_decl_column(0x14) + 1415$C$DW$T$82 .dwtag DW_TAG_volatile_type + 1416 .dwattr $C$DW$T$82, DW_AT_type(*$C$DW$T$40) + 1417 + 1418$C$DW$T$41 .dwtag DW_TAG_structure_type + 1419 .dwattr $C$DW$T$41, DW_AT_byte_size(0x04) + 1420$C$DW$115 .dwtag DW_TAG_member + 1421 .dwattr $C$DW$115, DW_AT_type(*$C$DW$T$11) + 1422 .dwattr $C$DW$115, DW_AT_name("C24_BLK_IDX") + 1423 .dwattr $C$DW$115, DW_AT_TI_symbol_name("C24_BLK_IDX") + 1424 .dwattr $C$DW$115, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x08) + 1425 .dwattr $C$DW$115, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1426 .dwattr $C$DW$115, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1427 .dwattr $C$DW$115, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1428 .dwattr $C$DW$115, DW_AT_decl_line(0x6b) + 1429 .dwattr $C$DW$115, DW_AT_decl_column(0x0d) + 1430$C$DW$116 .dwtag DW_TAG_member +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 27 + + 1431 .dwattr $C$DW$116, DW_AT_type(*$C$DW$T$11) + 1432 .dwattr $C$DW$116, DW_AT_name("rsvd8") + 1433 .dwattr $C$DW$116, DW_AT_TI_symbol_name("rsvd8") + 1434 .dwattr $C$DW$116, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x08) + 1435 .dwattr $C$DW$116, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1436 .dwattr $C$DW$116, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1437 .dwattr $C$DW$116, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1438 .dwattr $C$DW$116, DW_AT_decl_line(0x6c) + 1439 .dwattr $C$DW$116, DW_AT_decl_column(0x0d) + 1440$C$DW$117 .dwtag DW_TAG_member + 1441 .dwattr $C$DW$117, DW_AT_type(*$C$DW$T$11) + 1442 .dwattr $C$DW$117, DW_AT_name("C25_BLK_IDX") + 1443 .dwattr $C$DW$117, DW_AT_TI_symbol_name("C25_BLK_IDX") + 1444 .dwattr $C$DW$117, DW_AT_bit_offset(0x08), DW_AT_bit_size(0x08) + 1445 .dwattr $C$DW$117, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1446 .dwattr $C$DW$117, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1447 .dwattr $C$DW$117, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1448 .dwattr $C$DW$117, DW_AT_decl_line(0x6d) + 1449 .dwattr $C$DW$117, DW_AT_decl_column(0x0d) + 1450$C$DW$118 .dwtag DW_TAG_member + 1451 .dwattr $C$DW$118, DW_AT_type(*$C$DW$T$11) + 1452 .dwattr $C$DW$118, DW_AT_name("rsvd24") + 1453 .dwattr $C$DW$118, DW_AT_TI_symbol_name("rsvd24") + 1454 .dwattr $C$DW$118, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x08) + 1455 .dwattr $C$DW$118, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1456 .dwattr $C$DW$118, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1457 .dwattr $C$DW$118, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1458 .dwattr $C$DW$118, DW_AT_decl_line(0x6e) + 1459 .dwattr $C$DW$118, DW_AT_decl_column(0x0d) + 1460 .dwendtag $C$DW$T$41 + 1461 + 1462 .dwattr $C$DW$T$41, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1463 .dwattr $C$DW$T$41, DW_AT_decl_line(0x6a) + 1464 .dwattr $C$DW$T$41, DW_AT_decl_column(0x13) + 1465$C$DW$T$84 .dwtag DW_TAG_volatile_type + 1466 .dwattr $C$DW$T$84, DW_AT_type(*$C$DW$T$41) + 1467 + 1468$C$DW$T$42 .dwtag DW_TAG_structure_type + 1469 .dwattr $C$DW$T$42, DW_AT_byte_size(0x04) + 1470$C$DW$119 .dwtag DW_TAG_member + 1471 .dwattr $C$DW$119, DW_AT_type(*$C$DW$T$11) + 1472 .dwattr $C$DW$119, DW_AT_name("C26_BLK_IDX") + 1473 .dwattr $C$DW$119, DW_AT_TI_symbol_name("C26_BLK_IDX") + 1474 .dwattr $C$DW$119, DW_AT_bit_offset(0x18), DW_AT_bit_size(0x08) + 1475 .dwattr $C$DW$119, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1476 .dwattr $C$DW$119, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1477 .dwattr $C$DW$119, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1478 .dwattr $C$DW$119, DW_AT_decl_line(0x78) + 1479 .dwattr $C$DW$119, DW_AT_decl_column(0x0d) + 1480$C$DW$120 .dwtag DW_TAG_member + 1481 .dwattr $C$DW$120, DW_AT_type(*$C$DW$T$11) + 1482 .dwattr $C$DW$120, DW_AT_name("rsvd8") + 1483 .dwattr $C$DW$120, DW_AT_TI_symbol_name("rsvd8") + 1484 .dwattr $C$DW$120, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x08) + 1485 .dwattr $C$DW$120, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 28 + + 1486 .dwattr $C$DW$120, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1487 .dwattr $C$DW$120, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1488 .dwattr $C$DW$120, DW_AT_decl_line(0x79) + 1489 .dwattr $C$DW$120, DW_AT_decl_column(0x0d) + 1490$C$DW$121 .dwtag DW_TAG_member + 1491 .dwattr $C$DW$121, DW_AT_type(*$C$DW$T$11) + 1492 .dwattr $C$DW$121, DW_AT_name("C27_BLK_IDX") + 1493 .dwattr $C$DW$121, DW_AT_TI_symbol_name("C27_BLK_IDX") + 1494 .dwattr $C$DW$121, DW_AT_bit_offset(0x08), DW_AT_bit_size(0x08) + 1495 .dwattr $C$DW$121, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1496 .dwattr $C$DW$121, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1497 .dwattr $C$DW$121, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1498 .dwattr $C$DW$121, DW_AT_decl_line(0x7a) + 1499 .dwattr $C$DW$121, DW_AT_decl_column(0x0d) + 1500$C$DW$122 .dwtag DW_TAG_member + 1501 .dwattr $C$DW$122, DW_AT_type(*$C$DW$T$11) + 1502 .dwattr $C$DW$122, DW_AT_name("rsvd24") + 1503 .dwattr $C$DW$122, DW_AT_TI_symbol_name("rsvd24") + 1504 .dwattr $C$DW$122, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x08) + 1505 .dwattr $C$DW$122, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1506 .dwattr $C$DW$122, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1507 .dwattr $C$DW$122, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1508 .dwattr $C$DW$122, DW_AT_decl_line(0x7b) + 1509 .dwattr $C$DW$122, DW_AT_decl_column(0x0d) + 1510 .dwendtag $C$DW$T$42 + 1511 + 1512 .dwattr $C$DW$T$42, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1513 .dwattr $C$DW$T$42, DW_AT_decl_line(0x77) + 1514 .dwattr $C$DW$T$42, DW_AT_decl_column(0x13) + 1515$C$DW$T$86 .dwtag DW_TAG_volatile_type + 1516 .dwattr $C$DW$T$86, DW_AT_type(*$C$DW$T$42) + 1517 + 1518$C$DW$T$43 .dwtag DW_TAG_structure_type + 1519 .dwattr $C$DW$T$43, DW_AT_byte_size(0x04) + 1520$C$DW$123 .dwtag DW_TAG_member + 1521 .dwattr $C$DW$123, DW_AT_type(*$C$DW$T$11) + 1522 .dwattr $C$DW$123, DW_AT_name("C28_BLK_POINTER") + 1523 .dwattr $C$DW$123, DW_AT_TI_symbol_name("C28_BLK_POINTER") + 1524 .dwattr $C$DW$123, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x10) + 1525 .dwattr $C$DW$123, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1526 .dwattr $C$DW$123, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1527 .dwattr $C$DW$123, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1528 .dwattr $C$DW$123, DW_AT_decl_line(0x85) + 1529 .dwattr $C$DW$123, DW_AT_decl_column(0x0d) + 1530$C$DW$124 .dwtag DW_TAG_member + 1531 .dwattr $C$DW$124, DW_AT_type(*$C$DW$T$11) + 1532 .dwattr $C$DW$124, DW_AT_name("C29_BLK_POINTER") + 1533 .dwattr $C$DW$124, DW_AT_TI_symbol_name("C29_BLK_POINTER") + 1534 .dwattr $C$DW$124, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x10) + 1535 .dwattr $C$DW$124, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1536 .dwattr $C$DW$124, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1537 .dwattr $C$DW$124, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1538 .dwattr $C$DW$124, DW_AT_decl_line(0x86) + 1539 .dwattr $C$DW$124, DW_AT_decl_column(0x0d) + 1540 .dwendtag $C$DW$T$43 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 29 + + 1541 + 1542 .dwattr $C$DW$T$43, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1543 .dwattr $C$DW$T$43, DW_AT_decl_line(0x84) + 1544 .dwattr $C$DW$T$43, DW_AT_decl_column(0x13) + 1545$C$DW$T$88 .dwtag DW_TAG_volatile_type + 1546 .dwattr $C$DW$T$88, DW_AT_type(*$C$DW$T$43) + 1547 + 1548$C$DW$T$44 .dwtag DW_TAG_structure_type + 1549 .dwattr $C$DW$T$44, DW_AT_byte_size(0x04) + 1550$C$DW$125 .dwtag DW_TAG_member + 1551 .dwattr $C$DW$125, DW_AT_type(*$C$DW$T$11) + 1552 .dwattr $C$DW$125, DW_AT_name("C30_BLK_POINTER") + 1553 .dwattr $C$DW$125, DW_AT_TI_symbol_name("C30_BLK_POINTER") + 1554 .dwattr $C$DW$125, DW_AT_bit_offset(0x10), DW_AT_bit_size(0x10) + 1555 .dwattr $C$DW$125, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1556 .dwattr $C$DW$125, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1557 .dwattr $C$DW$125, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1558 .dwattr $C$DW$125, DW_AT_decl_line(0x90) + 1559 .dwattr $C$DW$125, DW_AT_decl_column(0x0d) + 1560$C$DW$126 .dwtag DW_TAG_member + 1561 .dwattr $C$DW$126, DW_AT_type(*$C$DW$T$11) + 1562 .dwattr $C$DW$126, DW_AT_name("C31_BLK_POINTER") + 1563 .dwattr $C$DW$126, DW_AT_TI_symbol_name("C31_BLK_POINTER") + 1564 .dwattr $C$DW$126, DW_AT_bit_offset(0x00), DW_AT_bit_size(0x10) + 1565 .dwattr $C$DW$126, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1566 .dwattr $C$DW$126, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1567 .dwattr $C$DW$126, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1568 .dwattr $C$DW$126, DW_AT_decl_line(0x91) + 1569 .dwattr $C$DW$126, DW_AT_decl_column(0x0d) + 1570 .dwendtag $C$DW$T$44 + 1571 + 1572 .dwattr $C$DW$T$44, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1573 .dwattr $C$DW$T$44, DW_AT_decl_line(0x8f) + 1574 .dwattr $C$DW$T$44, DW_AT_decl_column(0x13) + 1575$C$DW$T$90 .dwtag DW_TAG_volatile_type + 1576 .dwattr $C$DW$T$90, DW_AT_type(*$C$DW$T$44) + 1577 + 1578$C$DW$T$46 .dwtag DW_TAG_structure_type + 1579 .dwattr $C$DW$T$46, DW_AT_byte_size(0x30) + 1580$C$DW$127 .dwtag DW_TAG_member + 1581 .dwattr $C$DW$127, DW_AT_type(*$C$DW$T$75) + 1582 .dwattr $C$DW$127, DW_AT_name("$P$T13") + 1583 .dwattr $C$DW$127, DW_AT_TI_symbol_name("$P$T13") + 1584 .dwattr $C$DW$127, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1585 .dwattr $C$DW$127, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1586 .dwattr $C$DW$127, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1587 .dwattr $C$DW$127, DW_AT_decl_line(0x29) + 1588 .dwattr $C$DW$127, DW_AT_decl_column(0x02) + 1589$C$DW$128 .dwtag DW_TAG_member + 1590 .dwattr $C$DW$128, DW_AT_type(*$C$DW$T$77) + 1591 .dwattr $C$DW$128, DW_AT_name("$P$T14") + 1592 .dwattr $C$DW$128, DW_AT_TI_symbol_name("$P$T14") + 1593 .dwattr $C$DW$128, DW_AT_data_member_location[DW_OP_plus_uconst 0x4] + 1594 .dwattr $C$DW$128, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1595 .dwattr $C$DW$128, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 30 + + 1596 .dwattr $C$DW$128, DW_AT_decl_line(0x3b) + 1597 .dwattr $C$DW$128, DW_AT_decl_column(0x02) + 1598$C$DW$129 .dwtag DW_TAG_member + 1599 .dwattr $C$DW$129, DW_AT_type(*$C$DW$T$79) + 1600 .dwattr $C$DW$129, DW_AT_name("$P$T15") + 1601 .dwattr $C$DW$129, DW_AT_TI_symbol_name("$P$T15") + 1602 .dwattr $C$DW$129, DW_AT_data_member_location[DW_OP_plus_uconst 0x8] + 1603 .dwattr $C$DW$129, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1604 .dwattr $C$DW$129, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1605 .dwattr $C$DW$129, DW_AT_decl_line(0x46) + 1606 .dwattr $C$DW$129, DW_AT_decl_column(0x02) + 1607$C$DW$130 .dwtag DW_TAG_member + 1608 .dwattr $C$DW$130, DW_AT_type(*$C$DW$T$81) + 1609 .dwattr $C$DW$130, DW_AT_name("$P$T16") + 1610 .dwattr $C$DW$130, DW_AT_TI_symbol_name("$P$T16") + 1611 .dwattr $C$DW$130, DW_AT_data_member_location[DW_OP_plus_uconst 0xc] + 1612 .dwattr $C$DW$130, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1613 .dwattr $C$DW$130, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1614 .dwattr $C$DW$130, DW_AT_decl_line(0x50) + 1615 .dwattr $C$DW$130, DW_AT_decl_column(0x02) + 1616$C$DW$131 .dwtag DW_TAG_member + 1617 .dwattr $C$DW$131, DW_AT_type(*$C$DW$T$83) + 1618 .dwattr $C$DW$131, DW_AT_name("$P$T17") + 1619 .dwattr $C$DW$131, DW_AT_TI_symbol_name("$P$T17") + 1620 .dwattr $C$DW$131, DW_AT_data_member_location[DW_OP_plus_uconst 0x10] + 1621 .dwattr $C$DW$131, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1622 .dwattr $C$DW$131, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1623 .dwattr $C$DW$131, DW_AT_decl_line(0x5a) + 1624 .dwattr $C$DW$131, DW_AT_decl_column(0x02) + 1625$C$DW$132 .dwtag DW_TAG_member + 1626 .dwattr $C$DW$132, DW_AT_type(*$C$DW$T$45) + 1627 .dwattr $C$DW$132, DW_AT_name("rsvd14") + 1628 .dwattr $C$DW$132, DW_AT_TI_symbol_name("rsvd14") + 1629 .dwattr $C$DW$132, DW_AT_data_member_location[DW_OP_plus_uconst 0x14] + 1630 .dwattr $C$DW$132, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1631 .dwattr $C$DW$132, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1632 .dwattr $C$DW$132, DW_AT_decl_line(0x63) + 1633 .dwattr $C$DW$132, DW_AT_decl_column(0x0b) + 1634$C$DW$133 .dwtag DW_TAG_member + 1635 .dwattr $C$DW$133, DW_AT_type(*$C$DW$T$85) + 1636 .dwattr $C$DW$133, DW_AT_name("$P$T18") + 1637 .dwattr $C$DW$133, DW_AT_TI_symbol_name("$P$T18") + 1638 .dwattr $C$DW$133, DW_AT_data_member_location[DW_OP_plus_uconst 0x20] + 1639 .dwattr $C$DW$133, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1640 .dwattr $C$DW$133, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1641 .dwattr $C$DW$133, DW_AT_decl_line(0x67) + 1642 .dwattr $C$DW$133, DW_AT_decl_column(0x02) + 1643$C$DW$134 .dwtag DW_TAG_member + 1644 .dwattr $C$DW$134, DW_AT_type(*$C$DW$T$87) + 1645 .dwattr $C$DW$134, DW_AT_name("$P$T19") + 1646 .dwattr $C$DW$134, DW_AT_TI_symbol_name("$P$T19") + 1647 .dwattr $C$DW$134, DW_AT_data_member_location[DW_OP_plus_uconst 0x24] + 1648 .dwattr $C$DW$134, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1649 .dwattr $C$DW$134, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1650 .dwattr $C$DW$134, DW_AT_decl_line(0x74) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 31 + + 1651 .dwattr $C$DW$134, DW_AT_decl_column(0x02) + 1652$C$DW$135 .dwtag DW_TAG_member + 1653 .dwattr $C$DW$135, DW_AT_type(*$C$DW$T$89) + 1654 .dwattr $C$DW$135, DW_AT_name("$P$T20") + 1655 .dwattr $C$DW$135, DW_AT_TI_symbol_name("$P$T20") + 1656 .dwattr $C$DW$135, DW_AT_data_member_location[DW_OP_plus_uconst 0x28] + 1657 .dwattr $C$DW$135, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1658 .dwattr $C$DW$135, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1659 .dwattr $C$DW$135, DW_AT_decl_line(0x81) + 1660 .dwattr $C$DW$135, DW_AT_decl_column(0x02) + 1661$C$DW$136 .dwtag DW_TAG_member + 1662 .dwattr $C$DW$136, DW_AT_type(*$C$DW$T$91) + 1663 .dwattr $C$DW$136, DW_AT_name("$P$T21") + 1664 .dwattr $C$DW$136, DW_AT_TI_symbol_name("$P$T21") + 1665 .dwattr $C$DW$136, DW_AT_data_member_location[DW_OP_plus_uconst 0x2c] + 1666 .dwattr $C$DW$136, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1667 .dwattr $C$DW$136, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1668 .dwattr $C$DW$136, DW_AT_decl_line(0x8c) + 1669 .dwattr $C$DW$136, DW_AT_decl_column(0x02) + 1670 .dwendtag $C$DW$T$46 + 1671 + 1672 .dwattr $C$DW$T$46, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1673 .dwattr $C$DW$T$46, DW_AT_decl_line(0x26) + 1674 .dwattr $C$DW$T$46, DW_AT_decl_column(0x10) + 1675$C$DW$T$102 .dwtag DW_TAG_typedef, DW_AT_name("pruCtrl") + 1676 .dwattr $C$DW$T$102, DW_AT_type(*$C$DW$T$46) + 1677 .dwattr $C$DW$T$102, DW_AT_language(DW_LANG_C) + 1678 .dwattr $C$DW$T$102, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/ + 1679 .dwattr $C$DW$T$102, DW_AT_decl_line(0x95) + 1680 .dwattr $C$DW$T$102, DW_AT_decl_column(0x03) + 1681 + 1682$C$DW$T$49 .dwtag DW_TAG_union_type + 1683 .dwattr $C$DW$T$49, DW_AT_byte_size(0x04) + 1684$C$DW$137 .dwtag DW_TAG_member + 1685 .dwattr $C$DW$137, DW_AT_type(*$C$DW$T$47) + 1686 .dwattr $C$DW$137, DW_AT_name("REVID") + 1687 .dwattr $C$DW$137, DW_AT_TI_symbol_name("REVID") + 1688 .dwattr $C$DW$137, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1689 .dwattr $C$DW$137, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1690 .dwattr $C$DW$137, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1691 .dwattr $C$DW$137, DW_AT_decl_line(0x2a) + 1692 .dwattr $C$DW$137, DW_AT_decl_column(0x15) + 1693$C$DW$138 .dwtag DW_TAG_member + 1694 .dwattr $C$DW$138, DW_AT_type(*$C$DW$T$48) + 1695 .dwattr $C$DW$138, DW_AT_name("REVID_bit") + 1696 .dwattr $C$DW$138, DW_AT_TI_symbol_name("REVID_bit") + 1697 .dwattr $C$DW$138, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1698 .dwattr $C$DW$138, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1699 .dwattr $C$DW$138, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1700 .dwattr $C$DW$138, DW_AT_decl_line(0x2e) + 1701 .dwattr $C$DW$138, DW_AT_decl_column(0x05) + 1702 .dwendtag $C$DW$T$49 + 1703 + 1704 .dwattr $C$DW$T$49, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1705 .dwattr $C$DW$T$49, DW_AT_decl_line(0x29) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 32 + + 1706 .dwattr $C$DW$T$49, DW_AT_decl_column(0x08) + 1707 + 1708$C$DW$T$51 .dwtag DW_TAG_union_type + 1709 .dwattr $C$DW$T$51, DW_AT_byte_size(0x04) + 1710$C$DW$139 .dwtag DW_TAG_member + 1711 .dwattr $C$DW$139, DW_AT_type(*$C$DW$T$47) + 1712 .dwattr $C$DW$139, DW_AT_name("SYSCFG") + 1713 .dwattr $C$DW$139, DW_AT_TI_symbol_name("SYSCFG") + 1714 .dwattr $C$DW$139, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1715 .dwattr $C$DW$139, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1716 .dwattr $C$DW$139, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1717 .dwattr $C$DW$139, DW_AT_decl_line(0x34) + 1718 .dwattr $C$DW$139, DW_AT_decl_column(0x15) + 1719$C$DW$140 .dwtag DW_TAG_member + 1720 .dwattr $C$DW$140, DW_AT_type(*$C$DW$T$50) + 1721 .dwattr $C$DW$140, DW_AT_name("SYSCFG_bit") + 1722 .dwattr $C$DW$140, DW_AT_TI_symbol_name("SYSCFG_bit") + 1723 .dwattr $C$DW$140, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1724 .dwattr $C$DW$140, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1725 .dwattr $C$DW$140, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1726 .dwattr $C$DW$140, DW_AT_decl_line(0x3c) + 1727 .dwattr $C$DW$140, DW_AT_decl_column(0x05) + 1728 .dwendtag $C$DW$T$51 + 1729 + 1730 .dwattr $C$DW$T$51, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1731 .dwattr $C$DW$T$51, DW_AT_decl_line(0x33) + 1732 .dwattr $C$DW$T$51, DW_AT_decl_column(0x08) + 1733 + 1734$C$DW$T$53 .dwtag DW_TAG_union_type + 1735 .dwattr $C$DW$T$53, DW_AT_byte_size(0x04) + 1736$C$DW$141 .dwtag DW_TAG_member + 1737 .dwattr $C$DW$141, DW_AT_type(*$C$DW$T$47) + 1738 .dwattr $C$DW$141, DW_AT_name("GPCFG0") + 1739 .dwattr $C$DW$141, DW_AT_TI_symbol_name("GPCFG0") + 1740 .dwattr $C$DW$141, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1741 .dwattr $C$DW$141, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1742 .dwattr $C$DW$141, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1743 .dwattr $C$DW$141, DW_AT_decl_line(0x42) + 1744 .dwattr $C$DW$141, DW_AT_decl_column(0x15) + 1745$C$DW$142 .dwtag DW_TAG_member + 1746 .dwattr $C$DW$142, DW_AT_type(*$C$DW$T$52) + 1747 .dwattr $C$DW$142, DW_AT_name("GPCFG0_bit") + 1748 .dwattr $C$DW$142, DW_AT_TI_symbol_name("GPCFG0_bit") + 1749 .dwattr $C$DW$142, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1750 .dwattr $C$DW$142, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1751 .dwattr $C$DW$142, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1752 .dwattr $C$DW$142, DW_AT_decl_line(0x4f) + 1753 .dwattr $C$DW$142, DW_AT_decl_column(0x05) + 1754 .dwendtag $C$DW$T$53 + 1755 + 1756 .dwattr $C$DW$T$53, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1757 .dwattr $C$DW$T$53, DW_AT_decl_line(0x41) + 1758 .dwattr $C$DW$T$53, DW_AT_decl_column(0x08) + 1759 + 1760$C$DW$T$55 .dwtag DW_TAG_union_type +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 33 + + 1761 .dwattr $C$DW$T$55, DW_AT_byte_size(0x04) + 1762$C$DW$143 .dwtag DW_TAG_member + 1763 .dwattr $C$DW$143, DW_AT_type(*$C$DW$T$47) + 1764 .dwattr $C$DW$143, DW_AT_name("GPCFG1") + 1765 .dwattr $C$DW$143, DW_AT_TI_symbol_name("GPCFG1") + 1766 .dwattr $C$DW$143, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1767 .dwattr $C$DW$143, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1768 .dwattr $C$DW$143, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1769 .dwattr $C$DW$143, DW_AT_decl_line(0x55) + 1770 .dwattr $C$DW$143, DW_AT_decl_column(0x15) + 1771$C$DW$144 .dwtag DW_TAG_member + 1772 .dwattr $C$DW$144, DW_AT_type(*$C$DW$T$54) + 1773 .dwattr $C$DW$144, DW_AT_name("GPCFG1_bit") + 1774 .dwattr $C$DW$144, DW_AT_TI_symbol_name("GPCFG1_bit") + 1775 .dwattr $C$DW$144, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1776 .dwattr $C$DW$144, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1777 .dwattr $C$DW$144, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1778 .dwattr $C$DW$144, DW_AT_decl_line(0x62) + 1779 .dwattr $C$DW$144, DW_AT_decl_column(0x05) + 1780 .dwendtag $C$DW$T$55 + 1781 + 1782 .dwattr $C$DW$T$55, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1783 .dwattr $C$DW$T$55, DW_AT_decl_line(0x54) + 1784 .dwattr $C$DW$T$55, DW_AT_decl_column(0x08) + 1785 + 1786$C$DW$T$57 .dwtag DW_TAG_union_type + 1787 .dwattr $C$DW$T$57, DW_AT_byte_size(0x04) + 1788$C$DW$145 .dwtag DW_TAG_member + 1789 .dwattr $C$DW$145, DW_AT_type(*$C$DW$T$47) + 1790 .dwattr $C$DW$145, DW_AT_name("CGR") + 1791 .dwattr $C$DW$145, DW_AT_TI_symbol_name("CGR") + 1792 .dwattr $C$DW$145, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1793 .dwattr $C$DW$145, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1794 .dwattr $C$DW$145, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1795 .dwattr $C$DW$145, DW_AT_decl_line(0x68) + 1796 .dwattr $C$DW$145, DW_AT_decl_column(0x15) + 1797$C$DW$146 .dwtag DW_TAG_member + 1798 .dwattr $C$DW$146, DW_AT_type(*$C$DW$T$56) + 1799 .dwattr $C$DW$146, DW_AT_name("CGR_bit") + 1800 .dwattr $C$DW$146, DW_AT_TI_symbol_name("CGR_bit") + 1801 .dwattr $C$DW$146, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1802 .dwattr $C$DW$146, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1803 .dwattr $C$DW$146, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1804 .dwattr $C$DW$146, DW_AT_decl_line(0x7e) + 1805 .dwattr $C$DW$146, DW_AT_decl_column(0x05) + 1806 .dwendtag $C$DW$T$57 + 1807 + 1808 .dwattr $C$DW$T$57, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1809 .dwattr $C$DW$T$57, DW_AT_decl_line(0x67) + 1810 .dwattr $C$DW$T$57, DW_AT_decl_column(0x08) + 1811 + 1812$C$DW$T$59 .dwtag DW_TAG_union_type + 1813 .dwattr $C$DW$T$59, DW_AT_byte_size(0x04) + 1814$C$DW$147 .dwtag DW_TAG_member + 1815 .dwattr $C$DW$147, DW_AT_type(*$C$DW$T$47) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 34 + + 1816 .dwattr $C$DW$147, DW_AT_name("ISRP") + 1817 .dwattr $C$DW$147, DW_AT_TI_symbol_name("ISRP") + 1818 .dwattr $C$DW$147, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1819 .dwattr $C$DW$147, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1820 .dwattr $C$DW$147, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1821 .dwattr $C$DW$147, DW_AT_decl_line(0x84) + 1822 .dwattr $C$DW$147, DW_AT_decl_column(0x15) + 1823$C$DW$148 .dwtag DW_TAG_member + 1824 .dwattr $C$DW$148, DW_AT_type(*$C$DW$T$58) + 1825 .dwattr $C$DW$148, DW_AT_name("ISRP_bit") + 1826 .dwattr $C$DW$148, DW_AT_TI_symbol_name("ISRP_bit") + 1827 .dwattr $C$DW$148, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1828 .dwattr $C$DW$148, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1829 .dwattr $C$DW$148, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1830 .dwattr $C$DW$148, DW_AT_decl_line(0x8d) + 1831 .dwattr $C$DW$148, DW_AT_decl_column(0x05) + 1832 .dwendtag $C$DW$T$59 + 1833 + 1834 .dwattr $C$DW$T$59, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1835 .dwattr $C$DW$T$59, DW_AT_decl_line(0x83) + 1836 .dwattr $C$DW$T$59, DW_AT_decl_column(0x08) + 1837 + 1838$C$DW$T$61 .dwtag DW_TAG_union_type + 1839 .dwattr $C$DW$T$61, DW_AT_byte_size(0x04) + 1840$C$DW$149 .dwtag DW_TAG_member + 1841 .dwattr $C$DW$149, DW_AT_type(*$C$DW$T$47) + 1842 .dwattr $C$DW$149, DW_AT_name("ISP") + 1843 .dwattr $C$DW$149, DW_AT_TI_symbol_name("ISP") + 1844 .dwattr $C$DW$149, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1845 .dwattr $C$DW$149, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1846 .dwattr $C$DW$149, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1847 .dwattr $C$DW$149, DW_AT_decl_line(0x93) + 1848 .dwattr $C$DW$149, DW_AT_decl_column(0x15) + 1849$C$DW$150 .dwtag DW_TAG_member + 1850 .dwattr $C$DW$150, DW_AT_type(*$C$DW$T$60) + 1851 .dwattr $C$DW$150, DW_AT_name("ISP_bit") + 1852 .dwattr $C$DW$150, DW_AT_TI_symbol_name("ISP_bit") + 1853 .dwattr $C$DW$150, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1854 .dwattr $C$DW$150, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1855 .dwattr $C$DW$150, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1856 .dwattr $C$DW$150, DW_AT_decl_line(0x9c) + 1857 .dwattr $C$DW$150, DW_AT_decl_column(0x05) + 1858 .dwendtag $C$DW$T$61 + 1859 + 1860 .dwattr $C$DW$T$61, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1861 .dwattr $C$DW$T$61, DW_AT_decl_line(0x92) + 1862 .dwattr $C$DW$T$61, DW_AT_decl_column(0x08) + 1863 + 1864$C$DW$T$63 .dwtag DW_TAG_union_type + 1865 .dwattr $C$DW$T$63, DW_AT_byte_size(0x04) + 1866$C$DW$151 .dwtag DW_TAG_member + 1867 .dwattr $C$DW$151, DW_AT_type(*$C$DW$T$47) + 1868 .dwattr $C$DW$151, DW_AT_name("IESP") + 1869 .dwattr $C$DW$151, DW_AT_TI_symbol_name("IESP") + 1870 .dwattr $C$DW$151, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 35 + + 1871 .dwattr $C$DW$151, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1872 .dwattr $C$DW$151, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1873 .dwattr $C$DW$151, DW_AT_decl_line(0xa1) + 1874 .dwattr $C$DW$151, DW_AT_decl_column(0x15) + 1875$C$DW$152 .dwtag DW_TAG_member + 1876 .dwattr $C$DW$152, DW_AT_type(*$C$DW$T$62) + 1877 .dwattr $C$DW$152, DW_AT_name("IESP_bit") + 1878 .dwattr $C$DW$152, DW_AT_TI_symbol_name("IESP_bit") + 1879 .dwattr $C$DW$152, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1880 .dwattr $C$DW$152, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1881 .dwattr $C$DW$152, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1882 .dwattr $C$DW$152, DW_AT_decl_line(0xaa) + 1883 .dwattr $C$DW$152, DW_AT_decl_column(0x05) + 1884 .dwendtag $C$DW$T$63 + 1885 + 1886 .dwattr $C$DW$T$63, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1887 .dwattr $C$DW$T$63, DW_AT_decl_line(0xa0) + 1888 .dwattr $C$DW$T$63, DW_AT_decl_column(0x08) + 1889 + 1890$C$DW$T$65 .dwtag DW_TAG_union_type + 1891 .dwattr $C$DW$T$65, DW_AT_byte_size(0x04) + 1892$C$DW$153 .dwtag DW_TAG_member + 1893 .dwattr $C$DW$153, DW_AT_type(*$C$DW$T$47) + 1894 .dwattr $C$DW$153, DW_AT_name("IECP") + 1895 .dwattr $C$DW$153, DW_AT_TI_symbol_name("IECP") + 1896 .dwattr $C$DW$153, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1897 .dwattr $C$DW$153, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1898 .dwattr $C$DW$153, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1899 .dwattr $C$DW$153, DW_AT_decl_line(0xb0) + 1900 .dwattr $C$DW$153, DW_AT_decl_column(0x15) + 1901$C$DW$154 .dwtag DW_TAG_member + 1902 .dwattr $C$DW$154, DW_AT_type(*$C$DW$T$64) + 1903 .dwattr $C$DW$154, DW_AT_name("IECP_bit") + 1904 .dwattr $C$DW$154, DW_AT_TI_symbol_name("IECP_bit") + 1905 .dwattr $C$DW$154, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1906 .dwattr $C$DW$154, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1907 .dwattr $C$DW$154, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1908 .dwattr $C$DW$154, DW_AT_decl_line(0xb8) + 1909 .dwattr $C$DW$154, DW_AT_decl_column(0x05) + 1910 .dwendtag $C$DW$T$65 + 1911 + 1912 .dwattr $C$DW$T$65, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1913 .dwattr $C$DW$T$65, DW_AT_decl_line(0xaf) + 1914 .dwattr $C$DW$T$65, DW_AT_decl_column(0x08) + 1915 + 1916$C$DW$T$67 .dwtag DW_TAG_union_type + 1917 .dwattr $C$DW$T$67, DW_AT_byte_size(0x04) + 1918$C$DW$155 .dwtag DW_TAG_member + 1919 .dwattr $C$DW$155, DW_AT_type(*$C$DW$T$47) + 1920 .dwattr $C$DW$155, DW_AT_name("PMAO") + 1921 .dwattr $C$DW$155, DW_AT_TI_symbol_name("PMAO") + 1922 .dwattr $C$DW$155, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1923 .dwattr $C$DW$155, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1924 .dwattr $C$DW$155, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1925 .dwattr $C$DW$155, DW_AT_decl_line(0xc1) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 36 + + 1926 .dwattr $C$DW$155, DW_AT_decl_column(0x15) + 1927$C$DW$156 .dwtag DW_TAG_member + 1928 .dwattr $C$DW$156, DW_AT_type(*$C$DW$T$66) + 1929 .dwattr $C$DW$156, DW_AT_name("PMAO_bit") + 1930 .dwattr $C$DW$156, DW_AT_TI_symbol_name("PMAO_bit") + 1931 .dwattr $C$DW$156, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1932 .dwattr $C$DW$156, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1933 .dwattr $C$DW$156, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1934 .dwattr $C$DW$156, DW_AT_decl_line(0xc7) + 1935 .dwattr $C$DW$156, DW_AT_decl_column(0x05) + 1936 .dwendtag $C$DW$T$67 + 1937 + 1938 .dwattr $C$DW$T$67, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1939 .dwattr $C$DW$T$67, DW_AT_decl_line(0xc0) + 1940 .dwattr $C$DW$T$67, DW_AT_decl_column(0x08) + 1941 + 1942$C$DW$T$69 .dwtag DW_TAG_union_type + 1943 .dwattr $C$DW$T$69, DW_AT_byte_size(0x04) + 1944$C$DW$157 .dwtag DW_TAG_member + 1945 .dwattr $C$DW$157, DW_AT_type(*$C$DW$T$47) + 1946 .dwattr $C$DW$157, DW_AT_name("IEPCLK") + 1947 .dwattr $C$DW$157, DW_AT_TI_symbol_name("IEPCLK") + 1948 .dwattr $C$DW$157, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1949 .dwattr $C$DW$157, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1950 .dwattr $C$DW$157, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1951 .dwattr $C$DW$157, DW_AT_decl_line(0xd0) + 1952 .dwattr $C$DW$157, DW_AT_decl_column(0x15) + 1953$C$DW$158 .dwtag DW_TAG_member + 1954 .dwattr $C$DW$158, DW_AT_type(*$C$DW$T$68) + 1955 .dwattr $C$DW$158, DW_AT_name("IEPCLK_bit") + 1956 .dwattr $C$DW$158, DW_AT_TI_symbol_name("IEPCLK_bit") + 1957 .dwattr $C$DW$158, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1958 .dwattr $C$DW$158, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1959 .dwattr $C$DW$158, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1960 .dwattr $C$DW$158, DW_AT_decl_line(0xd5) + 1961 .dwattr $C$DW$158, DW_AT_decl_column(0x05) + 1962 .dwendtag $C$DW$T$69 + 1963 + 1964 .dwattr $C$DW$T$69, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1965 .dwattr $C$DW$T$69, DW_AT_decl_line(0xcf) + 1966 .dwattr $C$DW$T$69, DW_AT_decl_column(0x08) + 1967 + 1968$C$DW$T$71 .dwtag DW_TAG_union_type + 1969 .dwattr $C$DW$T$71, DW_AT_byte_size(0x04) + 1970$C$DW$159 .dwtag DW_TAG_member + 1971 .dwattr $C$DW$159, DW_AT_type(*$C$DW$T$47) + 1972 .dwattr $C$DW$159, DW_AT_name("SPP") + 1973 .dwattr $C$DW$159, DW_AT_TI_symbol_name("SPP") + 1974 .dwattr $C$DW$159, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1975 .dwattr $C$DW$159, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1976 .dwattr $C$DW$159, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1977 .dwattr $C$DW$159, DW_AT_decl_line(0xdb) + 1978 .dwattr $C$DW$159, DW_AT_decl_column(0x15) + 1979$C$DW$160 .dwtag DW_TAG_member + 1980 .dwattr $C$DW$160, DW_AT_type(*$C$DW$T$70) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 37 + + 1981 .dwattr $C$DW$160, DW_AT_name("SPP_bit") + 1982 .dwattr $C$DW$160, DW_AT_TI_symbol_name("SPP_bit") + 1983 .dwattr $C$DW$160, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 1984 .dwattr $C$DW$160, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 1985 .dwattr $C$DW$160, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 1986 .dwattr $C$DW$160, DW_AT_decl_line(0xe1) + 1987 .dwattr $C$DW$160, DW_AT_decl_column(0x05) + 1988 .dwendtag $C$DW$T$71 + 1989 + 1990 .dwattr $C$DW$T$71, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 1991 .dwattr $C$DW$T$71, DW_AT_decl_line(0xda) + 1992 .dwattr $C$DW$T$71, DW_AT_decl_column(0x08) + 1993 + 1994$C$DW$T$73 .dwtag DW_TAG_union_type + 1995 .dwattr $C$DW$T$73, DW_AT_byte_size(0x04) + 1996$C$DW$161 .dwtag DW_TAG_member + 1997 .dwattr $C$DW$161, DW_AT_type(*$C$DW$T$47) + 1998 .dwattr $C$DW$161, DW_AT_name("PIN_MX") + 1999 .dwattr $C$DW$161, DW_AT_TI_symbol_name("PIN_MX") + 2000 .dwattr $C$DW$161, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2001 .dwattr $C$DW$161, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2002 .dwattr $C$DW$161, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2003 .dwattr $C$DW$161, DW_AT_decl_line(0xe9) + 2004 .dwattr $C$DW$161, DW_AT_decl_column(0x15) + 2005$C$DW$162 .dwtag DW_TAG_member + 2006 .dwattr $C$DW$162, DW_AT_type(*$C$DW$T$72) + 2007 .dwattr $C$DW$162, DW_AT_name("PIN_MX_bit") + 2008 .dwattr $C$DW$162, DW_AT_TI_symbol_name("PIN_MX_bit") + 2009 .dwattr $C$DW$162, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2010 .dwattr $C$DW$162, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2011 .dwattr $C$DW$162, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2012 .dwattr $C$DW$162, DW_AT_decl_line(0xee) + 2013 .dwattr $C$DW$162, DW_AT_decl_column(0x05) + 2014 .dwendtag $C$DW$T$73 + 2015 + 2016 .dwattr $C$DW$T$73, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2017 .dwattr $C$DW$T$73, DW_AT_decl_line(0xe8) + 2018 .dwattr $C$DW$T$73, DW_AT_decl_column(0x08) + 2019 + 2020$C$DW$T$75 .dwtag DW_TAG_union_type + 2021 .dwattr $C$DW$T$75, DW_AT_byte_size(0x04) + 2022$C$DW$163 .dwtag DW_TAG_member + 2023 .dwattr $C$DW$163, DW_AT_type(*$C$DW$T$47) + 2024 .dwattr $C$DW$163, DW_AT_name("CTRL") + 2025 .dwattr $C$DW$163, DW_AT_TI_symbol_name("CTRL") + 2026 .dwattr $C$DW$163, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2027 .dwattr $C$DW$163, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2028 .dwattr $C$DW$163, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2029 .dwattr $C$DW$163, DW_AT_decl_line(0x2a) + 2030 .dwattr $C$DW$163, DW_AT_decl_column(0x15) + 2031$C$DW$164 .dwtag DW_TAG_member + 2032 .dwattr $C$DW$164, DW_AT_type(*$C$DW$T$74) + 2033 .dwattr $C$DW$164, DW_AT_name("CTRL_bit") + 2034 .dwattr $C$DW$164, DW_AT_TI_symbol_name("CTRL_bit") + 2035 .dwattr $C$DW$164, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 38 + + 2036 .dwattr $C$DW$164, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2037 .dwattr $C$DW$164, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2038 .dwattr $C$DW$164, DW_AT_decl_line(0x36) + 2039 .dwattr $C$DW$164, DW_AT_decl_column(0x05) + 2040 .dwendtag $C$DW$T$75 + 2041 + 2042 .dwattr $C$DW$T$75, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2043 .dwattr $C$DW$T$75, DW_AT_decl_line(0x29) + 2044 .dwattr $C$DW$T$75, DW_AT_decl_column(0x08) + 2045 + 2046$C$DW$T$77 .dwtag DW_TAG_union_type + 2047 .dwattr $C$DW$T$77, DW_AT_byte_size(0x04) + 2048$C$DW$165 .dwtag DW_TAG_member + 2049 .dwattr $C$DW$165, DW_AT_type(*$C$DW$T$47) + 2050 .dwattr $C$DW$165, DW_AT_name("STS") + 2051 .dwattr $C$DW$165, DW_AT_TI_symbol_name("STS") + 2052 .dwattr $C$DW$165, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2053 .dwattr $C$DW$165, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2054 .dwattr $C$DW$165, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2055 .dwattr $C$DW$165, DW_AT_decl_line(0x3c) + 2056 .dwattr $C$DW$165, DW_AT_decl_column(0x15) + 2057$C$DW$166 .dwtag DW_TAG_member + 2058 .dwattr $C$DW$166, DW_AT_type(*$C$DW$T$76) + 2059 .dwattr $C$DW$166, DW_AT_name("STS_bit") + 2060 .dwattr $C$DW$166, DW_AT_TI_symbol_name("STS_bit") + 2061 .dwattr $C$DW$166, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2062 .dwattr $C$DW$166, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2063 .dwattr $C$DW$166, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2064 .dwattr $C$DW$166, DW_AT_decl_line(0x41) + 2065 .dwattr $C$DW$166, DW_AT_decl_column(0x05) + 2066 .dwendtag $C$DW$T$77 + 2067 + 2068 .dwattr $C$DW$T$77, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2069 .dwattr $C$DW$T$77, DW_AT_decl_line(0x3b) + 2070 .dwattr $C$DW$T$77, DW_AT_decl_column(0x08) + 2071 + 2072$C$DW$T$79 .dwtag DW_TAG_union_type + 2073 .dwattr $C$DW$T$79, DW_AT_byte_size(0x04) + 2074$C$DW$167 .dwtag DW_TAG_member + 2075 .dwattr $C$DW$167, DW_AT_type(*$C$DW$T$47) + 2076 .dwattr $C$DW$167, DW_AT_name("WAKEUP_EN") + 2077 .dwattr $C$DW$167, DW_AT_TI_symbol_name("WAKEUP_EN") + 2078 .dwattr $C$DW$167, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2079 .dwattr $C$DW$167, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2080 .dwattr $C$DW$167, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2081 .dwattr $C$DW$167, DW_AT_decl_line(0x47) + 2082 .dwattr $C$DW$167, DW_AT_decl_column(0x15) + 2083$C$DW$168 .dwtag DW_TAG_member + 2084 .dwattr $C$DW$168, DW_AT_type(*$C$DW$T$78) + 2085 .dwattr $C$DW$168, DW_AT_name("WAKEUP_EN_bit") + 2086 .dwattr $C$DW$168, DW_AT_TI_symbol_name("WAKEUP_EN_bit") + 2087 .dwattr $C$DW$168, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2088 .dwattr $C$DW$168, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2089 .dwattr $C$DW$168, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2090 .dwattr $C$DW$168, DW_AT_decl_line(0x4b) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 39 + + 2091 .dwattr $C$DW$168, DW_AT_decl_column(0x05) + 2092 .dwendtag $C$DW$T$79 + 2093 + 2094 .dwattr $C$DW$T$79, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2095 .dwattr $C$DW$T$79, DW_AT_decl_line(0x46) + 2096 .dwattr $C$DW$T$79, DW_AT_decl_column(0x08) + 2097 + 2098$C$DW$T$81 .dwtag DW_TAG_union_type + 2099 .dwattr $C$DW$T$81, DW_AT_byte_size(0x04) + 2100$C$DW$169 .dwtag DW_TAG_member + 2101 .dwattr $C$DW$169, DW_AT_type(*$C$DW$T$47) + 2102 .dwattr $C$DW$169, DW_AT_name("CYCLE") + 2103 .dwattr $C$DW$169, DW_AT_TI_symbol_name("CYCLE") + 2104 .dwattr $C$DW$169, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2105 .dwattr $C$DW$169, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2106 .dwattr $C$DW$169, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2107 .dwattr $C$DW$169, DW_AT_decl_line(0x51) + 2108 .dwattr $C$DW$169, DW_AT_decl_column(0x15) + 2109$C$DW$170 .dwtag DW_TAG_member + 2110 .dwattr $C$DW$170, DW_AT_type(*$C$DW$T$80) + 2111 .dwattr $C$DW$170, DW_AT_name("CYCLE_bit") + 2112 .dwattr $C$DW$170, DW_AT_TI_symbol_name("CYCLE_bit") + 2113 .dwattr $C$DW$170, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2114 .dwattr $C$DW$170, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2115 .dwattr $C$DW$170, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2116 .dwattr $C$DW$170, DW_AT_decl_line(0x55) + 2117 .dwattr $C$DW$170, DW_AT_decl_column(0x05) + 2118 .dwendtag $C$DW$T$81 + 2119 + 2120 .dwattr $C$DW$T$81, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2121 .dwattr $C$DW$T$81, DW_AT_decl_line(0x50) + 2122 .dwattr $C$DW$T$81, DW_AT_decl_column(0x08) + 2123 + 2124$C$DW$T$83 .dwtag DW_TAG_union_type + 2125 .dwattr $C$DW$T$83, DW_AT_byte_size(0x04) + 2126$C$DW$171 .dwtag DW_TAG_member + 2127 .dwattr $C$DW$171, DW_AT_type(*$C$DW$T$47) + 2128 .dwattr $C$DW$171, DW_AT_name("STALL") + 2129 .dwattr $C$DW$171, DW_AT_TI_symbol_name("STALL") + 2130 .dwattr $C$DW$171, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2131 .dwattr $C$DW$171, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2132 .dwattr $C$DW$171, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2133 .dwattr $C$DW$171, DW_AT_decl_line(0x5b) + 2134 .dwattr $C$DW$171, DW_AT_decl_column(0x15) + 2135$C$DW$172 .dwtag DW_TAG_member + 2136 .dwattr $C$DW$172, DW_AT_type(*$C$DW$T$82) + 2137 .dwattr $C$DW$172, DW_AT_name("STALL_bit") + 2138 .dwattr $C$DW$172, DW_AT_TI_symbol_name("STALL_bit") + 2139 .dwattr $C$DW$172, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2140 .dwattr $C$DW$172, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2141 .dwattr $C$DW$172, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2142 .dwattr $C$DW$172, DW_AT_decl_line(0x5f) + 2143 .dwattr $C$DW$172, DW_AT_decl_column(0x05) + 2144 .dwendtag $C$DW$T$83 + 2145 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 40 + + 2146 .dwattr $C$DW$T$83, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2147 .dwattr $C$DW$T$83, DW_AT_decl_line(0x5a) + 2148 .dwattr $C$DW$T$83, DW_AT_decl_column(0x08) + 2149 + 2150$C$DW$T$85 .dwtag DW_TAG_union_type + 2151 .dwattr $C$DW$T$85, DW_AT_byte_size(0x04) + 2152$C$DW$173 .dwtag DW_TAG_member + 2153 .dwattr $C$DW$173, DW_AT_type(*$C$DW$T$47) + 2154 .dwattr $C$DW$173, DW_AT_name("CTBIR0") + 2155 .dwattr $C$DW$173, DW_AT_TI_symbol_name("CTBIR0") + 2156 .dwattr $C$DW$173, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2157 .dwattr $C$DW$173, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2158 .dwattr $C$DW$173, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2159 .dwattr $C$DW$173, DW_AT_decl_line(0x68) + 2160 .dwattr $C$DW$173, DW_AT_decl_column(0x15) + 2161$C$DW$174 .dwtag DW_TAG_member + 2162 .dwattr $C$DW$174, DW_AT_type(*$C$DW$T$84) + 2163 .dwattr $C$DW$174, DW_AT_name("CTBIR0_bit") + 2164 .dwattr $C$DW$174, DW_AT_TI_symbol_name("CTBIR0_bit") + 2165 .dwattr $C$DW$174, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2166 .dwattr $C$DW$174, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2167 .dwattr $C$DW$174, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2168 .dwattr $C$DW$174, DW_AT_decl_line(0x6f) + 2169 .dwattr $C$DW$174, DW_AT_decl_column(0x05) + 2170 .dwendtag $C$DW$T$85 + 2171 + 2172 .dwattr $C$DW$T$85, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2173 .dwattr $C$DW$T$85, DW_AT_decl_line(0x67) + 2174 .dwattr $C$DW$T$85, DW_AT_decl_column(0x08) + 2175 + 2176$C$DW$T$87 .dwtag DW_TAG_union_type + 2177 .dwattr $C$DW$T$87, DW_AT_byte_size(0x04) + 2178$C$DW$175 .dwtag DW_TAG_member + 2179 .dwattr $C$DW$175, DW_AT_type(*$C$DW$T$47) + 2180 .dwattr $C$DW$175, DW_AT_name("CTBIR1") + 2181 .dwattr $C$DW$175, DW_AT_TI_symbol_name("CTBIR1") + 2182 .dwattr $C$DW$175, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2183 .dwattr $C$DW$175, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2184 .dwattr $C$DW$175, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2185 .dwattr $C$DW$175, DW_AT_decl_line(0x75) + 2186 .dwattr $C$DW$175, DW_AT_decl_column(0x15) + 2187$C$DW$176 .dwtag DW_TAG_member + 2188 .dwattr $C$DW$176, DW_AT_type(*$C$DW$T$86) + 2189 .dwattr $C$DW$176, DW_AT_name("CTBIR1_bit") + 2190 .dwattr $C$DW$176, DW_AT_TI_symbol_name("CTBIR1_bit") + 2191 .dwattr $C$DW$176, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2192 .dwattr $C$DW$176, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2193 .dwattr $C$DW$176, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2194 .dwattr $C$DW$176, DW_AT_decl_line(0x7c) + 2195 .dwattr $C$DW$176, DW_AT_decl_column(0x05) + 2196 .dwendtag $C$DW$T$87 + 2197 + 2198 .dwattr $C$DW$T$87, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2199 .dwattr $C$DW$T$87, DW_AT_decl_line(0x74) + 2200 .dwattr $C$DW$T$87, DW_AT_decl_column(0x08) +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 41 + + 2201 + 2202$C$DW$T$89 .dwtag DW_TAG_union_type + 2203 .dwattr $C$DW$T$89, DW_AT_byte_size(0x04) + 2204$C$DW$177 .dwtag DW_TAG_member + 2205 .dwattr $C$DW$177, DW_AT_type(*$C$DW$T$47) + 2206 .dwattr $C$DW$177, DW_AT_name("CTPPR0") + 2207 .dwattr $C$DW$177, DW_AT_TI_symbol_name("CTPPR0") + 2208 .dwattr $C$DW$177, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2209 .dwattr $C$DW$177, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2210 .dwattr $C$DW$177, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2211 .dwattr $C$DW$177, DW_AT_decl_line(0x82) + 2212 .dwattr $C$DW$177, DW_AT_decl_column(0x15) + 2213$C$DW$178 .dwtag DW_TAG_member + 2214 .dwattr $C$DW$178, DW_AT_type(*$C$DW$T$88) + 2215 .dwattr $C$DW$178, DW_AT_name("CTPPR0_bit") + 2216 .dwattr $C$DW$178, DW_AT_TI_symbol_name("CTPPR0_bit") + 2217 .dwattr $C$DW$178, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2218 .dwattr $C$DW$178, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2219 .dwattr $C$DW$178, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2220 .dwattr $C$DW$178, DW_AT_decl_line(0x87) + 2221 .dwattr $C$DW$178, DW_AT_decl_column(0x05) + 2222 .dwendtag $C$DW$T$89 + 2223 + 2224 .dwattr $C$DW$T$89, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2225 .dwattr $C$DW$T$89, DW_AT_decl_line(0x81) + 2226 .dwattr $C$DW$T$89, DW_AT_decl_column(0x08) + 2227 + 2228$C$DW$T$91 .dwtag DW_TAG_union_type + 2229 .dwattr $C$DW$T$91, DW_AT_byte_size(0x04) + 2230$C$DW$179 .dwtag DW_TAG_member + 2231 .dwattr $C$DW$179, DW_AT_type(*$C$DW$T$47) + 2232 .dwattr $C$DW$179, DW_AT_name("CTPPR1") + 2233 .dwattr $C$DW$179, DW_AT_TI_symbol_name("CTPPR1") + 2234 .dwattr $C$DW$179, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2235 .dwattr $C$DW$179, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2236 .dwattr $C$DW$179, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2237 .dwattr $C$DW$179, DW_AT_decl_line(0x8d) + 2238 .dwattr $C$DW$179, DW_AT_decl_column(0x15) + 2239$C$DW$180 .dwtag DW_TAG_member + 2240 .dwattr $C$DW$180, DW_AT_type(*$C$DW$T$90) + 2241 .dwattr $C$DW$180, DW_AT_name("CTPPR1_bit") + 2242 .dwattr $C$DW$180, DW_AT_TI_symbol_name("CTPPR1_bit") + 2243 .dwattr $C$DW$180, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2244 .dwattr $C$DW$180, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2245 .dwattr $C$DW$180, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/pr + 2246 .dwattr $C$DW$180, DW_AT_decl_line(0x92) + 2247 .dwattr $C$DW$180, DW_AT_decl_column(0x05) + 2248 .dwendtag $C$DW$T$91 + 2249 + 2250 .dwattr $C$DW$T$91, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/am335x/p + 2251 .dwattr $C$DW$T$91, DW_AT_decl_line(0x8c) + 2252 .dwattr $C$DW$T$91, DW_AT_decl_column(0x08) + 2253$C$DW$T$2 .dwtag DW_TAG_unspecified_type + 2254 .dwattr $C$DW$T$2, DW_AT_name("void") + 2255$C$DW$T$4 .dwtag DW_TAG_base_type +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 42 + + 2256 .dwattr $C$DW$T$4, DW_AT_encoding(DW_ATE_boolean) + 2257 .dwattr $C$DW$T$4, DW_AT_name("bool") + 2258 .dwattr $C$DW$T$4, DW_AT_byte_size(0x01) + 2259$C$DW$T$5 .dwtag DW_TAG_base_type + 2260 .dwattr $C$DW$T$5, DW_AT_encoding(DW_ATE_signed_char) + 2261 .dwattr $C$DW$T$5, DW_AT_name("signed char") + 2262 .dwattr $C$DW$T$5, DW_AT_byte_size(0x01) + 2263$C$DW$T$6 .dwtag DW_TAG_base_type + 2264 .dwattr $C$DW$T$6, DW_AT_encoding(DW_ATE_unsigned_char) + 2265 .dwattr $C$DW$T$6, DW_AT_name("unsigned char") + 2266 .dwattr $C$DW$T$6, DW_AT_byte_size(0x01) + 2267$C$DW$T$7 .dwtag DW_TAG_base_type + 2268 .dwattr $C$DW$T$7, DW_AT_encoding(DW_ATE_signed_char) + 2269 .dwattr $C$DW$T$7, DW_AT_name("wchar_t") + 2270 .dwattr $C$DW$T$7, DW_AT_byte_size(0x04) + 2271$C$DW$T$8 .dwtag DW_TAG_base_type + 2272 .dwattr $C$DW$T$8, DW_AT_encoding(DW_ATE_signed) + 2273 .dwattr $C$DW$T$8, DW_AT_name("short") + 2274 .dwattr $C$DW$T$8, DW_AT_byte_size(0x02) + 2275$C$DW$T$9 .dwtag DW_TAG_base_type + 2276 .dwattr $C$DW$T$9, DW_AT_encoding(DW_ATE_unsigned) + 2277 .dwattr $C$DW$T$9, DW_AT_name("unsigned short") + 2278 .dwattr $C$DW$T$9, DW_AT_byte_size(0x02) + 2279$C$DW$T$10 .dwtag DW_TAG_base_type + 2280 .dwattr $C$DW$T$10, DW_AT_encoding(DW_ATE_signed) + 2281 .dwattr $C$DW$T$10, DW_AT_name("int") + 2282 .dwattr $C$DW$T$10, DW_AT_byte_size(0x04) + 2283$C$DW$T$11 .dwtag DW_TAG_base_type + 2284 .dwattr $C$DW$T$11, DW_AT_encoding(DW_ATE_unsigned) + 2285 .dwattr $C$DW$T$11, DW_AT_name("unsigned int") + 2286 .dwattr $C$DW$T$11, DW_AT_byte_size(0x04) + 2287$C$DW$T$32 .dwtag DW_TAG_typedef, DW_AT_name("uint32_t") + 2288 .dwattr $C$DW$T$32, DW_AT_type(*$C$DW$T$11) + 2289 .dwattr $C$DW$T$32, DW_AT_language(DW_LANG_C) + 2290 .dwattr $C$DW$T$32, DW_AT_decl_file("/usr/share/ti/cgt-pru/include/stdint.h") + 2291 .dwattr $C$DW$T$32, DW_AT_decl_line(0x2f) + 2292 .dwattr $C$DW$T$32, DW_AT_decl_column(0x1c) + 2293 + 2294$C$DW$T$33 .dwtag DW_TAG_array_type + 2295 .dwattr $C$DW$T$33, DW_AT_type(*$C$DW$T$32) + 2296 .dwattr $C$DW$T$33, DW_AT_language(DW_LANG_C) + 2297 .dwattr $C$DW$T$33, DW_AT_byte_size(0x04) + 2298$C$DW$181 .dwtag DW_TAG_subrange_type + 2299 .dwattr $C$DW$181, DW_AT_upper_bound(0x00) + 2300 .dwendtag $C$DW$T$33 + 2301 + 2302 + 2303$C$DW$T$34 .dwtag DW_TAG_array_type + 2304 .dwattr $C$DW$T$34, DW_AT_type(*$C$DW$T$32) + 2305 .dwattr $C$DW$T$34, DW_AT_language(DW_LANG_C) + 2306 .dwattr $C$DW$T$34, DW_AT_byte_size(0x08) + 2307$C$DW$182 .dwtag DW_TAG_subrange_type + 2308 .dwattr $C$DW$182, DW_AT_upper_bound(0x01) + 2309 .dwendtag $C$DW$T$34 + 2310 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 43 + + 2311 + 2312$C$DW$T$45 .dwtag DW_TAG_array_type + 2313 .dwattr $C$DW$T$45, DW_AT_type(*$C$DW$T$32) + 2314 .dwattr $C$DW$T$45, DW_AT_language(DW_LANG_C) + 2315 .dwattr $C$DW$T$45, DW_AT_byte_size(0x0c) + 2316$C$DW$183 .dwtag DW_TAG_subrange_type + 2317 .dwattr $C$DW$183, DW_AT_upper_bound(0x02) + 2318 .dwendtag $C$DW$T$45 + 2319 + 2320$C$DW$T$47 .dwtag DW_TAG_volatile_type + 2321 .dwattr $C$DW$T$47, DW_AT_type(*$C$DW$T$32) + 2322$C$DW$T$12 .dwtag DW_TAG_base_type + 2323 .dwattr $C$DW$T$12, DW_AT_encoding(DW_ATE_signed) + 2324 .dwattr $C$DW$T$12, DW_AT_name("long") + 2325 .dwattr $C$DW$T$12, DW_AT_byte_size(0x04) + 2326$C$DW$T$13 .dwtag DW_TAG_base_type + 2327 .dwattr $C$DW$T$13, DW_AT_encoding(DW_ATE_unsigned) + 2328 .dwattr $C$DW$T$13, DW_AT_name("unsigned long") + 2329 .dwattr $C$DW$T$13, DW_AT_byte_size(0x04) + 2330$C$DW$T$14 .dwtag DW_TAG_base_type + 2331 .dwattr $C$DW$T$14, DW_AT_encoding(DW_ATE_signed) + 2332 .dwattr $C$DW$T$14, DW_AT_name("long long") + 2333 .dwattr $C$DW$T$14, DW_AT_byte_size(0x08) + 2334$C$DW$T$15 .dwtag DW_TAG_base_type + 2335 .dwattr $C$DW$T$15, DW_AT_encoding(DW_ATE_unsigned) + 2336 .dwattr $C$DW$T$15, DW_AT_name("unsigned long long") + 2337 .dwattr $C$DW$T$15, DW_AT_byte_size(0x08) + 2338$C$DW$T$16 .dwtag DW_TAG_base_type + 2339 .dwattr $C$DW$T$16, DW_AT_encoding(DW_ATE_float) + 2340 .dwattr $C$DW$T$16, DW_AT_name("float") + 2341 .dwattr $C$DW$T$16, DW_AT_byte_size(0x04) + 2342$C$DW$T$17 .dwtag DW_TAG_base_type + 2343 .dwattr $C$DW$T$17, DW_AT_encoding(DW_ATE_float) + 2344 .dwattr $C$DW$T$17, DW_AT_name("double") + 2345 .dwattr $C$DW$T$17, DW_AT_byte_size(0x08) + 2346$C$DW$T$18 .dwtag DW_TAG_base_type + 2347 .dwattr $C$DW$T$18, DW_AT_encoding(DW_ATE_float) + 2348 .dwattr $C$DW$T$18, DW_AT_name("long double") + 2349 .dwattr $C$DW$T$18, DW_AT_byte_size(0x08) + 2350 + 2351$C$DW$T$92 .dwtag DW_TAG_structure_type + 2352 .dwattr $C$DW$T$92, DW_AT_name("my_resource_table") + 2353 .dwattr $C$DW$T$92, DW_AT_byte_size(0x14) + 2354$C$DW$184 .dwtag DW_TAG_member + 2355 .dwattr $C$DW$184, DW_AT_type(*$C$DW$T$93) + 2356 .dwattr $C$DW$184, DW_AT_name("base") + 2357 .dwattr $C$DW$184, DW_AT_TI_symbol_name("base") + 2358 .dwattr $C$DW$184, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2359 .dwattr $C$DW$184, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2360 .dwattr $C$DW$184, DW_AT_decl_file("/var/lib/cloud9/common/resource_table_empty.h") + 2361 .dwattr $C$DW$184, DW_AT_decl_line(0x18) + 2362 .dwattr $C$DW$184, DW_AT_decl_column(0x18) + 2363$C$DW$185 .dwtag DW_TAG_member + 2364 .dwattr $C$DW$185, DW_AT_type(*$C$DW$T$33) + 2365 .dwattr $C$DW$185, DW_AT_name("offset") +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 44 + + 2366 .dwattr $C$DW$185, DW_AT_TI_symbol_name("offset") + 2367 .dwattr $C$DW$185, DW_AT_data_member_location[DW_OP_plus_uconst 0x10] + 2368 .dwattr $C$DW$185, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2369 .dwattr $C$DW$185, DW_AT_decl_file("/var/lib/cloud9/common/resource_table_empty.h") + 2370 .dwattr $C$DW$185, DW_AT_decl_line(0x1a) + 2371 .dwattr $C$DW$185, DW_AT_decl_column(0x0b) + 2372 .dwendtag $C$DW$T$92 + 2373 + 2374 .dwattr $C$DW$T$92, DW_AT_decl_file("/var/lib/cloud9/common/resource_table_empty.h") + 2375 .dwattr $C$DW$T$92, DW_AT_decl_line(0x17) + 2376 .dwattr $C$DW$T$92, DW_AT_decl_column(0x08) + 2377 + 2378$C$DW$T$93 .dwtag DW_TAG_structure_type + 2379 .dwattr $C$DW$T$93, DW_AT_name("resource_table") + 2380 .dwattr $C$DW$T$93, DW_AT_byte_size(0x10) + 2381$C$DW$186 .dwtag DW_TAG_member + 2382 .dwattr $C$DW$186, DW_AT_type(*$C$DW$T$32) + 2383 .dwattr $C$DW$186, DW_AT_name("ver") + 2384 .dwattr $C$DW$186, DW_AT_TI_symbol_name("ver") + 2385 .dwattr $C$DW$186, DW_AT_data_member_location[DW_OP_plus_uconst 0x0] + 2386 .dwattr $C$DW$186, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2387 .dwattr $C$DW$186, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/rsc_types + 2388 .dwattr $C$DW$186, DW_AT_decl_line(0x59) + 2389 .dwattr $C$DW$186, DW_AT_decl_column(0x0b) + 2390$C$DW$187 .dwtag DW_TAG_member + 2391 .dwattr $C$DW$187, DW_AT_type(*$C$DW$T$32) + 2392 .dwattr $C$DW$187, DW_AT_name("num") + 2393 .dwattr $C$DW$187, DW_AT_TI_symbol_name("num") + 2394 .dwattr $C$DW$187, DW_AT_data_member_location[DW_OP_plus_uconst 0x4] + 2395 .dwattr $C$DW$187, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2396 .dwattr $C$DW$187, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/rsc_types + 2397 .dwattr $C$DW$187, DW_AT_decl_line(0x5a) + 2398 .dwattr $C$DW$187, DW_AT_decl_column(0x0b) + 2399$C$DW$188 .dwtag DW_TAG_member + 2400 .dwattr $C$DW$188, DW_AT_type(*$C$DW$T$34) + 2401 .dwattr $C$DW$188, DW_AT_name("reserved") + 2402 .dwattr $C$DW$188, DW_AT_TI_symbol_name("reserved") + 2403 .dwattr $C$DW$188, DW_AT_data_member_location[DW_OP_plus_uconst 0x8] + 2404 .dwattr $C$DW$188, DW_AT_accessibility(DW_ACCESS_ai_64_lic) + 2405 .dwattr $C$DW$188, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/rsc_types + 2406 .dwattr $C$DW$188, DW_AT_decl_line(0x5b) + 2407 .dwattr $C$DW$188, DW_AT_decl_column(0x0b) + 2408 .dwendtag $C$DW$T$93 + 2409 + 2410 .dwattr $C$DW$T$93, DW_AT_decl_file("/usr/lib/ti/pru-software-support-package/include/rsc_type + 2411 .dwattr $C$DW$T$93, DW_AT_decl_line(0x58) + 2412 .dwattr $C$DW$T$93, DW_AT_decl_column(0x08) + 2413 .dwattr $C$DW$CU, DW_AT_language(DW_LANG_C) + 2414 + 2415;*************************************************************** + 2416;* DWARF CIE ENTRIES * + 2417;*************************************************************** + 2418 + 2419$C$DW$CIE .dwcie 14 + 2420 .dwcfi cfa_register, 8 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 45 + + 2421 .dwcfi cfa_offset, 0 + 2422 .dwcfi undefined, 0 + 2423 .dwcfi undefined, 1 + 2424 .dwcfi undefined, 2 + 2425 .dwcfi undefined, 3 + 2426 .dwcfi undefined, 4 + 2427 .dwcfi undefined, 5 + 2428 .dwcfi undefined, 6 + 2429 .dwcfi undefined, 7 + 2430 .dwcfi same_value, 8 + 2431 .dwcfi same_value, 9 + 2432 .dwcfi same_value, 10 + 2433 .dwcfi same_value, 11 + 2434 .dwcfi undefined, 12 + 2435 .dwcfi undefined, 13 + 2436 .dwcfi undefined, 14 + 2437 .dwcfi undefined, 15 + 2438 .dwcfi same_value, 16 + 2439 .dwcfi same_value, 17 + 2440 .dwcfi same_value, 18 + 2441 .dwcfi same_value, 19 + 2442 .dwcfi same_value, 20 + 2443 .dwcfi same_value, 21 + 2444 .dwcfi same_value, 22 + 2445 .dwcfi same_value, 23 + 2446 .dwcfi same_value, 24 + 2447 .dwcfi same_value, 25 + 2448 .dwcfi same_value, 26 + 2449 .dwcfi same_value, 27 + 2450 .dwcfi same_value, 28 + 2451 .dwcfi same_value, 29 + 2452 .dwcfi same_value, 30 + 2453 .dwcfi same_value, 31 + 2454 .dwcfi same_value, 32 + 2455 .dwcfi same_value, 33 + 2456 .dwcfi same_value, 34 + 2457 .dwcfi same_value, 35 + 2458 .dwcfi same_value, 36 + 2459 .dwcfi same_value, 37 + 2460 .dwcfi same_value, 38 + 2461 .dwcfi same_value, 39 + 2462 .dwcfi same_value, 40 + 2463 .dwcfi same_value, 41 + 2464 .dwcfi same_value, 42 + 2465 .dwcfi same_value, 43 + 2466 .dwcfi same_value, 44 + 2467 .dwcfi same_value, 45 + 2468 .dwcfi same_value, 46 + 2469 .dwcfi same_value, 47 + 2470 .dwcfi same_value, 48 + 2471 .dwcfi same_value, 49 + 2472 .dwcfi same_value, 50 + 2473 .dwcfi same_value, 51 + 2474 .dwcfi same_value, 52 + 2475 .dwcfi same_value, 53 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 46 + + 2476 .dwcfi same_value, 54 + 2477 .dwcfi same_value, 55 + 2478 .dwcfi undefined, 56 + 2479 .dwcfi undefined, 57 + 2480 .dwcfi undefined, 58 + 2481 .dwcfi undefined, 59 + 2482 .dwcfi undefined, 60 + 2483 .dwcfi undefined, 61 + 2484 .dwcfi undefined, 62 + 2485 .dwcfi undefined, 63 + 2486 .dwcfi undefined, 64 + 2487 .dwcfi undefined, 65 + 2488 .dwcfi undefined, 66 + 2489 .dwcfi undefined, 67 + 2490 .dwcfi undefined, 68 + 2491 .dwcfi undefined, 69 + 2492 .dwcfi undefined, 70 + 2493 .dwcfi undefined, 71 + 2494 .dwcfi undefined, 72 + 2495 .dwcfi undefined, 73 + 2496 .dwcfi undefined, 74 + 2497 .dwcfi undefined, 75 + 2498 .dwcfi undefined, 76 + 2499 .dwcfi undefined, 77 + 2500 .dwcfi undefined, 78 + 2501 .dwcfi undefined, 79 + 2502 .dwcfi undefined, 80 + 2503 .dwcfi undefined, 81 + 2504 .dwcfi undefined, 82 + 2505 .dwcfi undefined, 83 + 2506 .dwcfi undefined, 84 + 2507 .dwcfi undefined, 85 + 2508 .dwcfi undefined, 86 + 2509 .dwcfi undefined, 87 + 2510 .dwcfi undefined, 88 + 2511 .dwcfi undefined, 89 + 2512 .dwcfi undefined, 90 + 2513 .dwcfi undefined, 91 + 2514 .dwcfi undefined, 92 + 2515 .dwcfi undefined, 93 + 2516 .dwcfi undefined, 94 + 2517 .dwcfi undefined, 95 + 2518 .dwcfi undefined, 96 + 2519 .dwcfi undefined, 97 + 2520 .dwcfi undefined, 98 + 2521 .dwcfi undefined, 99 + 2522 .dwcfi undefined, 100 + 2523 .dwcfi undefined, 101 + 2524 .dwcfi undefined, 102 + 2525 .dwcfi undefined, 103 + 2526 .dwcfi undefined, 104 + 2527 .dwcfi undefined, 105 + 2528 .dwcfi undefined, 106 + 2529 .dwcfi undefined, 107 + 2530 .dwcfi undefined, 108 +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 47 + + 2531 .dwcfi undefined, 109 + 2532 .dwcfi undefined, 110 + 2533 .dwcfi undefined, 111 + 2534 .dwcfi undefined, 112 + 2535 .dwcfi undefined, 113 + 2536 .dwcfi undefined, 114 + 2537 .dwcfi undefined, 115 + 2538 .dwcfi undefined, 116 + 2539 .dwcfi undefined, 117 + 2540 .dwcfi undefined, 118 + 2541 .dwcfi undefined, 119 + 2542 .dwcfi undefined, 120 + 2543 .dwcfi undefined, 121 + 2544 .dwcfi undefined, 122 + 2545 .dwcfi undefined, 123 + 2546 .dwcfi undefined, 124 + 2547 .dwcfi undefined, 125 + 2548 .dwcfi undefined, 126 + 2549 .dwcfi undefined, 127 + 2550 .dwcfi undefined, 128 + 2551 .dwcfi undefined, 129 + 2552 .dwcfi undefined, 130 + 2553 .dwcfi undefined, 131 + 2554 .dwcfi undefined, 132 + 2555 .dwcfi undefined, 133 + 2556 .dwcfi undefined, 134 + 2557 .dwendentry + 2558 + 2559;*************************************************************** + 2560;* DWARF REGISTER MAP * + 2561;*************************************************************** + 2562 + 2563$C$DW$189 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R0_b0") + 2564 .dwattr $C$DW$189, DW_AT_location[DW_OP_reg0] + 2565$C$DW$190 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R0_b1") + 2566 .dwattr $C$DW$190, DW_AT_location[DW_OP_reg1] + 2567$C$DW$191 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R0_b2") + 2568 .dwattr $C$DW$191, DW_AT_location[DW_OP_reg2] + 2569$C$DW$192 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R0_b3") + 2570 .dwattr $C$DW$192, DW_AT_location[DW_OP_reg3] + 2571$C$DW$193 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R1_b0") + 2572 .dwattr $C$DW$193, DW_AT_location[DW_OP_reg4] + 2573$C$DW$194 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R1_b1") + 2574 .dwattr $C$DW$194, DW_AT_location[DW_OP_reg5] + 2575$C$DW$195 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R1_b2") + 2576 .dwattr $C$DW$195, DW_AT_location[DW_OP_reg6] + 2577$C$DW$196 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R1_b3") + 2578 .dwattr $C$DW$196, DW_AT_location[DW_OP_reg7] + 2579$C$DW$197 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R2_b0") + 2580 .dwattr $C$DW$197, DW_AT_location[DW_OP_reg8] + 2581$C$DW$198 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R2_b1") + 2582 .dwattr $C$DW$198, DW_AT_location[DW_OP_reg9] + 2583$C$DW$199 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R2_b2") + 2584 .dwattr $C$DW$199, DW_AT_location[DW_OP_reg10] + 2585$C$DW$200 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R2_b3") +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 48 + + 2586 .dwattr $C$DW$200, DW_AT_location[DW_OP_reg11] + 2587$C$DW$201 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R3_b0") + 2588 .dwattr $C$DW$201, DW_AT_location[DW_OP_reg12] + 2589$C$DW$202 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R3_b1") + 2590 .dwattr $C$DW$202, DW_AT_location[DW_OP_reg13] + 2591$C$DW$203 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R3_b2") + 2592 .dwattr $C$DW$203, DW_AT_location[DW_OP_reg14] + 2593$C$DW$204 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R3_b3") + 2594 .dwattr $C$DW$204, DW_AT_location[DW_OP_reg15] + 2595$C$DW$205 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R4_b0") + 2596 .dwattr $C$DW$205, DW_AT_location[DW_OP_reg16] + 2597$C$DW$206 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R4_b1") + 2598 .dwattr $C$DW$206, DW_AT_location[DW_OP_reg17] + 2599$C$DW$207 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R4_b2") + 2600 .dwattr $C$DW$207, DW_AT_location[DW_OP_reg18] + 2601$C$DW$208 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R4_b3") + 2602 .dwattr $C$DW$208, DW_AT_location[DW_OP_reg19] + 2603$C$DW$209 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R5_b0") + 2604 .dwattr $C$DW$209, DW_AT_location[DW_OP_reg20] + 2605$C$DW$210 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R5_b1") + 2606 .dwattr $C$DW$210, DW_AT_location[DW_OP_reg21] + 2607$C$DW$211 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R5_b2") + 2608 .dwattr $C$DW$211, DW_AT_location[DW_OP_reg22] + 2609$C$DW$212 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R5_b3") + 2610 .dwattr $C$DW$212, DW_AT_location[DW_OP_reg23] + 2611$C$DW$213 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R6_b0") + 2612 .dwattr $C$DW$213, DW_AT_location[DW_OP_reg24] + 2613$C$DW$214 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R6_b1") + 2614 .dwattr $C$DW$214, DW_AT_location[DW_OP_reg25] + 2615$C$DW$215 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R6_b2") + 2616 .dwattr $C$DW$215, DW_AT_location[DW_OP_reg26] + 2617$C$DW$216 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R6_b3") + 2618 .dwattr $C$DW$216, DW_AT_location[DW_OP_reg27] + 2619$C$DW$217 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R7_b0") + 2620 .dwattr $C$DW$217, DW_AT_location[DW_OP_reg28] + 2621$C$DW$218 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R7_b1") + 2622 .dwattr $C$DW$218, DW_AT_location[DW_OP_reg29] + 2623$C$DW$219 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R7_b2") + 2624 .dwattr $C$DW$219, DW_AT_location[DW_OP_reg30] + 2625$C$DW$220 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R7_b3") + 2626 .dwattr $C$DW$220, DW_AT_location[DW_OP_reg31] + 2627$C$DW$221 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R8_b0") + 2628 .dwattr $C$DW$221, DW_AT_location[DW_OP_regx 0x20] + 2629$C$DW$222 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R8_b1") + 2630 .dwattr $C$DW$222, DW_AT_location[DW_OP_regx 0x21] + 2631$C$DW$223 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R8_b2") + 2632 .dwattr $C$DW$223, DW_AT_location[DW_OP_regx 0x22] + 2633$C$DW$224 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R8_b3") + 2634 .dwattr $C$DW$224, DW_AT_location[DW_OP_regx 0x23] + 2635$C$DW$225 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R9_b0") + 2636 .dwattr $C$DW$225, DW_AT_location[DW_OP_regx 0x24] + 2637$C$DW$226 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R9_b1") + 2638 .dwattr $C$DW$226, DW_AT_location[DW_OP_regx 0x25] + 2639$C$DW$227 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R9_b2") + 2640 .dwattr $C$DW$227, DW_AT_location[DW_OP_regx 0x26] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 49 + + 2641$C$DW$228 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R9_b3") + 2642 .dwattr $C$DW$228, DW_AT_location[DW_OP_regx 0x27] + 2643$C$DW$229 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R10_b0") + 2644 .dwattr $C$DW$229, DW_AT_location[DW_OP_regx 0x28] + 2645$C$DW$230 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R10_b1") + 2646 .dwattr $C$DW$230, DW_AT_location[DW_OP_regx 0x29] + 2647$C$DW$231 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R10_b2") + 2648 .dwattr $C$DW$231, DW_AT_location[DW_OP_regx 0x2a] + 2649$C$DW$232 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R10_b3") + 2650 .dwattr $C$DW$232, DW_AT_location[DW_OP_regx 0x2b] + 2651$C$DW$233 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R11_b0") + 2652 .dwattr $C$DW$233, DW_AT_location[DW_OP_regx 0x2c] + 2653$C$DW$234 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R11_b1") + 2654 .dwattr $C$DW$234, DW_AT_location[DW_OP_regx 0x2d] + 2655$C$DW$235 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R11_b2") + 2656 .dwattr $C$DW$235, DW_AT_location[DW_OP_regx 0x2e] + 2657$C$DW$236 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R11_b3") + 2658 .dwattr $C$DW$236, DW_AT_location[DW_OP_regx 0x2f] + 2659$C$DW$237 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R12_b0") + 2660 .dwattr $C$DW$237, DW_AT_location[DW_OP_regx 0x30] + 2661$C$DW$238 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R12_b1") + 2662 .dwattr $C$DW$238, DW_AT_location[DW_OP_regx 0x31] + 2663$C$DW$239 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R12_b2") + 2664 .dwattr $C$DW$239, DW_AT_location[DW_OP_regx 0x32] + 2665$C$DW$240 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R12_b3") + 2666 .dwattr $C$DW$240, DW_AT_location[DW_OP_regx 0x33] + 2667$C$DW$241 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R13_b0") + 2668 .dwattr $C$DW$241, DW_AT_location[DW_OP_regx 0x34] + 2669$C$DW$242 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R13_b1") + 2670 .dwattr $C$DW$242, DW_AT_location[DW_OP_regx 0x35] + 2671$C$DW$243 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R13_b2") + 2672 .dwattr $C$DW$243, DW_AT_location[DW_OP_regx 0x36] + 2673$C$DW$244 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R13_b3") + 2674 .dwattr $C$DW$244, DW_AT_location[DW_OP_regx 0x37] + 2675$C$DW$245 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R14_b0") + 2676 .dwattr $C$DW$245, DW_AT_location[DW_OP_regx 0x38] + 2677$C$DW$246 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R14_b1") + 2678 .dwattr $C$DW$246, DW_AT_location[DW_OP_regx 0x39] + 2679$C$DW$247 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R14_b2") + 2680 .dwattr $C$DW$247, DW_AT_location[DW_OP_regx 0x3a] + 2681$C$DW$248 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R14_b3") + 2682 .dwattr $C$DW$248, DW_AT_location[DW_OP_regx 0x3b] + 2683$C$DW$249 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R15_b0") + 2684 .dwattr $C$DW$249, DW_AT_location[DW_OP_regx 0x3c] + 2685$C$DW$250 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R15_b1") + 2686 .dwattr $C$DW$250, DW_AT_location[DW_OP_regx 0x3d] + 2687$C$DW$251 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R15_b2") + 2688 .dwattr $C$DW$251, DW_AT_location[DW_OP_regx 0x3e] + 2689$C$DW$252 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R15_b3") + 2690 .dwattr $C$DW$252, DW_AT_location[DW_OP_regx 0x3f] + 2691$C$DW$253 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R16_b0") + 2692 .dwattr $C$DW$253, DW_AT_location[DW_OP_regx 0x40] + 2693$C$DW$254 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R16_b1") + 2694 .dwattr $C$DW$254, DW_AT_location[DW_OP_regx 0x41] + 2695$C$DW$255 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R16_b2") +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 50 + + 2696 .dwattr $C$DW$255, DW_AT_location[DW_OP_regx 0x42] + 2697$C$DW$256 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R16_b3") + 2698 .dwattr $C$DW$256, DW_AT_location[DW_OP_regx 0x43] + 2699$C$DW$257 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R17_b0") + 2700 .dwattr $C$DW$257, DW_AT_location[DW_OP_regx 0x44] + 2701$C$DW$258 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R17_b1") + 2702 .dwattr $C$DW$258, DW_AT_location[DW_OP_regx 0x45] + 2703$C$DW$259 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R17_b2") + 2704 .dwattr $C$DW$259, DW_AT_location[DW_OP_regx 0x46] + 2705$C$DW$260 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R17_b3") + 2706 .dwattr $C$DW$260, DW_AT_location[DW_OP_regx 0x47] + 2707$C$DW$261 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R18_b0") + 2708 .dwattr $C$DW$261, DW_AT_location[DW_OP_regx 0x48] + 2709$C$DW$262 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R18_b1") + 2710 .dwattr $C$DW$262, DW_AT_location[DW_OP_regx 0x49] + 2711$C$DW$263 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R18_b2") + 2712 .dwattr $C$DW$263, DW_AT_location[DW_OP_regx 0x4a] + 2713$C$DW$264 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R18_b3") + 2714 .dwattr $C$DW$264, DW_AT_location[DW_OP_regx 0x4b] + 2715$C$DW$265 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R19_b0") + 2716 .dwattr $C$DW$265, DW_AT_location[DW_OP_regx 0x4c] + 2717$C$DW$266 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R19_b1") + 2718 .dwattr $C$DW$266, DW_AT_location[DW_OP_regx 0x4d] + 2719$C$DW$267 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R19_b2") + 2720 .dwattr $C$DW$267, DW_AT_location[DW_OP_regx 0x4e] + 2721$C$DW$268 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R19_b3") + 2722 .dwattr $C$DW$268, DW_AT_location[DW_OP_regx 0x4f] + 2723$C$DW$269 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R20_b0") + 2724 .dwattr $C$DW$269, DW_AT_location[DW_OP_regx 0x50] + 2725$C$DW$270 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R20_b1") + 2726 .dwattr $C$DW$270, DW_AT_location[DW_OP_regx 0x51] + 2727$C$DW$271 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R20_b2") + 2728 .dwattr $C$DW$271, DW_AT_location[DW_OP_regx 0x52] + 2729$C$DW$272 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R20_b3") + 2730 .dwattr $C$DW$272, DW_AT_location[DW_OP_regx 0x53] + 2731$C$DW$273 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R21_b0") + 2732 .dwattr $C$DW$273, DW_AT_location[DW_OP_regx 0x54] + 2733$C$DW$274 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R21_b1") + 2734 .dwattr $C$DW$274, DW_AT_location[DW_OP_regx 0x55] + 2735$C$DW$275 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R21_b2") + 2736 .dwattr $C$DW$275, DW_AT_location[DW_OP_regx 0x56] + 2737$C$DW$276 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R21_b3") + 2738 .dwattr $C$DW$276, DW_AT_location[DW_OP_regx 0x57] + 2739$C$DW$277 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R22_b0") + 2740 .dwattr $C$DW$277, DW_AT_location[DW_OP_regx 0x58] + 2741$C$DW$278 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R22_b1") + 2742 .dwattr $C$DW$278, DW_AT_location[DW_OP_regx 0x59] + 2743$C$DW$279 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R22_b2") + 2744 .dwattr $C$DW$279, DW_AT_location[DW_OP_regx 0x5a] + 2745$C$DW$280 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R22_b3") + 2746 .dwattr $C$DW$280, DW_AT_location[DW_OP_regx 0x5b] + 2747$C$DW$281 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R23_b0") + 2748 .dwattr $C$DW$281, DW_AT_location[DW_OP_regx 0x5c] + 2749$C$DW$282 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R23_b1") + 2750 .dwattr $C$DW$282, DW_AT_location[DW_OP_regx 0x5d] +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 51 + + 2751$C$DW$283 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R23_b2") + 2752 .dwattr $C$DW$283, DW_AT_location[DW_OP_regx 0x5e] + 2753$C$DW$284 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R23_b3") + 2754 .dwattr $C$DW$284, DW_AT_location[DW_OP_regx 0x5f] + 2755$C$DW$285 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R24_b0") + 2756 .dwattr $C$DW$285, DW_AT_location[DW_OP_regx 0x60] + 2757$C$DW$286 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R24_b1") + 2758 .dwattr $C$DW$286, DW_AT_location[DW_OP_regx 0x61] + 2759$C$DW$287 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R24_b2") + 2760 .dwattr $C$DW$287, DW_AT_location[DW_OP_regx 0x62] + 2761$C$DW$288 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R24_b3") + 2762 .dwattr $C$DW$288, DW_AT_location[DW_OP_regx 0x63] + 2763$C$DW$289 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R25_b0") + 2764 .dwattr $C$DW$289, DW_AT_location[DW_OP_regx 0x64] + 2765$C$DW$290 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R25_b1") + 2766 .dwattr $C$DW$290, DW_AT_location[DW_OP_regx 0x65] + 2767$C$DW$291 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R25_b2") + 2768 .dwattr $C$DW$291, DW_AT_location[DW_OP_regx 0x66] + 2769$C$DW$292 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R25_b3") + 2770 .dwattr $C$DW$292, DW_AT_location[DW_OP_regx 0x67] + 2771$C$DW$293 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R26_b0") + 2772 .dwattr $C$DW$293, DW_AT_location[DW_OP_regx 0x68] + 2773$C$DW$294 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R26_b1") + 2774 .dwattr $C$DW$294, DW_AT_location[DW_OP_regx 0x69] + 2775$C$DW$295 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R26_b2") + 2776 .dwattr $C$DW$295, DW_AT_location[DW_OP_regx 0x6a] + 2777$C$DW$296 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R26_b3") + 2778 .dwattr $C$DW$296, DW_AT_location[DW_OP_regx 0x6b] + 2779$C$DW$297 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R27_b0") + 2780 .dwattr $C$DW$297, DW_AT_location[DW_OP_regx 0x6c] + 2781$C$DW$298 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R27_b1") + 2782 .dwattr $C$DW$298, DW_AT_location[DW_OP_regx 0x6d] + 2783$C$DW$299 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R27_b2") + 2784 .dwattr $C$DW$299, DW_AT_location[DW_OP_regx 0x6e] + 2785$C$DW$300 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R27_b3") + 2786 .dwattr $C$DW$300, DW_AT_location[DW_OP_regx 0x6f] + 2787$C$DW$301 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R28_b0") + 2788 .dwattr $C$DW$301, DW_AT_location[DW_OP_regx 0x70] + 2789$C$DW$302 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R28_b1") + 2790 .dwattr $C$DW$302, DW_AT_location[DW_OP_regx 0x71] + 2791$C$DW$303 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R28_b2") + 2792 .dwattr $C$DW$303, DW_AT_location[DW_OP_regx 0x72] + 2793$C$DW$304 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R28_b3") + 2794 .dwattr $C$DW$304, DW_AT_location[DW_OP_regx 0x73] + 2795$C$DW$305 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R29_b0") + 2796 .dwattr $C$DW$305, DW_AT_location[DW_OP_regx 0x74] + 2797$C$DW$306 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R29_b1") + 2798 .dwattr $C$DW$306, DW_AT_location[DW_OP_regx 0x75] + 2799$C$DW$307 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R29_b2") + 2800 .dwattr $C$DW$307, DW_AT_location[DW_OP_regx 0x76] + 2801$C$DW$308 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R29_b3") + 2802 .dwattr $C$DW$308, DW_AT_location[DW_OP_regx 0x77] + 2803$C$DW$309 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R30_b0") + 2804 .dwattr $C$DW$309, DW_AT_location[DW_OP_regx 0x78] + 2805$C$DW$310 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R30_b1") +PRU Assembler Unix v2.1.5 Fri Jun 5 14:14:07 2020 + +Tools Copyright (c) 2012-2017 Texas Instruments Incorporated +/tmp/cloud9-examples/cycle.pru0.asm PAGE 52 + + 2806 .dwattr $C$DW$310, DW_AT_location[DW_OP_regx 0x79] + 2807$C$DW$311 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R30_b2") + 2808 .dwattr $C$DW$311, DW_AT_location[DW_OP_regx 0x7a] + 2809$C$DW$312 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R30_b3") + 2810 .dwattr $C$DW$312, DW_AT_location[DW_OP_regx 0x7b] + 2811$C$DW$313 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R31_b0") + 2812 .dwattr $C$DW$313, DW_AT_location[DW_OP_regx 0x7c] + 2813$C$DW$314 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R31_b1") + 2814 .dwattr $C$DW$314, DW_AT_location[DW_OP_regx 0x7d] + 2815$C$DW$315 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R31_b2") + 2816 .dwattr $C$DW$315, DW_AT_location[DW_OP_regx 0x7e] + 2817$C$DW$316 .dwtag DW_TAG_TI_assign_register, DW_AT_name("R31_b3") + 2818 .dwattr $C$DW$316, DW_AT_location[DW_OP_regx 0x7f] + 2819 .dwendtag $C$DW$CU + 2820 + +No Assembly Errors, No Assembly Warnings diff --git a/pru-cookbook/07more/code/delay-test.pru0.c b/pru-cookbook/07more/code/delay-test.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..337ecbb23cb0920a687b30b3e50dcdbf5ae86f86 --- /dev/null +++ b/pru-cookbook/07more/code/delay-test.pru0.c @@ -0,0 +1,28 @@ +// Shows how to call an assembly routine with one parameter +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +// The function is defined in delay.asm in same dir +// We just need to add a declaration here, the defination can be +// seperately linked +extern void my_delay_cycles(uint32_t); + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_31; // Select which pin to toggle.; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while(1) { + __R30 |= gpio; // Set the GPIO pin to 1 + my_delay_cycles(1); + __R30 &= ~gpio; // Clear the GPIO pin + my_delay_cycles(1); + } +} diff --git a/pru-cookbook/07more/code/delay-test2.pru0.c b/pru-cookbook/07more/code/delay-test2.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..763b3a7d7b0452139cb817b0d56bf5e13688e0f6 --- /dev/null +++ b/pru-cookbook/07more/code/delay-test2.pru0.c @@ -0,0 +1,32 @@ +// Shows how to call an assembly routine with a return value +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define TEST 100 + +// The function is defined in delay.asm in same dir +// We just need to add a declaration here, the defination can be +// seperately linked +extern uint32_t my_delay_cycles(uint32_t); + +uint32_t ret; + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_31; // Select which pin to toggle.; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while(1) { + __R30 |= gpio; // Set the GPIO pin to 1 + ret = my_delay_cycles(1); + __R30 &= ~gpio; // Clear the GPIO pin + ret = my_delay_cycles(1); + } +} diff --git a/pru-cookbook/07more/code/delay.pru0.asm b/pru-cookbook/07more/code/delay.pru0.asm new file mode 100644 index 0000000000000000000000000000000000000000..1c1cce7d1d986e55bc3bc3f8bf3cc3e3d783fc21 --- /dev/null +++ b/pru-cookbook/07more/code/delay.pru0.asm @@ -0,0 +1,9 @@ +; This is an example of how to call an assembly routine from C. +; Mark A. Yoder, 9-July-2018 + .global my_delay_cycles +my_delay_cycles: +delay: + sub r14, r14, 1 ; The first argument is passed in r14 + qbne delay, r14, 0 + + jmp r3.w2 ; r3 contains the return address \ No newline at end of file diff --git a/pru-cookbook/07more/code/delay2.pru0.asm b/pru-cookbook/07more/code/delay2.pru0.asm new file mode 100644 index 0000000000000000000000000000000000000000..e1f1da03d37ece54ca7c577b4244bd8abbbc921f --- /dev/null +++ b/pru-cookbook/07more/code/delay2.pru0.asm @@ -0,0 +1,15 @@ +; This is an example of how to call an assembly routine from C with a retun value. +; Mark A. Yoder, 9-July-2018 + + .cdecls "delay-test2.pru0.c" + + .global my_delay_cycles +my_delay_cycles: +delay: + sub r14, r14, 1 ; The first argument is passed in r14 + qbne delay, r14, 0 + + ldi r14, TEST ; TEST is defined in delay-test2.c + ; r14 is the return register + + jmp r3.w2 ; r3 contains the return address \ No newline at end of file diff --git a/pru-cookbook/07more/code/logic.c b/pru-cookbook/07more/code/logic.c new file mode 100644 index 0000000000000000000000000000000000000000..f774708d22f3ef26ede77fe45a8057d7f1e905f2 --- /dev/null +++ b/pru-cookbook/07more/code/logic.c @@ -0,0 +1,34 @@ +// Access the CYCLE and STALL registers +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + // Copy lower 8 bits to r16 + struct { + uint8_t b0; + uint8_t b1; + uint8_t b2; + uint8_t b3; + } r17; + + // Clear SYSCFG[STANDBY_INIT] to enable OCP master port + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + +// r16.b0 = __R31.b0; + __asm(" mov r17.b0, r31.b0"); + __delay_cycles(1); + r17.b1 = (uint32_t) __R31; + __delay_cycles(1); + r17.b2 = (uint32_t) __R31; + __delay_cycles(1); + r17.b3 = (uint32_t) __R31; + + __xout(10, 16, 0, r17); + + __halt(); +} diff --git a/pru-cookbook/07more/code/logic_setup.sh b/pru-cookbook/07more/code/logic_setup.sh new file mode 100644 index 0000000000000000000000000000000000000000..2253459cf1f8efde2e61281525f26872a3940a42 --- /dev/null +++ b/pru-cookbook/07more/code/logic_setup.sh @@ -0,0 +1,30 @@ +#!/bin/bash +# +export PRUN=0 +export TARGET=logic +echo PRUN=$PRUN +echo TARGET=$TARGET + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_31 P9_29 P9_30 P9_28 P9_92 P9_27 P9_91 P9_25 " +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_36 P1_33 P2_32 P2_30 P1_31 P2_34 P2_28 P1_29" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruin + config-pin -q $pin +done diff --git a/pru-cookbook/07more/code/resource_table_pru0.h b/pru-cookbook/07more/code/resource_table_pru0.h new file mode 100644 index 0000000000000000000000000000000000000000..06c148006b7a892d30f16d69891c6165e3525161 --- /dev/null +++ b/pru-cookbook/07more/code/resource_table_pru0.h @@ -0,0 +1,105 @@ +/* + * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +/* + * ======== rsc_table_pru.h ======== + * + * Define the resource table entries for PRU0. This will be + * incorporated into corresponding base images, and used by the remoteproc + * on the host-side to allocated/reserve resources. Note the remoteproc + * driver requires that all PRU firmware be built with a resource table. + * + * + */ + +#ifndef _RSC_TABLE_PRU_H_ +#define _RSC_TABLE_PRU_H_ + +#include <stddef.h> +#include <rsc_types.h> +#include "pru_virtio_ids.h" + +/* + * Sizes of the virtqueues (expressed in number of buffers supported, + * and must be power of 2) + */ +#define PRU_RPMSG_VQ0_SIZE 2 +#define PRU_RPMSG_VQ1_SIZE 2 + +/* flip up bits whose indices represent features we support */ +#define RPMSG_PRU_C0_FEATURES 1 + +/* Definition for unused interrupts */ +#define HOST_UNUSED 255 + +/* Mapping sysevts to a channel. Each pair contains a sysevt, channel */ +struct ch_map pru_intc_map[] = { {17, 1}, {18, 0}, {19, 2}, {20, 3}, {21, 0}, + {22, 1}, {24, 4}, {25, 5}, {26, 6}, {27, 7}, +}; + +struct my_resource_table { + struct resource_table base; + + uint32_t offset[1]; /* Should match 'num' in actual definition */ + + /* intc definition */ + struct fw_rsc_custom pru_ints; +}; + +#pragma DATA_SECTION(am335x_pru_remoteproc_ResourceTable, ".resource_table") +#pragma RETAIN(am335x_pru_remoteproc_ResourceTable) +struct my_resource_table am335x_pru_remoteproc_ResourceTable = { + 1, /* we're the first version that implements this */ + 1, /* number of entries in the table */ + 0, 0, /* reserved, must be zero */ + /* offsets to entries */ + { + offsetof(struct my_resource_table, pru_ints), + }, + + { + TYPE_CUSTOM, TYPE_PRU_INTS, + sizeof(struct fw_rsc_custom_ints), + { /* PRU_INTS version */ + 0x0000, + /* Channel-to-host mapping, 255 for unused */ + 0, 1, 2, 3, 0, 6, 1, 7, HOST_UNUSED, HOST_UNUSED, + /* Number of evts being mapped to channels */ + (sizeof(pru_intc_map) / sizeof(struct ch_map)), + /* Pointer to the structure containing mapped events */ + pru_intc_map, + }, + }, +}; + +#endif /* _RSC_TABLE_PRU_H_ */ diff --git a/pru-cookbook/07more/code/setup.sh b/pru-cookbook/07more/code/setup.sh new file mode 100644 index 0000000000000000000000000000000000000000..f1d0d9db334b125b68171dd18ce04abec94ace08 --- /dev/null +++ b/pru-cookbook/07more/code/setup.sh @@ -0,0 +1,30 @@ +#!/bin/bash +# +export TARGET=delay-test.pru0 +export TARGETasm=delay.pru0 +echo TARGET=$TARGET +echo TARGETasm=$TARGETasm + +# Configure the PRU pins based on which Beagle is running +machine=$(awk '{print $NF}' /proc/device-tree/model) +echo -n $machine +if [ $machine = "Black" ]; then + echo " Found" + pins="P9_31" +elif [ $machine = "Blue" ]; then + echo " Found" + pins="" +elif [ $machine = "PocketBeagle" ]; then + echo " Found" + pins="P1_33" +else + echo " Not Found" + pins="" +fi + +for pin in $pins +do + echo $pin + config-pin $pin pruout + config-pin -q $pin +done diff --git a/pru-cookbook/07more/code/xin.pru1.c b/pru-cookbook/07more/code/xin.pru1.c new file mode 100644 index 0000000000000000000000000000000000000000..56d4a68209d974c51e50559cc6d90ff9046f3a68 --- /dev/null +++ b/pru-cookbook/07more/code/xin.pru1.c @@ -0,0 +1,34 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/trees/master/examples/am335x/PRU_Direct_Connect1 +#include <stdint.h> +#include "resource_table_empty.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +typedef struct { + uint32_t reg5; + uint32_t reg6; + uint32_t reg7; + uint32_t reg8; + uint32_t reg9; + uint32_t reg10; +} bufferData; + +bufferData dmemBuf; + +/* PRU-to-ARM interrupt */ +#define PRU1_PRU0_INTERRUPT (18) +#define PRU1_ARM_INTERRUPT (20+16) + +void main(void) +{ + /* Let PRU0 know that I am awake */ + __R31 = PRU1_PRU0_INTERRUPT+16; + + /* XFR registers R5-R10 from PRU0 to PRU1 */ + /* 14 is the device_id that signifies a PRU to PRU transfer */ + __xin(14, 5, 0, dmemBuf); + + /* Halt the PRU core */ + __halt(); +} diff --git a/pru-cookbook/07more/code/xout-cycle.pru0.c b/pru-cookbook/07more/code/xout-cycle.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..339fedfc38eed9d9372a0b55ee74dfc031be70c2 --- /dev/null +++ b/pru-cookbook/07more/code/xout-cycle.pru0.c @@ -0,0 +1,62 @@ +// Version of xout.c with code to use CYCLE to count cycle times. +#include <stdint.h> +#include <pru_intc.h> +#include <pru_ctrl.h> +#include "resource_table_pru0.h" + +#define PRU0 + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +typedef struct { + uint32_t reg5; + uint32_t reg6; + uint32_t reg7; + uint32_t reg8; + uint32_t reg9; + uint32_t reg10; +} bufferData; + +bufferData dmemBuf; + +/* PRU-to-ARM interrupt */ +#define PRU1_PRU0_INTERRUPT (18) +#define PRU0_ARM_INTERRUPT (19+16) + +void main(void) +{ + uint32_t cycleXX; // Use a name that's easy to search + /* Clear the status of all interrupts */ + CT_INTC.SECR0 = 0xFFFFFFFF; + CT_INTC.SECR1 = 0xFFFFFFFF; + + /* Load the buffer with default values to transfer */ + dmemBuf.reg5 = 0xDEADBEEF; + dmemBuf.reg6 = 0xAAAAAAAA; + dmemBuf.reg7 = 0x12345678; + dmemBuf.reg8 = 0xBBBBBBBB; + dmemBuf.reg9 = 0x87654321; + dmemBuf.reg10 = 0xCCCCCCCC; + + /* Poll until R31.30 (PRU0 interrupt) is set + * This signals PRU1 is initialized */ + while ((__R31 & (1<<30)) == 0) { + } + + /* XFR registers R5-R10 from PRU0 to PRU1 */ + /* 14 is the device_id that signifies a PRU to PRU transfer */ + PRU0_CTRL.CTRL_bit.CTR_EN = 1; // Enable cycle counter + + __xout(14, 5, 0, dmemBuf); + + cycleXX = PRU0_CTRL.CYCLE; // Read cycle and store in a register + + /* Clear the status of the interrupt */ + CT_INTC.SICR = PRU1_PRU0_INTERRUPT; + + dmemBuf.reg5 = cycleXX; + + /* Halt the PRU core */ + __halt(); +} diff --git a/pru-cookbook/07more/code/xout.pru0.c b/pru-cookbook/07more/code/xout.pru0.c new file mode 100644 index 0000000000000000000000000000000000000000..bfcdbf5f927700c2d250abc0f28850bb4884392f --- /dev/null +++ b/pru-cookbook/07more/code/xout.pru0.c @@ -0,0 +1,52 @@ +// From: http://git.ti.com/pru-software-support-package/pru-software-support-package/trees/master/examples/am335x/PRU_Direct_Connect0 +#include <stdint.h> +#include <pru_intc.h> +#include "resource_table_pru0.h" + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +typedef struct { + uint32_t reg5; + uint32_t reg6; + uint32_t reg7; + uint32_t reg8; + uint32_t reg9; + uint32_t reg10; +} bufferData; + +bufferData dmemBuf; + +/* PRU-to-ARM interrupt */ +#define PRU1_PRU0_INTERRUPT (18) +#define PRU0_ARM_INTERRUPT (19+16) + +void main(void) +{ + /* Clear the status of all interrupts */ + CT_INTC.SECR0 = 0xFFFFFFFF; + CT_INTC.SECR1 = 0xFFFFFFFF; + + /* Load the buffer with default values to transfer */ + dmemBuf.reg5 = 0xDEADBEEF; + dmemBuf.reg6 = 0xAAAAAAAA; + dmemBuf.reg7 = 0x12345678; + dmemBuf.reg8 = 0xBBBBBBBB; + dmemBuf.reg9 = 0x87654321; + dmemBuf.reg10 = 0xCCCCCCCC; + + /* Poll until R31.30 (PRU0 interrupt) is set + * This signals PRU1 is initialized */ + while ((__R31 & (1<<30)) == 0) { + } + + /* XFR registers R5-R10 from PRU0 to PRU1 */ + /* 14 is the device_id that signifies a PRU to PRU transfer */ + __xout(14, 5, 0, dmemBuf); + + /* Clear the status of the interrupt */ + CT_INTC.SICR = PRU1_PRU0_INTERRUPT; + + /* Halt the PRU core */ + __halt(); +} diff --git a/pru-cookbook/07more/code/xout_run.sh b/pru-cookbook/07more/code/xout_run.sh new file mode 100755 index 0000000000000000000000000000000000000000..158cf472beddca5d3781df4026dbeea4de3338b7 --- /dev/null +++ b/pru-cookbook/07more/code/xout_run.sh @@ -0,0 +1,3 @@ +# Be sure to start PRU 0 before PRU 1. PRU 0 will wait for PRU 1 to signal it. +make TARGET=xout.pru0 +make TARGET=xin.pru1 diff --git a/pru-cookbook/07more/figures/my_delay_cycles.png b/pru-cookbook/07more/figures/my_delay_cycles.png new file mode 100644 index 0000000000000000000000000000000000000000..125a85c0ba66e1bbc3e2bb6bffefae68538f0216 Binary files /dev/null and b/pru-cookbook/07more/figures/my_delay_cycles.png differ diff --git a/pru-cookbook/07more/more.rst b/pru-cookbook/07more/more.rst new file mode 100644 index 0000000000000000000000000000000000000000..ae3abcf84c1598a23f54a2a0ea044952809d38ce --- /dev/null +++ b/pru-cookbook/07more/more.rst @@ -0,0 +1,443 @@ +.. _pru-cookbook-more: + +More Performance +################## + +So far in all our examples we've been able to meet our timing goals by writing +our code in the C programming language. The C compiler does a suprisingly +good job at generating code, most the time. However there are times +when very precise timing is needed and the compiler isn't doing it. + +At these times you need to write in assembly language. This chapter +introduces the PRU assembler and shows how to call assembly code from +C. Detailing on how to program in assembly are beyond the scope of this text. + +The following are resources used in this chapter. + +Resources +~~~~~~~~~~ + +* `PRU Optimizing C/C++ Compiler, v2.2, User's Guide <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ +* `PRU Assembly Language Tools User's Guide <http://www.ti.com/lit/ug/spruhv6b/spruhv6b.pdf>`_ +* `PRU Assembly Instruction User Guide <http://www.ti.com/lit/ug/spruij2/spruij2.pdf>`_ + +Calling Assembly from C +------------------------- + +Problem +~~~~~~~~~ + +You have some C code and you want to call an assembly language routine from it. + +Solution +~~~~~~~~~ + +You need to do two things, write the assembler file and modify the ``Makefile`` +to include it. For example, let's write our own ``my_delay_cycles`` routine in +in assembly. The intrinsic ``pass:[__]delay_cycles`` must be passed a compile time +constant. Our new ``delay_cycles`` can take a runtime delay value. + +:ref:`more_delay-test` is much like our other c code, but on line 10 we declare +``my_delay_cycles`` and then on lines 24 and 26 we'll call it with an argument of 1. + +.. _more_delay-test: + +delay-test.pru0.c +~~~~~~~~~~~~~~~~~~~ + +:download:`code/delay-test.pru0.c <code/delay-test.pru0.c>` + +:ref:`more_delay` is the assembly code. + +.. _more_delay: + +delay.pru0.asm +~~~~~~~~~~~~~~~ + +:download:`delay.pru0.asm <code/delay.pru0.asm>` + +The ``Makefile`` has one addition that needs to be made to compile both :ref:`more_delay-test` +and :ref:`more_delay`. +If you look in the local ``Makefile`` you'll see: + +.. _more_makefile: + +Makefile +~~~~~~~~~ + +:download:`Makefile <code/Makefile>` + +This Makefle includes a common Makfile at ``/var/lib/cloud9/common/Makefile``, this the Makefile +you need to edit. Edit ``/var/lib/cloud9/common/Makefile`` and go to line 195. + +.. code-block:: bash + + $(GEN_DIR)/%.out: $(GEN_DIR)/%.o *$(GEN_DIR)/$(TARGETasm).o* + @mkdir -p $(GEN_DIR) + @echo 'LD $^' + $(eval $(call target-to-proc,$@)) + $(eval $(call proc-to-build-vars,$@)) + @$(LD) $@ $^ $(LDFLAGS) + +Add ``*(GEN_DIR)/$(TARGETasm).o*`` as shown in bold above. You will want to remove +this addition once you are done with this example since it will break the other examples. + +The following will compile and run everything. + +.. code-block:: bash + + bone$ *config-pin P9_31 pruout* + bone$ *make TARGET=delay-test.pru0 TARGETasm=delay.pru0* + /var/lib/cloud9/common/Makefile:29: MODEL=TI_AM335x_BeagleBone_Black,TARGET=delay-test.pru0 + - Stopping PRU 0 + - copying firmware file /tmp/cloud9-examples/delay-test.pru0.out to /lib/firmware/am335x-pru0-fw + write_init_pins.sh + - Starting PRU 0 + MODEL = TI_AM335x_BeagleBone_Black + PROC = pru + PRUN = 0 + PRU_DIR = /sys/class/remoteproc/remoteproc1 + +The resulting output is shown in :ref:`more_my_delay_cycles`. + +.. _more_my_delay_cycles: + +Output of my_delay_cycles() +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. figure:: figures/my_delay_cycles.png + :align: center + :alt: Output of my_delay_cycles() + +Notice the on time is about 35ns and the off time is 30ns. + +Discission +~~~~~~~~~~~~ + +There is much to explain here. Let's start with :ref:`more_delay`. + +Line-by-line of delay.pru0.asm +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +.. table:: + + +-------+-------------------------------------------------------------------------------------------------------+ + |Line | Explanation | + +=======+=======================================================================================================+ + |3 | Declare `my_delay_cycles` to be global so the linker can find it. | + +-------+-------------------------------------------------------------------------------------------------------+ + |4 | Label the starting point for `my_delay_cycles`. | + +-------+-------------------------------------------------------------------------------------------------------+ + |5 | Label for our delay loop. | + +-------+-------------------------------------------------------------------------------------------------------+ + |6 | The first argument is passed in register ``r14``. Page 111 of | + | | `PRU Optimizing C/C++ Compiler, v2.2, User's Guide <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ | + | | gives the argument passing convention. Registers ``r14`` to ``r29`` are used | + | | to pass arguments, if there are more arguments, the argument stack (``r4``) | + | | is used. The other register conventions are found on page 108. | + | | Here we subtract 1 from ``r14`` and save it back into ``r14``. | + +-------+-------------------------------------------------------------------------------------------------------+ + |7 | `qbne` is a quick branch if not equal. | + +-------+-------------------------------------------------------------------------------------------------------+ + |9 | Once we've delayed enough we drop through the quick branch and | + | | hit the jump. The upper bits of register `r3` has the return address, | + | | therefore we return to the c code. | + +-------+-------------------------------------------------------------------------------------------------------+ + +:ref:`more_my_delay_cycles` shows the **on** time is 35ns and the off time is 30ns. +With 5ns/cycle this gives 7 cycles on and 6 off. These times make sense +because each instruction takes a cycle and you have, set ``R30``, jump to +``my_delay_cycles``, ``sub``, ``qbne``, ``jmp``. Plus the instruction (not seen) that +initilizes `r14` to the passed value. That's a total of six instructions. +The extra instruction is the branch at the bottom of the ``while`` loop. + + +Returning a Value from Assembly +-------------------------------- + +Problem +~~~~~~~~~ + +Your assembly code needs to return a value. + +Solution +~~~~~~~~~ + +``R14`` is how the return value is passed back. :ref:`more_test2` shows the c code. + +.. _more_test2: + +delay-test2.pru0.c +~~~~~~~~~~~~~~~~~~~ + +:download:`delay-test2.pru0.c <code/delay-test2.pru0.c>` + +:ref:`more_delay2` is the assembly code. + +.. _more_delay2: + +delay2.pru0.asm +~~~~~~~~~~~~~~~~~ + +:download:`delay2.pru0.asm <code/delay2.pru0.asm>` + +An additional feature is shown in line 4 of :ref:`more_delay2`. The +``.cdecls "delay-test2.pru0.c"`` says to include any defines from ``delay-test2.pru0.c`` +In this example, line 6 of :ref:`more_test2` `#defines` TEST and line 12 of +:ref:`more_delay2` reference it. + + +Using the Built-In Counter for Timing +--------------------------------------- + +Problem +~~~~~~~~~ + +I want to count how many cycles my routine takes. + +Solution +~~~~~~~~~ + +Each PRU has a ``CYCLE`` register which counts the number of cycles since +the PRU was enabled. They also have a ``STALL`` register that counts how +many times the PRU stalled fetching an instruction. +:ref:`more_cycle` shows they are used. + +.. _more_cycle: + +cycle.pru0.c - Code to count cycles. +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`delay2.pru0.asm <code/cycle.pru0.c>` + +Discission +~~~~~~~~~~~~ + +The code is mostly the same as other examples. ``cycle`` and ``stall`` end up in registers which +we can read using prudebug. :ref:`more_cycle_lines` is the Line-by-line. + +.. _more_cycle_lines: + +Line-by-line for cycle.pru0.c +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +-------+---------------------------------------------------------------------------------------+ + |Line | Explanation | + +=======+=======================================================================================+ + |4 | Include needed to reference `CYCLE` and `STALL`. | + +-------+---------------------------------------------------------------------------------------+ + |16 | Declaring `cycle` and `stall`. The compiler will optimize these and just | + | | keep them in registers. We'll have to look at the `cycle.pru0.lst` file to see where | + | | they are stored. | + +-------+---------------------------------------------------------------------------------------+ + |21 | Enables `CYCLE`. | + +-------+---------------------------------------------------------------------------------------+ + |26 | Reset `CYCLE`. It ignores the value assigned to it and always sets it | + | | to 0. `cycle` is on the right hand side to make the compiler give it it's own | + | | register. | + +-------+---------------------------------------------------------------------------------------+ + |28, 29 | Reads the `CYCLE` and `STALL` values into registers. | + +-------+---------------------------------------------------------------------------------------+ + +You can see where ``cycle`` and ``stall`` are stored by looking into :ref:`more_cycle_list0`. + +.. _more_cycle_list0: + +/tmp/cloud9-examples/cycle.pru0.lst Lines 113..119 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`cycle.pru0.lst lines=113..119 <code/cycle.pru0.lst>` + +Here the ``LDI32`` instruction loads the address ``0x22000`` into ``r0``. This is the offset to +the ``CTRL`` registers. Later in the file we see :ref:`more_cycle_list1`. + +.. _more_cycle_list1: + +/tmp/cloud9-examples/cycle.pru0.lst Lines 146..152 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +:download:`lines=146..152 <code/cycle.pru0.lst>` + + +The first ``LBBO`` takes the contents of ``r0`` and adds the offset 12 to it and copies 4 bytes +into ``r1``. This points to ``CYCLE``, so ``r1`` has the contents of ``CYCLE``. + +The second ``LBBO`` does the same, but with offset 16, which points to ``STALL``, +thus ``STALL`` is now in ``r0``. + +Now fire up **prudebug** and look at those registers. + +.. code-block:: bash + + bone$ *sudo prudebug* + PRU0> *r* + r + r + Register info for PRU0 + Control register: 0x00000009 + Reset PC:0x0000 STOPPED, FREE_RUN, COUNTER_ENABLED, NOT_SLEEPING, PROC_DISABLED + + Program counter: 0x0012 + Current instruction: HALT + + R00: *0x00000005* R08: 0x00000200 R16: 0x000003c6 R24: 0x00110210 + R01: *0x00000003* R09: 0x00000000 R17: 0x00000000 R25: 0x00000000 + R02: 0x000000fc R10: 0xfff4ea57 R18: 0x000003e6 R26: 0x6e616843 + R03: 0x0004272c R11: 0x5fac6373 R19: 0x30203020 R27: 0x206c656e + R04: 0xffffffff R12: 0x59bfeafc R20: 0x0000000a R28: 0x00003033 + R05: 0x00000007 R13: 0xa4c19eaf R21: 0x00757270 R29: 0x02100000 + R06: 0xefd30a00 R14: 0x00000005 R22: 0x0000001e R30: 0xa03f9990 + R07: 0x00020024 R15: 0x00000003 R23: 0x00000000 R31: 0x00000000 + + +So ``cycle`` is 3 and ``stall`` is 5. It must be one cycle to clear the GPIO and 2 cycles to read the +``CYCLE`` register and save it in the register. It's interesting there are 5 ``stall`` cycles. + +If you switch the order of lines 30 and 31 you'll see ``cycle`` is 7 and ``stall`` is 2. ``cycle`` now includes the +time needed to read ``stall`` and ``stall`` no longer includes the time to read ``cycle``. + +Xout and Xin - Transfering Between PRUs +--------------------------------------- + +Problem +~~~~~~~~~ + +I need to transfer data between PRUs quickly. + +Solution +~~~~~~~~~ + +The ``pass:[__]xout()`` and ``pass:[__]xin()`` intrinsics are able to transfer up to 30 registers between PRU 0 and PRU 1 quickly. +:ref:`more_xout` shows how ``xout()`` running on PRU 0 transfers six registers to PRU 1. + +.. _more_xout: + +xout.pru0.c +~~~~~~~~~~~~ + +:download:`xout.pru0.c <code/xout.pru0.c>` + +PRU 1 waits at line 41 until PRU 0 signals it. :ref:`more_xin` sends sends an +interupt to PRU 0 and waits for it to send the data. + +.. _more_xin: + +xin.pru1.c +~~~~~~~~~~~ + +:download:`xin.pru1.c <code/xin.pru1.c>` + +Use ``prudebug`` to see registers R5-R10 are transfered from PRU 0 to PRU 1. + +.. code-block:: bash + + PRU0> *r* + Register info for PRU0 + Control register: 0x00000001 + Reset PC:0x0000 STOPPED, FREE_RUN, COUNTER_DISABLED, NOT_SLEEPING, PROC_DISABLED + + Program counter: 0x0026 + Current instruction: HALT + + R00: 0x00000012 *R08: 0xbbbbbbbb* R16: 0x000003c6 R24: 0x00110210 + R01: 0x00020000 *R09: 0x87654321* R17: 0x00000000 R25: 0x00000000 + R02: 0x000000e4 *R10: 0xcccccccc* R18: 0x000003e6 R26: 0x6e616843 + R03: 0x0004272c R11: 0x5fac6373 R19: 0x30203020 R27: 0x206c656e + R04: 0xffffffff R12: 0x59bfeafc R20: 0x0000000a R28: 0x00003033 + *R05: 0xdeadbeef* R13: 0xa4c19eaf R21: 0x00757270 R29: 0x02100000 + *R06: 0xaaaaaaaa* R14: 0x00000005 R22: 0x0000001e R30: 0xa03f9990 + *R07: 0x12345678* R15: 0x00000003 R23: 0x00000000 R31: 0x00000000 + + PRU0> *pru 1* + pru 1 + Active PRU is PRU1. + + PRU1> *r* + r + Register info for PRU1 + Control register: 0x00000001 + Reset PC:0x0000 STOPPED, FREE_RUN, COUNTER_DISABLED, NOT_SLEEPING, PROC_DISABLED + + Program counter: 0x000b + Current instruction: HALT + + R00: 0x00000100 *R08: 0xbbbbbbbb* R16: 0xe9da228b R24: 0x28113189 + R01: 0xe48cdb1f *R09: 0x87654321* R17: 0x66621777 R25: 0xddd29ab1 + R02: 0x000000e4 *R10: 0xcccccccc* R18: 0x661f83ea R26: 0xcf1cd4a5 + R03: 0x0004db97 R11: 0xdec387d5 R19: 0xa85adb78 R27: 0x70af2d02 + R04: 0xa90e496f R12: 0xbeac3878 R20: 0x048fff22 R28: 0x7465f5f0 + *R05: 0xdeadbeef* R13: 0x5777b488 R21: 0xa32977c7 R29: 0xae96b530 + *R06: 0xaaaaaaaa* R14: 0xffa60550 R22: 0x99fb123e R30: 0x52c42a0d + *R07: 0x12345678* R15: 0xdeb2142d R23: 0xa353129d R31: 0x00000000 + + +Discussion +~~~~~~~~~ + +:ref:`more_zout_lines` shows the line-by-line for ``xout.pru0.c`` + +.. _more_zout_lines: + +xout.pru0.c Line-by-line +~~~~~~~~~~~~~~~~~~~~~~~~~ +.. table:: + + +-------+---------------------------------------------------------------------------------------------------------+ + |Line | Explanation | + +=======+=========================================================================================================+ + |4 | A different resource so PRU 0 can receive a signal from PRU 1. | + +-------+---------------------------------------------------------------------------------------------------------+ + |9-16 | ``dmemBuf`` holds the data to be sent to PRU 1. Each will be transfered | + | | to its corresponding register by ``xout()``. | + +-------+---------------------------------------------------------------------------------------------------------+ + |21-22 | Define the interupts we're using. | + +-------+---------------------------------------------------------------------------------------------------------+ + |27-28 | Clear the interrupts. | + +-------+---------------------------------------------------------------------------------------------------------+ + |31-36 | Initialize dmemBuf with easy to recognize values. | + +-------+---------------------------------------------------------------------------------------------------------+ + |40 | Wait for PRU 1 to signal. | + +-------+---------------------------------------------------------------------------------------------------------+ + |45 | ``pass:[__]xout()`` does a direct transfer to PRU 1. Page 92 of | + | | `PRU Optimizing C/C++ Compiler, v2.2, User's Guide <http://www.ti.com/lit/ug/spruhv7b/spruhv7b.pdf>`_ | + | | shows how to use `xout()`. The first argument, 14, says to do a direct transfer to PRU 1. If the | + | | first argument is 10, 11 or 12, the data is transfered to one of three scratchpad memories that | + | | PRU 1 can access later. The second argument, 5, says to start transfering with register ``r5`` | + | | and use as many regsiters as needed to transfer all of ``dmemBuf``. The third argument, 0, | + | | says to not use remapping. (See the User's Guide for details.) | + | | The final argument is the data to be transfered. | + +-------+---------------------------------------------------------------------------------------------------------+ + |48 | Clear the interupt so it can go again. | + +-------+---------------------------------------------------------------------------------------------------------+ + +:ref:`more_xin_lines` shows the line-by-line for ``xin.pru1.c``. + +.. _more_xin_lines: + +xin.pru1.c Line-by-line +~~~~~~~~~~~~~~~~~~~~~~~~ +.. table:: + + +-------+-----------------------------------------------------------+ + |Line | Explanation | + +=======+===========================================================+ + |8-15 | Place to put the received data. | + +-------+-----------------------------------------------------------+ + |26 | Signal PRU 0 | + +-------+-----------------------------------------------------------+ + |30 | Receive the data. The arguments are the same as `xout()`, | + | | 14 says to get the data directly from PRU 0. | + | | 5 says to start with register `r5`. | + | | `dmemBuf` is where to put the data. | + +-------+-----------------------------------------------------------+ + +If you really need speed, considering using ``pass:[__]xout()`` and ``pass:[__]xin()`` in assembly. + +Copyright +---------- + +copyright.c +~~~~~~~~~~~~~ + +:download:`copyright.c <code/copyright.c>` diff --git a/pru-cookbook/08ai/ai.rst b/pru-cookbook/08ai/ai.rst new file mode 100644 index 0000000000000000000000000000000000000000..db36f383f27b505f3f613a471bae81e0525db48e --- /dev/null +++ b/pru-cookbook/08ai/ai.rst @@ -0,0 +1,377 @@ +.. _pru-cookbook-ai: + +Moving to the BeagleBone AI +############################ + +So far all our examples have focussed mostly on the BeagleBone Black and Pocket Beagle. +These are both based on the am335x chip. The new kid on the block is the BeagleBone AI +which is based on the am5729. The new chip brings with it new capabilities one of +which is four PRUs. This chapter details what changes when moving from two to +four PRUs. + +The following are resources used in this chapter. + +Resources +~~~~~~~~~~~ + +* `AM572x Technical Reference Manual <http://www.ti.com/lit/pdf/spruhz6l>`_ (AI) +* `BeagleBone AI PRU pins <https://docs.google.com/spreadsheets/d/1dFSBVem86vAUD7MLXvqdS-N0Efi8_g_O1iTqzql8DAo/edit#gid=0>`_ + +Moving from two to four PRUs +============================= + +Problem +-------- + +You have code that works on the am335x PRUs and you want to move it to the +am5729 on the AI. + +Solution +-------- + +Things to consider when moving to the AI are: + +* Which pins are you going to use +* Which PRU are you going to run on + +Knowing which pins to use impacts the PRU you'll use. + +Discission +-------- + +The various System Reference Manuals (SRM's) list which pins go to the PRUs. +Here the tables are combined into one to make it easier to see what goes where. + +.. _aimapping_bits: + +Mapping bit positions to pin names +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. table:: + + +---+---+---------+-----------+-----------+-------------+ + |PRU|Bit|Black pin|AI PRU1 pin|AI PRU2 pin|Pocket pin | + |0 |0 |P9_31 | |P8_44 |P1.36 | + +---+---+---------+-----------+-----------+-------------+ + |0 |1 |P9_29 | |P8_41 |P1.33 | + +---+---+---------+-----------+-----------+-------------+ + |0 |2 |P9_30 | |P8_42/P8_21|P2.32 | + +---+---+---------+-----------+-----------+-------------+ + |0 |3 |P9_28 |P8_12 |P8_39/P8_20|P2.30 | + +---+---+---------+-----------+-----------+-------------+ + |0 |4 |P9_92 |P8_11 |P8_40/P8_25|P1.31 | + +---+---+---------+-----------+-----------+-------------+ + |0 |5 |P9_27 |P9_15 |P8_37/P8_24|P2.34 | + +---+---+---------+-----------+-----------+-------------+ + |0 |6 |P9_91 | |P8_38/P8_5 |P2.28 | + +---+---+---------+-----------+-----------+-------------+ + |0 |7 |P9_25 | |P8_36/P8_6 |P1.29 | + +---+---+---------+-----------+-----------+-------------+ + |0 |8 | | |P8_34/P8_23| | + +---+---+---------+-----------+-----------+-------------+ + |0 |9 | | |P8_35/P8_22| | + +---+---+---------+-----------+-----------+-------------+ + |0 |19 | | |P8_33/P8_3 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |11 | | |P8_31/P8_4 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |12 | | |P8_32 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |13 | | |P8_45 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |14 |P8_12(out) P8_16(in)||P9_11 |P2.24 | + +---+---+---------+-----------+-----------+-------------+ + |0 |15 |P8_11(out) P8_15(in)||P8_17/P9_13|P2.33 | + +---+---+---------+-----------+-----------+-------------+ + |0 |16 |P9_41(in) P9_26(in)| |P8_27 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |17 | |P9_26 |P8_28 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |18 | | |P8_29 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |19 | | |P8_30 | | + +---+---+---------+-----------+-----------+-------------+ + |0 |20 | | |P8_46/P8_8 | | + +---+---+---------+-----------+-----------+-------------+ + | | | | | | | + +---+---+---------+-----------+-----------+-------------+ + |1 |0 |P8_45 | |P8_32 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |1 |P8_46 |*P9_20* | | | + +---+---+---------+-----------+-----------+-------------+ + |1 |2 |P8_43 |*P9_19* | | | + +---+---+---------+-----------+-----------+-------------+ + |1 |3 |P8_44 |P9_41 | | | + +---+---+---------+-----------+-----------+-------------+ + |1 |4 |P8_41 | | | | + +---+---+---------+-----------+-----------+-------------+ + |1 |5 |P8_42 |*P8_18* |P9_25 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |6 |P8_39 |*P8_19* |P8_9 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |7 |P8_40 |*P8_13* |P9_31 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |8 |P8_27 | |P9_18 |P2.35 | + +---+---+---------+-----------+-----------+-------------+ + |1 |9 |P8_29 |P8_14 |P9_17 |P2.01 | + +---+---+---------+-----------+-----------+-------------+ + |1 |10 |P8_28 |P9_42 |P9_31 |P1.35 | + +---+---+---------+-----------+-----------+-------------+ + |1 |11 |P8_30 |P9_27 |P9_29 |P1.04 | + +---+---+---------+-----------+-----------+-------------+ + |1 |12 |P8_21 | |P9_30 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |13 |P8_20 | |P9_26 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |14 | |P9_14 |P9_42 |P1.32 | + +---+---+---------+-----------+-----------+-------------+ + |1 |15 | |*P9_16* |P8_10 |P1.30 | + +---+---+---------+-----------+-----------+-------------+ + |1 |16 |P9_26(in)|*P8_15* |P8_7 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |17 | |*P8_26* |P8_27 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |18 | |*P8_16* |P8_45 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |19 | | |P8_46 | | + +---+---+---------+-----------+-----------+-------------+ + |1 |19 | | |P8_43 | | + +---+---+---------+-----------+-----------+-------------+ + +The pins in *bold* are already configured as pru pins. See :ref:`ai_config` to +see what's currently configured as what. See :ref:`ai_device_tree` to +configure pins. + +.. _ai_config: + +Seeing how pins are configured +=============================== + +Problem +-------- + +You want to know how the pins are currently configured. + +Solution +--------- + +The ``show-pins.pl`` command does what you want, but you have to set it up first. + +.. code-block:: bash + + bone$ cd ~/bin + bone$ ln -s /opt/scripts/device/bone/show-pins.pl . + +This creates a symbolic link to the ``show-pins.pl`` command that is rather hidden +away. The link is put in the ``bin`` directory which is in the default command +``$PATH``. Now you can run ``show-pins.pl`` from anywhere. + +.. code-block:: bash + + bone$ *show-pins.pl* + P9.19a 16 R6 7 fast rx up i2c4_scl + P9.20a 17 T9 7 fast rx up i2c4_sda + P8.35b 57 AD9 e fast down gpio3_0 + P8.33b 58 AF9 e fast down gpio3_1 + ... + +Here you see ``P9.19a`` and ``P9.20a`` are configured for i2c with pull up resistors. +The ``P8`` pins are configured as gpio with pull down resistors. They are +both on gpio port 3. ``P8.35b`` is bit 0 while ``P8.33b`` is bit 1. You can find +which direction they are set by using ``gpioinfo`` and the chip number. +Unfortunately you subtract one from the port number to get the chip number. +So ``P8.35b`` is on chip number 2. + +.. code-block:: bash + + bone$ *gpioinfo 2* + line 0: unnamed unused *input* active-high + line 1: unnamed unused *input* active-high + line 2: unnamed unused input active-high + line 3: unnamed unused input active-high + line 4: unnamed unused input active-high + ... + +Here we see both (lines 0 and 1) are set to input. + +Adding ``-v`` gives more details. + +.. code-block:: bash + + bone$ *show-pins.pl -v* + ... + sysboot 14 14 H2 f fast down sysboot14 + sysboot 15 15 H3 f fast down sysboot15 + P9.19a 16 R6 7 fast rx up i2c4_scl + P9.20a 17 T9 7 fast rx up i2c4_sda + 18 T6 f fast down Driver off + 19 T7 f fast down Driver off + bluetooth in 20 P6 8 fast rx uart6_rxd mmc@480d1000 (wifibt_extra_pins_default) + bluetooth out 21 R9 8 fast rx uart6_txd mmc@480d1000 (wifibt_extra_pins_default) + ... + +The best way to use ``show-pins.pl`` is with ``grep``. To see all the pru pins try: + +.. code-block:: bash + + bone$ *show-pins.pl | grep -i pru | sort* + P8.13 100 D3 c fast rx pr1_pru1_gpi7 + P8.15b 109 A3 d fast down pr1_pru1_gpo16 + P8.16 111 B4 d fast down pr1_pru1_gpo18 + P8.18 98 F5 c fast rx pr1_pru1_gpi5 + P8.19 99 E6 c fast rx pr1_pru1_gpi6 + P8.26 110 B3 d fast down pr1_pru1_gpo17 + P9.16 108 C5 d fast down pr1_pru1_gpo15 + P9.19b 95 F4 c fast rx up pr1_pru1_gpi2 + P9.20b 94 D2 c fast rx up pr1_pru1_gpi1 + +Here we have nine pins configured for the PRU registers ``R30`` and ``R31``. +Five are input pins and four are out. + +.. _ai_device_tree: + +Configuring pins on the AI via device trees +============================================ + +Problem +-------- + +I want to configure another pin for the PRU, but I get an error. + +.. code-block:: bash + + bone$ *config-pin P9_31 pruout* + ERROR: open() for /sys/devices/platform/ocp/ocp:P9_31_pinmux/state failed, No such file or directory + +Solution +--------- + +The pins on the AI must be configure at boot time and therefor cannot be +configured with ``config-pin``. Instead you must edit the device tree. + +Discission +----------- + +Suppose you want to make ``P9_31`` a PRU output pin. First go to the +`am5729 System Reference Manual <https://github.com/beagleboard/beaglebone-ai/wiki/System-Reference-Manual#p8.10-p8.13>`_ +and look up ``P9_31``. + +.. tip:: + + The `BeagleBone AI PRU pins <https://docs.google.com/spreadsheets/d/1dFSBVem86vAUD7MLXvqdS-N0Efi8_g_O1iTqzql8DAo/edit#gid=0>`_ + table may be easier to use. + +``P9_31`` appears twice, as ``P9_31a`` and ``P9_31b``. Either should work, let's pick ``P9_31a``. + +.. warning:: + When you have two internal pins attached to the same header (either P8 or P9) + make sure only one is configured as an output. If both are outputs, you could + damage the AI. + +We see that when ``P9_31a`` is set to ``MODE13`` it will be a PRU **out** pin. +``MODE12`` makes it a PRU **in** pin. It appears at bit 10 on PRU2_1. + +Next, find which kernel you are running. + +.. code-block:: bash + bone$ uname -a + Linux ai 4.14.108-ti-r131 #1buster SMP PREEMPT Tue Mar 24 19:18:36 UTC 2020 armv7l GNU/Linux + +I'm running the 4.14 version. Now look in ``/opt/source`` for your kernel. + +.. code-block:: bash + + bone$ cd /opt/source/ + bone$ ls + adafruit-beaglebone-io-python dtb-5.4-ti rcpy + BBIOConfig librobotcontrol u-boot_v2019.04 + bb.org-overlays list.txt u-boot_v2019.07-rc4 + *dtb-4.14-ti* pyctrl + dtb-4.19-ti py-uio + +``am5729-beagleboneai.dts`` is the file we need to edit. Search for ``P9_31``. You'l see: + +.. code-block:: bash + :linenos: + + DRA7XX_CORE_IOPAD(0x36DC, MUX_MODE14) // B13: P9.30: mcasp1_axr10.off // + DRA7XX_CORE_IOPAD(0x36D4, *MUX_MODE13*) // B12: *P9.31a*: mcasp1_axr8.off // + DRA7XX_CORE_IOPAD(0x36A4, MUX_MODE14) // C14: P9.31b: mcasp1_aclkx.off // + +Change the ``MUX_MODE14`` to ``MUX_MODE13`` for output, or ``MUX_MODE12`` for input. + +Compile and install. The first time will take a while since it recompiles all the dts files. + +.. code-block:: bash + :linenos: + + bone$ make + ... + DTC src/arm/am335x-sl50.dtb + DTC src/arm/am5729-beagleboneai.dtb + DTC src/arm/am335x-nano.dtb + ... + bone$ sudo make install + ... + 'src/arm/am5729-beagleboneai.dtb' -> '/boot/dtbs/4.14.108-ti-r131/am5729-beagleboneai.dtb' + ... + bone$ reboot + ... + bone$ *show-pins.pl -v | sort | grep -i pru* + P8.13 100 D3 c fast rx pr1_pru1_gpi7 + P8.15b 109 A3 d fast down pr1_pru1_gpo16 + P8.16 111 B4 d fast down pr1_pru1_gpo18 + P8.18 98 F5 c fast rx pr1_pru1_gpi5 + P8.19 99 E6 c fast rx pr1_pru1_gpi6 + P8.26 110 B3 d fast down pr1_pru1_gpo17 + P9.16 108 C5 d fast down pr1_pru1_gpo15 + P9.19b 95 F4 c fast rx up pr1_pru1_gpi2 + P9.20b 94 D2 c fast rx up pr1_pru1_gpi1 + P9.31a 181 B12 d fast down pr2_pru1_gpo10 + + +There it is. `P9_31` is now a PRU output pin on PRU1_0, bit 3. + +.. _ai_using_pru_pins: + +Using the PRU pins +==================== + +Problem +-------- + +Once I have the PRU pins configured on the AI how do I use them? + +Solution +-------- + +In :ref:`ai_device_tree` we configured ``P9_31a`` to be a PRU pin. ``show-pins.pl`` showed +that it appears at ``pr2_pru1_gpo10``, which means pru2_1 accesses it using +bit 10 of register ``R30``. + +Discission +-------- + +It's easy to modify the pwm example from :ref:`blocks_pwm` to use this pin. +First copy the example you want to modify to ``pwm1.pru2_1.c``. The ``pru2_1`` in +the file name tells the Makefile to run the code on pru2_1. :ref:`ai_pwm1` shows +the adapted code. + +.. _ai_pwm1: + +pwm1.pru2_1.c +~~~~~~~~~~~~~~ + +:download:`pwm1.pru2_1.c <code/pwm1.pru2_1.c>` + + +One line 6 ``P9_31`` is defined as ``(0x1:ref:`10)``, which means shift ``1`` over by 10 bits. +That's the only change needed. Copy the local Makefile to the same directory and +compile and run. + +.. code-block:: bash + :linenos: + + bone$ make TARGET=pwm1.pru2_1 + +Attach an LED to ``P9_31`` and it should be blinking. diff --git a/pru-cookbook/08ai/code/Makefile b/pru-cookbook/08ai/code/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..a7557fdaa22988d89cec879477ded78522d7116f --- /dev/null +++ b/pru-cookbook/08ai/code/Makefile @@ -0,0 +1 @@ +include /var/lib/cloud9/common/Makefile diff --git a/pru-cookbook/08ai/code/pwm1.pru2_1.c b/pru-cookbook/08ai/code/pwm1.pru2_1.c new file mode 100644 index 0000000000000000000000000000000000000000..d35f2185cffd779cb1f18e252fe5252b803cd26e --- /dev/null +++ b/pru-cookbook/08ai/code/pwm1.pru2_1.c @@ -0,0 +1,24 @@ +#include <stdint.h> +#include <pru_cfg.h> +#include "resource_table_empty.h" +#include "prugpio.h" + +#define P9_31 (0x1<<10) + +volatile register uint32_t __R30; +volatile register uint32_t __R31; + +void main(void) +{ + uint32_t gpio = P9_31; // Select which pin to toggle.; + + /* Clear SYSCFG[STANDBY_INIT] to enable OCP master port */ + CT_CFG.SYSCFG_bit.STANDBY_INIT = 0; + + while(1) { + __R30 |= gpio; // Set the GPIO pin to 1 + __delay_cycles(100000000); + __R30 &= ~gpio; // Clear the GPIO pin + __delay_cycles(100000000); + } +} diff --git a/pru-cookbook/common/Makefile b/pru-cookbook/common/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..5b3999031af3d04540e6d4c8057624d1fe21664b --- /dev/null +++ b/pru-cookbook/common/Makefile @@ -0,0 +1,235 @@ +# +# Optional definitions +# +# TARGET - executable to create (currently supports single .c source) +# should have .<proc>(<n>) extension (lower case, check below for valid options) +# without TARGET, as of now, we don't have any build targets +# TODO: work with a list of targets, rather than a single file +# PRUN - TBD + +# +# Directories (external packages, etc.) +# +# COMMON points to where this primary Makefile and shell scripts live. +# PRU_CGT points to the TI PRU compiler directory. +# PRU_SUPPORT points to pru-software-support-package. +# C6X_CGT points to the TI C6X compiler directory. +# TIDL_API_DIR points to the TIDL API install directory. +# GEN_DIR points to where to build the binaries. +# MODEL is the board model (BeagleBoard.org_BeagleBone_AI, TI_AM335x_BeagleBone_Blue, etc.) +COMMON:=$(abspath $(lastword $(MAKEFILE_LIST)/..)) +PRU_CGT:=/usr/share/ti/cgt-pru +PRU_SUPPORT:=/usr/lib/ti/pru-software-support-package +PRU_STARTERWARE:=/usr/share/ti/starterware +C6X_CGT:=/usr/share/ti/cgt-c6x +TIDL_API_DIR:=/usr/share/ti/tidl/tidl_api +GEN_DIR:=/tmp/cloud9-examples +MODEL:=$(shell cat /proc/device-tree/model | sed 's/ /_/g' | tr -d '\000') +# $(warning MODEL=$(MODEL),TARGET=$(TARGET),COMMON=$(COMMON)) +$(warning MODEL=$(MODEL),TARGET=$(TARGET)) + +# +# Define PROC and build variables +# +# Arguments: +# $@ - target name +# +# Variables defined: +# CHIP - target system-on-chip (am57xx, am335x) +# CHIP_REV - target system-on-chip with rev (am572x_2_0, am335x) +# PROC - cpu type (arm, pru, c6x, etc.) +# PRUN - identifier for target PRU core +# PRU_DIR - control directory for the PRU +# CC - +# LD - +# LDFLAGS - +# CFLAGS - +# +# Depends on udev rules: https://github.com/beagleboard/customizations/blob/master/etc/udev/rules.d/86-remoteproc-noroot.rules +# Check which model +define target-to-proc = + ifeq ($(suffix $(basename $(1))),.pru1_0) + CHIP=am57xx + CHIP_REV=am572x_2_0 + PRU_DIR=/dev/remoteproc/pruss1-core0 + PRUN=1_0 + PROC=pru + EXE=.out + else ifeq ($(suffix $(basename $(1))),.pru1_1) + CHIP=am57xx + CHIP_REV=am572x_2_0 + PRU_DIR=/dev/remoteproc/pruss1-core1 + PRUN=1_1 + PROC=pru + EXE=.out + else ifeq ($(suffix $(basename $(1))),.pru2_0) + CHIP=am57xx + CHIP_REV=am572x_2_0 + PRU_DIR=/dev/remoteproc/pruss2-core0 + PRUN=2_0 + PROC=pru + EXE=.out + else ifeq ($(suffix $(basename $(1))),.pru2_1) + CHIP=am57xx + CHIP_REV=am572x_2_0 + PRU_DIR=/dev/remoteproc/pruss2-core1 + PRUN=2_1 + PROC=pru + EXE=.out + else ifeq ($(suffix $(basename $(1))),.pru0) + CHIP=am335x + CHIP_REV=am335x + PRU_DIR=/dev/remoteproc/pruss-core0 + PRUN=0 + PROC=pru + EXE=.out + else ifeq ($(suffix $(basename $(1))),.pru1) + CHIP=am335x + CHIP_REV=am335x + PRU_DIR=/dev/remoteproc/pruss-core1 + PRUN=1 + PROC=pru + EXE=.out + else ifeq ($(suffix $(basename $(1))),.tidl) + CHIP=am57xx + PROC=tidl + EXE=.so + else + PROC=arm + EXE=.out + endif +endef + +# +# Arguments: +# $@ - target name +# +# Variables required: +# CHIP - target system-on-chip (am57xx, am335x) +# CHIP_REV - target system-on-chip with rev (am572x_2_0, am335x) +# PROC - cpu type (arm, pru, c6x, etc.) +# PRUN - identifier for target PRU core +# PRU_DIR - control directory for the PRU +define proc-to-build-vars = + ifeq ($(PROC),pru) + LD=lnkpru -o + LDFLAGS=--reread_libs --warn_sections --stack_size=0x100 --heap_size=0x100 -m $(basename $(1)).map \ + -i$(PRU_CGT)/lib -i$(PRU_CGT)/include $(COMMON)/$(CHIP)_$(PROC).cmd --library=libc.a \ + --library=$(PRU_SUPPORT)/lib/rpmsg_lib.lib + CC=clpru -fe + CFLAGS=--include_path=$(COMMON) --include_path=$(PRU_SUPPORT)/include \ + --include_path=$(PRU_SUPPORT)/include/$(CHIP_REV) \ + --include_path=$(PRU_STARTERWARE)/include --include_path=$(PRU_STARTERWARE)/include/hw \ + --include_path=$(PRU_CGT)/include -DCHIP=$(CHIP) -DCHIP_IS_$(CHIP) -DMODEL=$(MODEL) -DPROC=$(PROC) -DPRUN=$(PRUN) \ + -v3 -O2 --printf_support=minimal --display_error_number --endian=little --hardware_mac=on \ + --obj_directory=$(GEN_DIR) --pp_directory=$(GEN_DIR) --asm_directory=$(GEN_DIR) -ppd -ppa --asm_listing \ + --c_src_interlist + else ifeq ($(PROC),tidl) + LD=g++ -o + LDFLAGS=-lopencv_highgui -lopencv_imgcodecs -lopencv_videoio -lopencv_imgproc -lopencv_core \ + $(TIDL_API_DIR)/tidl_api.a $(TIDL_API_DIR)/tidl_imgutil.a \ + -lTIOpenCL -locl_util -lpthread + CXX=g++ -c -o + CXXFLAGS=-DCHIP=$(CHIP) -DMODEL=$(MODEL) -DPROC=$(PROC) -fPIC -O3 -Wall -Werror -Wno-error=ignored-attributes \ + -I. -I$(TIDL_API_DIR)/inc -std=c++14 -I/usr/share/ti/opencl + else + LD=gcc -o + LDFLAGS=-lc -lm + CC=gcc -c -o + CFLAGS=-DCHIP=$(CHIP) -DMODEL=$(MODEL) -DPROC=$(PROC) + endif +endef + +# +# +# + +ifneq ($(TARGET),) +$(eval $(call target-to-proc,$(TARGET).out)) +# $(warning GEN_DIR=$(GEN_DIR),CHIP=$(CHIP),PROC=$(PROC),PRUN=$(PRUN),PRU_DIR=$(PRU_DIR),EXE=$(EXE)) + +all: stop install start + @echo "MODEL = $(MODEL)" + @echo "PROC = $(PROC)" + @echo "PRUN = $(PRUN)" + @echo "PRU_DIR = $(PRU_DIR)" + +# TODO: think about what we want to say if stop fails +stop: +ifneq ($(PRU_DIR),) + @echo "- Stopping PRU $(PRUN)" + @echo stop > $(PRU_DIR)/state || echo Cannot stop $(PRUN) +endif + +start: +ifneq ($(PRU_DIR),) + @echo write_init_pins.sh + @$(COMMON)/write_init_pins.sh /lib/firmware/$(CHIP)-pru$(PRUN)-fw + @echo "- Starting PRU $(PRUN)" + @echo start > $(PRU_DIR)/state +else ifeq ($(PROC),tidl) + ti-mct-heap-check -c + sudo mjpg_streamer -i "input_opencv.so -r 640x480 -d /dev/$(shell fgrep -v vpe /sys/class/video4linux/video*/name | perl -ne '/\/(video\d+)\/name/ && print $$1') \ + --filter ./$(TARGET)$(EXE)" -o "output_http.so -p 8090 -w /usr/share/mjpg-streamer/www" +else + ./$(TARGET)$(EXE) +endif + +install: $(GEN_DIR)/$(TARGET)$(EXE) +ifneq ($(PRU_DIR),) + @echo '- copying firmware file $(GEN_DIR)/$(TARGET).out to /lib/firmware/$(CHIP)-pru$(PRUN)-fw' + @cp $(GEN_DIR)/$(TARGET)$(EXE) /lib/firmware/$(CHIP)-pru$(PRUN)-fw +else + @cp $(GEN_DIR)/$(TARGET)$(EXE) ./$(TARGET)$(EXE) +endif +endif + +%.out: $(GEN_DIR)/%.out + @echo 'CP $^' + @cp $^ $@ + +%.so: $(GEN_DIR)/%.so + @echo 'CP $^' + @cp $^ $@ + +# TODO: look up how Makefiles typically break down the output file specification +# $(warning CHIP=$(CHIP),PROC=$(PROC)) +# $(warning CC=$(CC),LD=$(LD),CFLAGS=$(CFLAGS),LDFLAGS=$(LDFLAGS)) +$(GEN_DIR)/%.out: $(GEN_DIR)/%.o + @mkdir -p $(GEN_DIR) + @echo 'LD $^' + $(eval $(call target-to-proc,$@)) + $(eval $(call proc-to-build-vars,$@)) + @$(LD) $@ $^ $(LDFLAGS) + +$(GEN_DIR)/%.so: $(GEN_DIR)/%.o + @mkdir -p $(GEN_DIR) + @echo 'LD $^' + $(eval $(call target-to-proc,$@)) + $(eval $(call proc-to-build-vars,$@)) + @$(LD) $@ -shared $^ $(LDFLAGS) + +$(GEN_DIR)/%.o: %.asm + @mkdir -p $(GEN_DIR) + @echo 'CC $^' + $(eval $(call target-to-proc,$@)) + $(eval $(call proc-to-build-vars,$@)) + @$(CC) $@ $^ $(CFLAGS) + +$(GEN_DIR)/%.o: %.c + @mkdir -p $(GEN_DIR) + @echo 'CC $^' + $(eval $(call target-to-proc,$@)) + $(eval $(call proc-to-build-vars,$@)) + @$(CC) $@ $^ $(CFLAGS) + +$(GEN_DIR)/%.o: %.cpp + @mkdir -p $(GEN_DIR) + @echo 'CXX $^' + $(eval $(call target-to-proc,$@)) + $(eval $(call proc-to-build-vars,$@)) + @$(CXX) $@ $^ $(CXXFLAGS) + +clean: + @echo 'CLEAN' + @rm -rf $(GEN_DIR) diff --git a/pru-cookbook/common/README.md b/pru-cookbook/common/README.md new file mode 100644 index 0000000000000000000000000000000000000000..379ccc41af34ca13255e40b76c1b8461f5f78bea --- /dev/null +++ b/pru-cookbook/common/README.md @@ -0,0 +1,2 @@ +# common +These are the files used by all the platforms for running the PRUs. diff --git a/pru-cookbook/common/config-pin.sh b/pru-cookbook/common/config-pin.sh new file mode 100755 index 0000000000000000000000000000000000000000..682fecd697201463a0ab58d238aad25cfbffe760 --- /dev/null +++ b/pru-cookbook/common/config-pin.sh @@ -0,0 +1,39 @@ +#!/bin/bash +# Adds "in" and "out" to config-pin +# +# Usage: config-pin.sh P9_11 out + +pin=$1 # Name of pin to be changed +dir=$2 # Direction, either in or out + +if [ "$dir" != "in" ] && [ "$dir" != "out" ] ; then + # Use config-pin + /usr/bin/config-pin $* + exit +fi + +# Maps pin name (P9_11) to gpio number (gpio30) via libgpiod +# Find which gpio +chip=0 +found=0 + +# Check each gpio chip +while [ $chip -lt 4 ] ; do + # echo $chip + # Remove : from line number + line=$(gpioinfo $chip | grep $pin | awk '{ print substr($2, 1, length($2)-1) }') + + [ ! -z "$line" ] && found=1 && break + chip=$[$chip+1] +done + +# echo chip: $chip line: $line + +if [ $found -ne 0 ] ; then + gpio=gpio$[32*$chip+$line] + # echo $gpio + echo $dir > /sys/class/gpio/$gpio/direction + /usr/bin/config-pin $pin gpio +else + echo $pin not found +fi diff --git a/pru-cookbook/common/init_pins_empty.h b/pru-cookbook/common/init_pins_empty.h new file mode 100644 index 0000000000000000000000000000000000000000..9da8995e8c6ba25775f9748bc4366ce58d0e85c9 --- /dev/null +++ b/pru-cookbook/common/init_pins_empty.h @@ -0,0 +1,6 @@ +// Defins an empty table to write_init_pins wont' complain + +#pragma DATA_SECTION(init_pins, ".init_pins") +#pragma RETAIN(init_pins) +const char init_pins[] = + "\0\0"; diff --git a/pru-cookbook/common/prugpio.h b/pru-cookbook/common/prugpio.h new file mode 100644 index 0000000000000000000000000000000000000000..6d83e1d8a386442ea0f39895146b6fa040f3f0a6 --- /dev/null +++ b/pru-cookbook/common/prugpio.h @@ -0,0 +1,105 @@ +// This file defines the GPIO port addresses and PRU address + +#if defined(CHIP) && defined(CHIP_IS_am57xx) +#warning "Found AI" +// These are addresses for the am5729 +// First define the 9 GPIO port addresses +#define GPIO1 0x4AE10000 +#define GPIO2 0x48055000 +#define GPIO3 0x48057000 +#define GPIO4 0x48059000 +#define GPIO5 0x4805B000 +#define GPIO6 0x4805D000 +#define GPIO7 0x48051000 +#define GPIO8 0x48053000 + +// These define which bits in a given port go to the USR LEDs +// GPIO3 +#define USR0 (1<<17) +#define USR2 (1<<15) +#define USR3 (1<<14) +#define USR4 (1<< 7) +// GPIO5 +#define USR1 (1<< 5) +// The define a couple of GPIO pin addresses +// GPIO6 +#define P9_25 (1<<17) +// GPIO8 +#define P8_17 (1<18) + +// These define which bin in register R30 go to which header pins +// R30 bits - Output on pru1_1 +#define P9_14 (1<<14) +#define P9_16 (1<<15) +#define P8_15 (1<<16) +#define P8_26 (1<<17) +#define P8_16 (1<<18) +// R31 bits - Input on pru1_1 +#define P8_18 (1<<5) +#define P8_19 (1<<6) +#define P8_13 (1<<7) + +// Shared memory +#define AM33XX_DATARAM0_PHYS_BASE1 0x4b200000 +#define AM33XX_DATARAM1_PHYS_BASE1 0x4b202000 +#define AM33XX_PRUSS_SHAREDRAM_BASE1 0x4b210000 + +#define AM33XX_DATARAM0_PHYS_BASE2 0x4b280000 +#define AM33XX_DATARAM1_PHYS_BASE2 0x4b282000 +#define AM33XX_PRUSS_SHAREDRAM_BASE2 0x4b290000 + +#else + +#warning "Found am335x" +// These are addresses for the am35xx +#define GPIO0 0x44E07000 // From table 2.2 of am335x TRM +#define GPIO1 0x4804C000 +#define GPIO2 0x481AC000 +#define GPIO3 0x481AE000 + +// USR LED bit positions +// GPIO1 +#define USR0 (1<<21) +#define USR1 (1<<22) +#define USR2 (1<<23) +#define USR3 (1<<24) +// The define a couple of GPIO pin addresses on Black +// GPIO1 +#define P9_14 (1<<18) +#define P9_16 (1<<19) + +// The define a couple of GPIO pin addresses on Pocket +// GPIO1 +#define P2_1 (1<<18) +#define P1_32 (1<<10) + +// R30 output bits on pru0 +#define P9_31 (1<<0) +#define P9_29 (1<<1) +#define P9_30 (1<<2) +#define P9_28 (1<<3) +#define P9_92 (1<<4) +#define P9_27 (1<<5) +#define P9_91 (1<<6) +#define P9_25 (1<<7) + +// R30 output bits on pru0 on Pocket +#define P1_36 (1<<0) +#define P1_33 (1<<1) +#define P2_32 (1<<2) +#define P2_30 (1<<3) +#define P1_31 (1<<4) +#define P2_34 (1<<5) +#define P2_28 (1<<6) +#define P1_29 (1<<7) + +// Shared memory +#define AM33XX_DATARAM0_PHYS_BASE 0x4a300000 +#define AM33XX_DATARAM1_PHYS_BASE 0x4a302000 +#define AM33XX_PRUSS_SHAREDRAM_BASE 0x4a310000 + +#endif +// /4 to convert from byte address to word address +#define GPIO_CLEARDATAOUT 0x190/4 // Write 1 here to set a given bit +#define GPIO_SETDATAOUT 0x194/4 // A 1 here clears the corresponding bit +#define GPIO_DATAOUT 0x138/4 // For reading the GPIO registers diff --git a/pru-cookbook/common/resource_table_0.h b/pru-cookbook/common/resource_table_0.h new file mode 100644 index 0000000000000000000000000000000000000000..1c853b95775fd159d65ef12c71abfde1baff890e --- /dev/null +++ b/pru-cookbook/common/resource_table_0.h @@ -0,0 +1,136 @@ +/* + * Copyright (C) 2016 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#ifndef _RSC_TABLE_PRU_H_ +#define _RSC_TABLE_PRU_H_ + +#include <stddef.h> +#include <rsc_types.h> +#include "pru_virtio_ids.h" + +/* + * Sizes of the virtqueues (expressed in number of buffers supported, + * and must be power of 2) + */ +#define PRU_RPMSG_VQ0_SIZE 16 +#define PRU_RPMSG_VQ1_SIZE 16 + +/* + * The feature bitmap for virtio rpmsg + */ +#define VIRTIO_RPMSG_F_NS 0 //name service notifications + +/* This firmware supports name service notifications as one of its features */ +#define RPMSG_PRU_C0_FEATURES (1 << VIRTIO_RPMSG_F_NS) + +/* Definition for unused interrupts */ +#define HOST_UNUSED 255 + +/* Mapping sysevts to a channel. Each pair contains a sysevt, channel. */ +struct ch_map pru_intc_map[] = { + {16, 2}, + {17, 0}, +}; + +struct my_resource_table { + struct resource_table base; + + uint32_t offset[2]; /* Should match 'num' in actual definition */ + + /* rpmsg vdev entry */ + struct fw_rsc_vdev rpmsg_vdev; + struct fw_rsc_vdev_vring rpmsg_vring0; + struct fw_rsc_vdev_vring rpmsg_vring1; + + /* intc definition */ + struct fw_rsc_custom pru_ints; +}; + +#pragma DATA_SECTION(resourceTable, ".resource_table") +#pragma RETAIN(resourceTable) +struct my_resource_table resourceTable = { + 1, /* Resource table version: only version 1 is supported by the current driver */ + 2, /* number of entries in the table */ + 0, 0, /* reserved, must be zero */ + /* offsets to entries */ + { + offsetof(struct my_resource_table, rpmsg_vdev), + offsetof(struct my_resource_table, pru_ints), + }, + + /* rpmsg vdev entry */ + { + (uint32_t)TYPE_VDEV, //type + (uint32_t)VIRTIO_ID_RPMSG, //id + (uint32_t)0, //notifyid + (uint32_t)RPMSG_PRU_C0_FEATURES, //dfeatures + (uint32_t)0, //gfeatures + (uint32_t)0, //config_len + (uint8_t)0, //status + (uint8_t)2, //num_of_vrings, only two is supported + { (uint8_t)0, (uint8_t)0 }, //reserved + /* no config data */ + }, + /* the two vrings */ + { + 0, //da, will be populated by host, can't pass it in + 16, //align (bytes), + PRU_RPMSG_VQ0_SIZE, //num of descriptors + 0, //notifyid, will be populated, can't pass right now + 0 //reserved + }, + { + 0, //da, will be populated by host, can't pass it in + 16, //align (bytes), + PRU_RPMSG_VQ1_SIZE, //num of descriptors + 0, //notifyid, will be populated, can't pass right now + 0 //reserved + }, + + { + TYPE_CUSTOM, TYPE_PRU_INTS, + sizeof(struct fw_rsc_custom_ints), + { /* PRU_INTS version */ + 0x0000, + /* Channel-to-host mapping, 255 for unused */ + 0, HOST_UNUSED, 2, HOST_UNUSED, HOST_UNUSED, + HOST_UNUSED, HOST_UNUSED, HOST_UNUSED, HOST_UNUSED, HOST_UNUSED, + /* Number of evts being mapped to channels */ + (sizeof(pru_intc_map) / sizeof(struct ch_map)), + /* Pointer to the structure containing mapped events */ + pru_intc_map, + }, + }, +}; + +#endif /* _RSC_TABLE_PRU_H_ */ diff --git a/pru-cookbook/common/resource_table_1.h b/pru-cookbook/common/resource_table_1.h new file mode 100644 index 0000000000000000000000000000000000000000..b062c71f3e66559102cc8f65f9b8d8274b144aef --- /dev/null +++ b/pru-cookbook/common/resource_table_1.h @@ -0,0 +1,135 @@ +/* + * Copyright (C) 2016 Texas Instruments Incorporated - http://www.ti.com/ + * + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the + * distribution. + * + * * Neither the name of Texas Instruments Incorporated nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR + * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT + * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT + * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, + * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY + * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +#ifndef _RSC_TABLE_PRU_H_ +#define _RSC_TABLE_PRU_H_ + +#include <stddef.h> +#include <rsc_types.h> +#include "pru_virtio_ids.h" + +/* + * Sizes of the virtqueues (expressed in number of buffers supported, + * and must be power of 2) + */ +#define PRU_RPMSG_VQ0_SIZE 16 +#define PRU_RPMSG_VQ1_SIZE 16 + +/* + * The feature bitmap for virtio rpmsg + */ +#define VIRTIO_RPMSG_F_NS 0 //name service notifications + +/* This firmware supports name service notifications as one of its features */ +#define RPMSG_PRU_C0_FEATURES (1 << VIRTIO_RPMSG_F_NS) + +/* Definition for unused interrupts */ +#define HOST_UNUSED 255 + +/* Mapping sysevts to a channel. Each pair contains a sysevt, channel. */ +struct ch_map pru_intc_map[] = { {18, 3}, + {19, 1}, +}; + +struct my_resource_table { + struct resource_table base; + + uint32_t offset[2]; /* Should match 'num' in actual definition */ + + /* rpmsg vdev entry */ + struct fw_rsc_vdev rpmsg_vdev; + struct fw_rsc_vdev_vring rpmsg_vring0; + struct fw_rsc_vdev_vring rpmsg_vring1; + + /* intc definition */ + struct fw_rsc_custom pru_ints; +}; + +#pragma DATA_SECTION(resourceTable, ".resource_table") +#pragma RETAIN(resourceTable) +struct my_resource_table resourceTable = { + 1, /* Resource table version: only version 1 is supported by the current driver */ + 2, /* number of entries in the table */ + 0, 0, /* reserved, must be zero */ + /* offsets to entries */ + { + offsetof(struct my_resource_table, rpmsg_vdev), + offsetof(struct my_resource_table, pru_ints), + }, + + /* rpmsg vdev entry */ + { + (uint32_t)TYPE_VDEV, //type + (uint32_t)VIRTIO_ID_RPMSG, //id + (uint32_t)0, //notifyid + (uint32_t)RPMSG_PRU_C0_FEATURES, //dfeatures + (uint32_t)0, //gfeatures + (uint32_t)0, //config_len + (uint8_t)0, //status + (uint8_t)2, //num_of_vrings, only two is supported + { (uint8_t)0, (uint8_t)0 }, //reserved + /* no config data */ + }, + /* the two vrings */ + { + 0, //da, will be populated by host, can't pass it in + 16, //align (bytes), + PRU_RPMSG_VQ0_SIZE, //num of descriptors + 0, //notifyid, will be populated, can't pass right now + 0 //reserved + }, + { + 0, //da, will be populated by host, can't pass it in + 16, //align (bytes), + PRU_RPMSG_VQ1_SIZE, //num of descriptors + 0, //notifyid, will be populated, can't pass right now + 0 //reserved + }, + + { + TYPE_CUSTOM, TYPE_PRU_INTS, + sizeof(struct fw_rsc_custom_ints), + { /* PRU_INTS version */ + 0x0000, + /* Channel-to-host mapping, 255 for unused */ + HOST_UNUSED, 1, HOST_UNUSED, 3, HOST_UNUSED, + HOST_UNUSED, HOST_UNUSED, HOST_UNUSED, HOST_UNUSED, HOST_UNUSED, + /* Number of evts being mapped to channels */ + (sizeof(pru_intc_map) / sizeof(struct ch_map)), + /* Pointer to the structure containing mapped events */ + pru_intc_map, + }, + }, +}; + +#endif /* _RSC_TABLE_PRU_H_ */ diff --git a/pru-cookbook/common/resource_table_empty.h b/pru-cookbook/common/resource_table_empty.h new file mode 100644 index 0000000000000000000000000000000000000000..07e97d9b383c9d937d4b55e705024d6c28dc748a --- /dev/null +++ b/pru-cookbook/common/resource_table_empty.h @@ -0,0 +1,39 @@ +/* + * ======== resource_table_empty.h ======== + * + * Define the resource table entries for all PRU cores. This will be + * incorporated into corresponding base images, and used by the remoteproc + * on the host-side to allocated/reserve resources. Note the remoteproc + * driver requires that all PRU firmware be built with a resource table. + * + * This file contains an empty resource table. It can be used either as: + * + * 1) A template, or + * 2) As-is if a PRU application does not need to configure PRU_INTC + * or interact with the rpmsg driver + * + */ + +#ifndef _RSC_TABLE_PRU_H_ +#define _RSC_TABLE_PRU_H_ + +#include <stddef.h> +#include <rsc_types.h> + +struct my_resource_table { + struct resource_table base; + + uint32_t offset[1]; /* Should match 'num' in actual definition */ +}; + +#pragma DATA_SECTION(pru_remoteproc_ResourceTable, ".resource_table") +#pragma RETAIN(pru_remoteproc_ResourceTable) +struct my_resource_table pru_remoteproc_ResourceTable = { + 1, /* we're the first version that implements this */ + 0, /* number of entries in the table */ + 0, 0, /* reserved, must be zero */ + 0, /* offset[0] */ +}; + +#endif /* _RSC_TABLE_PRU_H_ */ + diff --git a/pru-cookbook/common/write_init_pins.sh b/pru-cookbook/common/write_init_pins.sh new file mode 100755 index 0000000000000000000000000000000000000000..f52082365a8b2a72a6c30d39938077bd9addf741 --- /dev/null +++ b/pru-cookbook/common/write_init_pins.sh @@ -0,0 +1,9 @@ +#!/bin/bash +init_pins=$(readelf -x .init_pins $1 | grep 0x000 | cut -d' ' -f4-7 | xxd -r -p | tr '\0' '\n' | paste - -) +while read -a line; do + if [ ${#line[@]} == 2 ]; then + echo writing \"${line[1]}\" to \"${line[0]}\" + echo ${line[1]} > ${line[0]} + sleep 0.1 + fi +done <<< "$init_pins" diff --git a/pru-cookbook/index.rst b/pru-cookbook/index.rst new file mode 100644 index 0000000000000000000000000000000000000000..981f2fb27405e0a3f32ceceab1cd7de890745413 --- /dev/null +++ b/pru-cookbook/index.rst @@ -0,0 +1,27 @@ +.. _pru-cookbook-home: + +PRU Cookbook +############# + +.. admonition:: Contributors + + - Author: `Mark A. Yoder <Mark.A.Yoder@Rose-Hulman.edu>`_ + - Book revision: v2.0 beta + +Outline + +A cookbook for programming the PRUs in C using remoteproc and +compiling on the Beagle + +.. toctree:: + :maxdepth: 1 + + 01case/case.rst + 02start/start.rst + 03details/details.rst + 04debug/debug.rst + 05blocks/blocks.rst + 06io/io.rst + 07more/more.rst + 08ai/ai.rst + projects.rst \ No newline at end of file diff --git a/pru-cookbook/projects.rst b/pru-cookbook/projects.rst new file mode 100644 index 0000000000000000000000000000000000000000..39263b91e03c611604c1e60d416f92e1db44e1b3 --- /dev/null +++ b/pru-cookbook/projects.rst @@ -0,0 +1,311 @@ +.. _pru-cookbook-projects: + +PRU Projects +############## + +Users of TI processors with PRU-ICSS have created application for many different uses. +A list of a few are shared below. For additional support resources, software and +documentation visit the PRU-ICSS wiki. + +PRU projects +~~~~~~~~~~~~~~ + +.. dropdown:: **LEDscape** + :open: + + **Description:** BeagleBone Black cape and firmware for driving a large number of WS281x LED strips. + + **Type:** Code Library Documentation and example projects. + + **References:** + * https://github.com/osresearch/LEDscape http://trmm.net/LEDscape + +.. dropdown:: **LDGraphy** + :open: + + **Description:** Laser direct lithography for printing PCBs. + + **Type:** Code Library and example project. + + **References:** + * https://github.com/hzeller/ldgraphy/blob/master/README.md + +.. dropdown:: **PRdUino** + :open: + + **Description:** This is a port of the Energia platform based on the Arduino framework allowing you to use Arduino software libraries on PRU. + + **Type:** Code Library + + **References:** + * https://github.com/lucas-ti/PRdUino + +.. dropdown:: **DMX Lighting** + :open: + + **Description:** Controlling professional lighting systems + + **Type:** Project Tutorial Code Library + + **References:** + * http://beagleboard.org/CapeContest/entries/BeagleBone+DMX+Cape/ + * http://blog.boxysean.com/2012/08/12/first-steps-with-the-beaglebone-pru/ + * https://github.com/boxysean/beaglebone-DMX + +.. dropdown:: **Interacto** + :open: + + **Description:** A cape making BeagleBone interactive with a triple-axis accelerometer, gyroscope + and magnetometer plus a 640 x 480/30 fps camera. All sensors are digital and communicate via I²C + to the BeagleBone. The camera frames are captured using the PRU-ICSS. The sensors on this cape + give hobbyists and students a starting point to easily build robots and flying drones. + + **Type:** Project 1 Project 2 Code Library + + **References:** + * http://beagleboard.org/CapeContest/entries/Interacto/ + * http://www.hitchhikeree.org/beaglebone_capes/interacto/ + * https://github.com/cclark2/interacto_bbone_cape + +.. dropdown:: **Replicape: 3D Printer** + :open: + + **Description:** Replicape is a high end 3D-printer electronics package in the form of a Cape + that can be placed on a BeagleBone Black. It has five high power stepper motors with cool running + MosFets and it has been designed to fit in small spaces without active cooling. For a Replicape + Daemon that processes G-code, see the Redeem Project + + **Type:** Project Code Library + + **References:** + * http://www.thing-printer.com/product/replicape/ + * https://bitbucket.org/intelligentagent/replicape/ + +.. dropdown:: **PyPRUSS: Python Library** + :open: + + **Description:** PyPRUSS is a Python library for programming the PRUs on BeagleBone (Black) + + **Type:** Code Library + + **References:** + http://hipstercircuits.com/pypruss-a-simple-pru-python-binding-for-beaglebone/ + +.. dropdown:: **Geiger** + :open: + + **Description:** The Geiger Cape, created by Matt Ranostay, is a design that measures radiation counts + from background and test sources by utilising multiple Geiger tubes. The cape can be used to detect + low-level radiation, which is needed in certain industries such as security and medical. + + **Type:** Project 1 Project 2 Code Library + + **References:** + * http://beagleboard.org/CapeContest/entries/Geiger+Cape/ + * http://elinux.org/BeagleBone/GeigerCapePrototype + * https://github.com/mranostay/beaglebone-telemetry-presentation + +.. dropdown:: **Servo Controller Foosball Table** + :open: + + **Description:** Used for ball tracking and motor control + + **Type:** Project Tutorial Code Library + + **References:** + * http://www.element14.com/community/community/knode/single-board_computers/next-gen_beaglebone/blog/2013/07/17/hackerspace-challenge--leeds-only-pru-can-make-the-leds-bright + * https://docs.google.com/spreadsheet/pub?key=0AmI_ryMKXUGJdDQ3LXB4X3VBWlpxQTFWbGh6RGJHUEE&output=html + * https://github.com/pbrook/pypruss + +.. dropdown:: **Imaging with connected camera** + :open: + + **Description:** Low resolution imaging ideal for machine vision use-cases, robotics and movement detection + + **Type:** Project Code Library + + **References:** + * http://www.element14.com/community/community/knode/single-board_computers/next-gen_beaglebone/blog/2013/08/18/bbb--imaging-with-a-pru-connected-camera + +.. dropdown:: **Computer Numerical Control (CNC) Translator** + :open: + + **Description:** Smooth stepper motor control; real embedded version of LinuxCNC + + **Type:** Tutorial Tutorial + + **References:** + * http://www.buildlog.net/blog/2013/09/cnc-translator-for-beaglebone/ http://bb-lcnc.blogspot.com/p/machinekit_16.html + +.. dropdown:: **Robotic Control** + :open: + + **Description:** Chubby SpiderBot + + **Type:** Project Code Library Project Reference + + **References:** + * http://www.youtube.com/watch?v=dEes9k7-DYY + * https://github.com/cagdasc/Chubby1_v1 + * http://www.youtube.com/watch?v=JXyewd98e9Q + * http://www.ti.com/lit/wp/spry235/spry235.pdf + +.. dropdown:: **Software UART** + :open: + + **Description:** Soft-UART implementation on the PRU of AM335x + + **Type:** Code Library Reference + + **References:** + * http://processors.wiki.ti.com/index.php/Soft-UART_Implementation_on_AM335X_PRU_-_Software_Users_Guide + +.. dropdown:: **Deviant LCD** + :open: + + **Description:** PRU bit-banged LCD interface @ 240x320 + + **Type:** Project Code Library + + **References:** + * http://www.beagleboard.org/CapeContest/entries/DeviantLCD/ + * https://github.com/cclark2/deviantlcd_bbone_cape + +.. dropdown:: **Nixie tube interface** + :open: + + **Description:** + + **Type:** Code Library + + **References:** + * https://github.com/mranostay/beagle-nixie + +.. dropdown:: **Thermal imaging camera** + :open: + + **Description:** Thermal camera using Beaglebone Black, a small LCD, and a thermal array sensor + + **Type:** Project Code Library + + **References:** + * https://element14.com/community/community/knode/single-board_computers/next-gen_beaglebone/blog/2013/06/07/bbb--building-a-thermal-imaging-camera + +.. dropdown:: **Sine wave generator using PWMs** + :open: + + **Description:** Simulation of a pulse width modulation + + **Type:** Project Reference Code Library + + **References:** + * http://elinux.org/ECE497_BeagleBone_PRU + * https://github.com/millerap/AM335x_PRU_BeagleBone + +.. dropdown:: **Emulated memory interface** + :open: + + **Description:** ABX loads amovie into the Beaglebone's memory and then launches the memory emulator + on the PRU sub-processor of the Beaglebone's ARM AM335x + + **Type:** Project + + **References:** + * https://github.com/lybrown/abx + +.. dropdown:: **6502 memory interface** + :open: + + **Description:** System permitting communication between Linux and 6502 processor + + **Type:** Project Code Library + + **References:** + * http://elinux.org/images/a/ac/What's_Old_Is_New-_A_6502-based_Remote_Processor.pdf + * https://github.com/lybrown/abx + +.. dropdown:: **JTAG/Debug** + :open: + + **Description:** Investigating the fastest way to program using JTAG and provide + for debugging facilities built into the Beaglebone. + + **Type:** Project + + **References:** + * http://beagleboard.org/project/PRUJTAG/ + +.. dropdown:: **High Speed Data Acquistion** + :open: + + **Description:** Reading data at high speeds + + **Type:** Reference + + **References:** + * http://www.element14.com/community/community/knode/single-board_computers/next-gen_beaglebone/blog/2013/08/04/bbb--high-speed-data-acquisition-and-web-based-ui + +.. dropdown:: **Prufh (PRU Forth)** + :open: + + **Description:** Forth Programming Language and Compiler. It consists of a compiler, + the forth system itself, and anoptional program for loading and communicating with the forth code proper. + + **Type:** Compiler + + **References:** + * https://github.com/biocode3D/prufh + +.. dropdown:: **VisualPRU** + :open: + + **Description:** VisualPRU is a minimal browser-based editor and debugger for the Beaglebone PRUs. + The app runs from a local server on the Beaglebone. + + **Type:** Editor and Debugger + + **References:** + * https://github.com/mmcdan/visualpru + +.. dropdown:: **libpruio** + :open: + + **Description:** Library for easy configuration and data handling at high speeds. + This library can configure and control the devices from single source (no need for + further overlays or the device tree compiler) + + **Type:** Documentation + + **References:** + * http://users.freebasic-portal.de/tjf/Projekte/libpruio/doc/html/index.html + * Library http://www.freebasic-portal.de/downloads/fb-on-arm/libpruio-325.html[(German)] + +.. dropdown:: **BeagleLogic** + :open: + + **Description:** 100MHz 14channel logic analyzer using both PRUs (one to capture and one to transfer the data) + + **Type:** Project + + **References:** + * http://beaglelogic.net + +.. dropdown:: **BeaglePilot** + :open: + + **Description:** Uses PRUs as part of code for a BeagleBone based autopilot + + **Type:** Code Library + + **References:** + * https://github.com/BeaglePilot/beaglepilot + +.. dropdown:: **PRU Speak** + :open: + + **Description:** Implements BotSpeak, a platform independent interpreter for tools like Labview, on the PRUs + + **Type:** Code Library + + **References:** + * https://github.com/deepakkarki/pruspeak \ No newline at end of file