upload android base code part6
This commit is contained in:
parent
421e214c7d
commit
4e516ec6ed
35396 changed files with 9188716 additions and 0 deletions
278
android/system/tpm/trunks/ftdi/support.c
Normal file
278
android/system/tpm/trunks/ftdi/support.c
Normal file
|
@ -0,0 +1,278 @@
|
|||
/*
|
||||
/*
|
||||
/*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 <string.h>
|
||||
|
||||
#include "trunks/ftdi/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;
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue