|  | /* | 
|  | ** Copyright (C) 2015 The Android Open Source Project | 
|  | ** | 
|  | ** Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | ** you may not use this file except in compliance with the License. | 
|  | ** You may obtain a copy of the License at | 
|  | ** | 
|  | **      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | ** | 
|  | ** Unless required by applicable law or agreed to in writing, software | 
|  | ** distributed under the License is distributed on an "AS IS" BASIS, | 
|  | ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | ** See the License for the specific language governing permissions and | 
|  | ** limitations under the License. | 
|  | **/ | 
|  | /* | 
|  | * This file was copied from https://github.com/devttys0/libmpsse.git (sha1 | 
|  | * f1a6744b), and modified to suite the Chromium OS project. | 
|  | * | 
|  | * Internal functions used by libmpsse. | 
|  | * | 
|  | * Craig Heffner | 
|  | * 27 December 2011 | 
|  | */ | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  |  | 
|  | #include "support.h" | 
|  |  | 
|  | /* Write data to the FTDI chip */ | 
|  | int raw_write(struct mpsse_context* mpsse, uint8_t* buf, int size) { | 
|  | int retval = MPSSE_FAIL; | 
|  |  | 
|  | if (mpsse->mode) { | 
|  | if (ftdi_write_data(&mpsse->ftdi, buf, size) == size) { | 
|  | retval = MPSSE_OK; | 
|  | } | 
|  | } | 
|  |  | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | /* Read data from the FTDI chip */ | 
|  | int raw_read(struct mpsse_context* mpsse, uint8_t* buf, int size) { | 
|  | int n = 0, r = 0; | 
|  |  | 
|  | if (mpsse->mode) { | 
|  | while (n < size) { | 
|  | r = ftdi_read_data(&mpsse->ftdi, buf, size); | 
|  | if (r < 0) | 
|  | break; | 
|  | n += r; | 
|  | } | 
|  |  | 
|  | if (mpsse->flush_after_read) { | 
|  | /* | 
|  | * Make sure the buffers are cleared after a read or subsequent reads may | 
|  | *fail. | 
|  | * | 
|  | * Is this needed anymore? It slows down repetitive read operations by | 
|  | *~8%. | 
|  | */ | 
|  | ftdi_usb_purge_rx_buffer(&mpsse->ftdi); | 
|  | } | 
|  | } | 
|  |  | 
|  | return n; | 
|  | } | 
|  |  | 
|  | /* Sets the read and write timeout periods for bulk usb data transfers. */ | 
|  | void set_timeouts(struct mpsse_context* mpsse, int timeout) { | 
|  | if (mpsse->mode) { | 
|  | mpsse->ftdi.usb_read_timeout = timeout; | 
|  | mpsse->ftdi.usb_write_timeout = timeout; | 
|  | } | 
|  |  | 
|  | return; | 
|  | } | 
|  |  | 
|  | /* Convert a frequency to a clock divisor */ | 
|  | uint16_t freq2div(uint32_t system_clock, uint32_t freq) { | 
|  | return (((system_clock / freq) / 2) - 1); | 
|  | } | 
|  |  | 
|  | /* Convert a clock divisor to a frequency */ | 
|  | uint32_t div2freq(uint32_t system_clock, uint16_t div) { | 
|  | return (system_clock / ((1 + div) * 2)); | 
|  | } | 
|  |  | 
|  | /* Builds a buffer of commands + data blocks */ | 
|  | uint8_t* build_block_buffer(struct mpsse_context* mpsse, | 
|  | uint8_t cmd, | 
|  | const uint8_t* data, | 
|  | int size, | 
|  | int* buf_size) { | 
|  | uint8_t* buf = NULL; | 
|  | int i = 0, j = 0, k = 0, dsize = 0, num_blocks = 0, total_size = 0, | 
|  | xfer_size = 0; | 
|  | uint16_t rsize = 0; | 
|  |  | 
|  | *buf_size = 0; | 
|  |  | 
|  | /* Data block size is 1 in I2C, or when in bitmode */ | 
|  | if (mpsse->mode == I2C || (cmd & MPSSE_BITMODE)) { | 
|  | xfer_size = 1; | 
|  | } else { | 
|  | xfer_size = mpsse->xsize; | 
|  | } | 
|  |  | 
|  | num_blocks = (size / xfer_size); | 
|  | if (size % xfer_size) { | 
|  | num_blocks++; | 
|  | } | 
|  |  | 
|  | /* The total size of the data will be the data size + the write command */ | 
|  | total_size = size + (CMD_SIZE * num_blocks); | 
|  |  | 
|  | /* In I2C we have to add 3 additional commands per data block */ | 
|  | if (mpsse->mode == I2C) { | 
|  | total_size += (CMD_SIZE * 3 * num_blocks); | 
|  | } | 
|  |  | 
|  | buf = malloc(total_size); | 
|  | if (buf) { | 
|  | memset(buf, 0, total_size); | 
|  |  | 
|  | for (j = 0; j < num_blocks; j++) { | 
|  | dsize = size - k; | 
|  | if (dsize > xfer_size) { | 
|  | dsize = xfer_size; | 
|  | } | 
|  |  | 
|  | /* The reported size of this block is block size - 1 */ | 
|  | rsize = dsize - 1; | 
|  |  | 
|  | /* For I2C we need to ensure that the clock pin is set low prior to | 
|  | * clocking out data */ | 
|  | if (mpsse->mode == I2C) { | 
|  | buf[i++] = SET_BITS_LOW; | 
|  | buf[i++] = mpsse->pstart & ~SK; | 
|  |  | 
|  | /* On receive, we need to ensure that the data out line is set as an | 
|  | * input to avoid contention on the bus */ | 
|  | if (cmd == mpsse->rx) { | 
|  | buf[i++] = mpsse->tris & ~DO; | 
|  | } else { | 
|  | buf[i++] = mpsse->tris; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Copy in the command for this block */ | 
|  | buf[i++] = cmd; | 
|  | buf[i++] = (rsize & 0xFF); | 
|  | if (!(cmd & MPSSE_BITMODE)) { | 
|  | buf[i++] = ((rsize >> 8) & 0xFF); | 
|  | } | 
|  |  | 
|  | /* On a write, copy the data to transmit after the command */ | 
|  | if (cmd == mpsse->tx || cmd == mpsse->txrx) { | 
|  | memcpy(buf + i, data + k, dsize); | 
|  |  | 
|  | /* i == offset into buf */ | 
|  | i += dsize; | 
|  | /* k == offset into data */ | 
|  | k += dsize; | 
|  | } | 
|  |  | 
|  | /* In I2C mode we need to clock one ACK bit after each byte */ | 
|  | if (mpsse->mode == I2C) { | 
|  | /* If we are receiving data, then we need to clock out an ACK for each | 
|  | * byte */ | 
|  | if (cmd == mpsse->rx) { | 
|  | buf[i++] = SET_BITS_LOW; | 
|  | buf[i++] = mpsse->pstart & ~SK; | 
|  | buf[i++] = mpsse->tris; | 
|  |  | 
|  | buf[i++] = mpsse->tx | MPSSE_BITMODE; | 
|  | buf[i++] = 0; | 
|  | buf[i++] = mpsse->tack; | 
|  | } | 
|  | /* If we are sending data, then we need to clock in an ACK for each | 
|  | * byte | 
|  | */ | 
|  | else if (cmd == mpsse->tx) { | 
|  | /* Need to make data out an input to avoid contention on the bus when | 
|  | * the slave sends an ACK */ | 
|  | buf[i++] = SET_BITS_LOW; | 
|  | buf[i++] = mpsse->pstart & ~SK; | 
|  | buf[i++] = mpsse->tris & ~DO; | 
|  |  | 
|  | buf[i++] = mpsse->rx | MPSSE_BITMODE; | 
|  | buf[i++] = 0; | 
|  | buf[i++] = SEND_IMMEDIATE; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | *buf_size = i; | 
|  | } | 
|  |  | 
|  | return buf; | 
|  | } | 
|  |  | 
|  | /* Set the low bit pins high/low */ | 
|  | int set_bits_low(struct mpsse_context* mpsse, int port) { | 
|  | char buf[CMD_SIZE] = {0}; | 
|  |  | 
|  | buf[0] = SET_BITS_LOW; | 
|  | buf[1] = port; | 
|  | buf[2] = mpsse->tris; | 
|  |  | 
|  | return raw_write(mpsse, (uint8_t*)&buf, sizeof(buf)); | 
|  | } | 
|  |  | 
|  | /* Set the high bit pins high/low */ | 
|  | int set_bits_high(struct mpsse_context* mpsse, int port) { | 
|  | char buf[CMD_SIZE] = {0}; | 
|  |  | 
|  | buf[0] = SET_BITS_HIGH; | 
|  | buf[1] = port; | 
|  | buf[2] = mpsse->trish; | 
|  |  | 
|  | return raw_write(mpsse, (uint8_t*)&buf, sizeof(buf)); | 
|  | } | 
|  |  | 
|  | /* Set the GPIO pins high/low */ | 
|  | int gpio_write(struct mpsse_context* mpsse, int pin, int direction) { | 
|  | int retval = MPSSE_FAIL; | 
|  |  | 
|  | if (mpsse->mode == BITBANG) { | 
|  | if (direction == HIGH) { | 
|  | mpsse->bitbang |= (1 << pin); | 
|  | } else { | 
|  | mpsse->bitbang &= ~(1 << pin); | 
|  | } | 
|  |  | 
|  | if (set_bits_high(mpsse, mpsse->bitbang) == MPSSE_OK) { | 
|  | retval = raw_write(mpsse, (uint8_t*)&mpsse->bitbang, 1); | 
|  | } | 
|  | } else { | 
|  | /* The first four pins can't be changed unless we are in a stopped status | 
|  | */ | 
|  | if (pin < NUM_GPIOL_PINS && mpsse->status == STOPPED) { | 
|  | /* Convert pin number (0-3) to the corresponding pin bit */ | 
|  | pin = (GPIO0 << pin); | 
|  |  | 
|  | if (direction == HIGH) { | 
|  | mpsse->pstart |= pin; | 
|  | mpsse->pidle |= pin; | 
|  | mpsse->pstop |= pin; | 
|  | } else { | 
|  | mpsse->pstart &= ~pin; | 
|  | mpsse->pidle &= ~pin; | 
|  | mpsse->pstop &= ~pin; | 
|  | } | 
|  |  | 
|  | retval = set_bits_low(mpsse, mpsse->pstop); | 
|  | } else if (pin >= NUM_GPIOL_PINS && pin < NUM_GPIO_PINS) { | 
|  | /* Convert pin number (4 - 11) to the corresponding pin bit */ | 
|  | pin -= NUM_GPIOL_PINS; | 
|  |  | 
|  | if (direction == HIGH) { | 
|  | mpsse->gpioh |= (1 << pin); | 
|  | } else { | 
|  | mpsse->gpioh &= ~(1 << pin); | 
|  | } | 
|  |  | 
|  | retval = set_bits_high(mpsse, mpsse->gpioh); | 
|  | } | 
|  | } | 
|  |  | 
|  | return retval; | 
|  | } | 
|  |  | 
|  | /* Checks if a given MPSSE context is valid. */ | 
|  | int is_valid_context(struct mpsse_context* mpsse) { | 
|  | return mpsse != NULL; | 
|  | } |