|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 |
- /* Teensyduino Core Library
- * http://www.pjrc.com/teensy/
- * Copyright (c) 2019 PJRC.COM, LLC.
- *
- * Permission is hereby granted, free of charge, to any person obtaining
- * a copy of this software and associated documentation files (the
- * "Software"), to deal in the Software without restriction, including
- * without limitation the rights to use, copy, modify, merge, publish,
- * distribute, sublicense, and/or sell copies of the Software, and to
- * permit persons to whom the Software is furnished to do so, subject to
- * the following conditions:
- *
- * 1. The above copyright notice and this permission notice shall be
- * included in all copies or substantial portions of the Software.
- *
- * 2. If the Software is incorporated into a build system that allows
- * selection among a list of target devices, then similar target
- * devices manufactured by PJRC.COM must be included in the list of
- * target devices and selectable in the same manner.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
- * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
- * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
- * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- * SOFTWARE.
- */
-
- #include "usb_dev.h"
- #include "usb_touch.h"
- #include "core_pins.h" // for millis()
- #include <string.h> // for memcpy()
- #include "avr/pgmspace.h" // for PROGMEM, DMAMEM, FASTRUN
- #include "debug/printf.h"
- #include "core_pins.h"
-
-
- #ifdef MULTITOUCH_INTERFACE // defined by usb_dev.h -> usb_desc.h
-
- static uint8_t prev_id=0;
- static uint8_t scan_index=0;
- static uint8_t scan_count=0;
- static uint8_t contactid[MULTITOUCH_FINGERS];
- static uint8_t pressure[MULTITOUCH_FINGERS];
- static uint16_t xpos[MULTITOUCH_FINGERS];
- static uint16_t ypos[MULTITOUCH_FINGERS];
- static uint16_t scan_timestamp;
-
- #define TX_NUM 20
- #define TX_BUFSIZE 32
- static transfer_t tx_transfer[TX_NUM] __attribute__ ((used, aligned(32)));
- DMAMEM static uint8_t txbuffer[TX_NUM * TX_BUFSIZE] __attribute__ ((aligned(32)));
- static uint8_t tx_head=0;
- #if MULTITOUCH_SIZE > TX_BUFSIZE
- #error "Internal error, transmit buffer size is too small for touchscreen endpoint"
- #endif
-
-
- void usb_touchscreen_configure(void)
- {
- memset(tx_transfer, 0, sizeof(tx_transfer));
- tx_head = 0;
- usb_config_tx(MULTITOUCH_ENDPOINT, MULTITOUCH_SIZE, 0, NULL);
- // TODO: need to cause usb_touchscreen_update_callback to be run on SOF
- }
-
-
- void usb_touchscreen_press(uint8_t finger, uint32_t x, uint32_t y, uint32_t press)
- {
- if (finger >= MULTITOUCH_FINGERS) return;
- if (press > 255) press = 255;
- if (x > 32767) x = 32767;
- if (y > 32767) y = 32767;
- __disable_irq();
- if (pressure[finger] == 0 && press > 0) {
- // Begin a new finger touch, assign a
- // new Contact Identifier (usage 0x51)
- uint8_t id = prev_id + 1;
- if (id == 0 || id > 127) id = 1;
- prev_id = id;
- contactid[finger] = (id << 1) | 1;
- }
- xpos[finger] = x;
- ypos[finger] = y;
- pressure[finger] = press;
- __enable_irq();
- }
-
- // User should allow at least 12 ms after release before starting a
- // new touch with press, in case the host's driver does not properly
- // parse a change in contact identifier to mean a new touch.
-
- void usb_touchscreen_release(uint8_t finger)
- {
- if (finger >= MULTITOUCH_FINGERS) return;
- pressure[finger] = 0;
- }
-
-
- static int transmit(const uint8_t *data)
- {
- if (!usb_configuration) return 0;
- uint32_t head = tx_head;
- transfer_t *xfer = tx_transfer + head;
- uint32_t status = usb_transfer_status(xfer);
- if ((status & 0x80)) return 0;
- if (status & 0x68) {
- // TODO: what if status has errors???
- }
- delayNanoseconds(30); // TODO: is this needed? (probably not)
- uint8_t *buffer = txbuffer + head * TX_BUFSIZE;
- memcpy(buffer, data, MULTITOUCH_SIZE);
- usb_prepare_transfer(xfer, buffer, MULTITOUCH_SIZE, 0);
- arm_dcache_flush_delete(buffer, TX_BUFSIZE);
- usb_transmit(MULTITOUCH_ENDPOINT, xfer);
- if (++head >= TX_NUM) head = 0;
- tx_head = head;
- return 1;
- }
-
- // touch report
- // 0: contact id + on/off
- // 1: pressure
- // 2: X lsb
- // 3: X msb
- // 4: Y lsb
- // 5: Y msb
- // 6: scan time lsb
- // 7: scan time msb
-
-
- // Called by the start-of-frame interrupt.
- //
- void usb_touchscreen_update_callback(void)
- {
- // TODO: if 480 speed, run only every 8th time
-
- if (scan_index == 0) {
- if (usb_tx_packet_count(MULTITOUCH_ENDPOINT) > 1) {
- // wait to begin another scan if if more than
- // one prior packet remains to transmit
- return;
- }
- scan_timestamp = millis() * 10;
- scan_count = 0;
- }
- while (scan_index < MULTITOUCH_FINGERS) {
- uint32_t press = pressure[scan_index];
- uint32_t id = contactid[scan_index];
- if (id) {
- if (press == 0) {
- // End a finger touch. One final packet sent
- // with same Contact Identifier (usage 0x51)
- // and Tip Switch (usage 0x42) = zero
- id &= 0xFE;
- contactid[scan_index] = 0;
- }
- uint8_t buffer[MULTITOUCH_SIZE]; // MULTITOUCH_SIZE = 8
- buffer[0] = id;
- buffer[1] = press;
- buffer[2] = xpos[scan_index];
- buffer[3] = xpos[scan_index] >> 8;
- buffer[4] = ypos[scan_index];
- buffer[5] = ypos[scan_index] >> 8;
- buffer[6] = scan_timestamp;
- buffer[7] = scan_timestamp >> 8;
- if (transmit(buffer)) scan_index++;
- return;
- }
- scan_index++;
- }
- if (++scan_count >= MULTITOUCH_FINGERS) {
- // when done early, wait to restart another scan
- scan_index = 0;
- }
- }
-
-
- #endif // MULTITOUCH_INTERFACE
|