677 lines
15 KiB
C
677 lines
15 KiB
C
/* MSPDebug - debugging tool for the eZ430
|
|
* Copyright (C) 2009, 2010 Daniel Beer
|
|
* Copyright (C) 2010 Andrew Armenia
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
|
|
#include "flash_bsl.h"
|
|
#include "util.h"
|
|
#include "output.h"
|
|
#include "fet_error.h"
|
|
|
|
struct flash_bsl_device {
|
|
struct device base;
|
|
|
|
int serial_fd;
|
|
int long_password;
|
|
};
|
|
|
|
|
|
|
|
#define MAX_PACKET 256
|
|
|
|
|
|
/* adapted from TI's published BSL source code */
|
|
#define CRC_INIT 0xffff
|
|
static uint16_t crc_ccitt(const uint8_t *data, int len) {
|
|
uint16_t crc = CRC_INIT;
|
|
uint16_t temp;
|
|
int i;
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
temp = ((crc >> 8) ^ data[i]) & 0xff;
|
|
temp ^= (temp >> 4);
|
|
crc = (crc << 8) ^ (temp << 12) ^ (temp << 5) ^ temp;
|
|
}
|
|
|
|
return crc;
|
|
}
|
|
|
|
static void crc_selftest(void) {
|
|
/* These test vectors are from page 30 of TI doc SLAU319A */
|
|
uint16_t crc_expected = 0x5590;
|
|
uint16_t crc_actual = crc_ccitt((uint8_t *)"\x52\x02", 2);
|
|
if (crc_expected != crc_actual) {
|
|
printc_err("flash_bsl: CRC malfunction (expected 0x%04x got 0x%04x)\n",
|
|
crc_expected, crc_actual);
|
|
}
|
|
|
|
crc_expected = 0x121d;
|
|
crc_actual = crc_ccitt((uint8_t *)"\x3a\x04\x01", 3);
|
|
if (crc_expected != crc_actual) {
|
|
printc_err("flash_bsl: CRC malfunction (expected 0x%04x got 0x%04x)\n",
|
|
crc_expected, crc_actual);
|
|
}
|
|
|
|
crc_expected = 0x528b;
|
|
crc_actual = crc_ccitt((uint8_t *)"\x1a", 1);
|
|
if (crc_expected != crc_actual) {
|
|
printc_err("flash_bsl: CRC malfunction (expected 0x%04x got 0x%04x)\n",
|
|
crc_expected, crc_actual);
|
|
}
|
|
|
|
}
|
|
|
|
#define RX_DATA_BLOCK 0x10
|
|
#define RX_DATA_BLOCK_FAST 0x1b
|
|
#define RX_PASSWORD 0x11
|
|
#define ERASE_SEGMENT 0x12
|
|
#define UNLOCK_LOCK_INFO 0x13
|
|
#define MASS_ERASE 0x15
|
|
#define CRC_CHECK 0x16
|
|
#define LOAD_PC 0x17
|
|
#define TX_DATA_BLOCK 0x18
|
|
#define TX_BSL_VERSION 0x19
|
|
#define TX_BUFFER_SIZE 0x1a
|
|
|
|
static int flash_bsl_send(struct flash_bsl_device *dev, const uint8_t *data, int len)
|
|
{
|
|
uint16_t crc;
|
|
uint8_t cmd_buf[MAX_PACKET + 5];
|
|
uint8_t response;
|
|
|
|
#if defined(FLASH_BSL_VERBOSE)
|
|
debug_hexdump("flash_bsl: sending", data, len);
|
|
#endif
|
|
|
|
crc = crc_ccitt(data, len);
|
|
|
|
if (len > MAX_PACKET) {
|
|
printc_err("flash_bsl: attempted to transmit long packet (len=%d)\n", len);
|
|
return -1;
|
|
}
|
|
|
|
cmd_buf[0] = 0x80;
|
|
cmd_buf[1] = (len & 0xff);
|
|
cmd_buf[2] = (len >> 8) & 0xff;
|
|
memcpy(cmd_buf + 3, data, len);
|
|
cmd_buf[len + 3] = (crc & 0xff);
|
|
cmd_buf[len + 4] = (crc >> 8) & 0xff;
|
|
|
|
if (write_all(dev->serial_fd, cmd_buf, len + 5) < 0) {
|
|
printc_err("flash_bsl: serial write failed\n");
|
|
return -1;
|
|
}
|
|
|
|
if (read_all_with_timeout(dev->serial_fd, &response, 1) < 0) {
|
|
if (errno == ETIMEDOUT) {
|
|
printc_err("flash_bsl: serial read timed out\n");
|
|
} else {
|
|
printc_err("flash_bsl: serial read failed\n");
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
if (response != 0) {
|
|
switch (response) {
|
|
case 0x51:
|
|
printc_err("flash_bsl: BSL reports incorrect packet header\n");
|
|
break;
|
|
case 0x52:
|
|
printc_err("flash_bsl: BSL reports checksum incorrect\n");
|
|
break;
|
|
case 0x53:
|
|
printc_err("flash_bsl: BSL got zero-size packet\n");
|
|
break;
|
|
case 0x54:
|
|
printc_err("flash_bsl: BSL receive buffer overflowed\n");
|
|
break;
|
|
case 0x55:
|
|
printc_err("flash_bsl: (known-)unknown error\n");
|
|
break;
|
|
case 0x56:
|
|
printc_err("flash_bsl: unknown baud rate\n");
|
|
break;
|
|
default:
|
|
printc_err("flash_bsl: unknown unknown error\n");
|
|
break;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int flash_bsl_recv(struct flash_bsl_device *dev,
|
|
uint8_t *recv_buf, int buf_len)
|
|
{
|
|
uint8_t header[3];
|
|
uint8_t crc_bytes[2];
|
|
uint16_t recv_len;
|
|
uint16_t crc_value;
|
|
|
|
if (read_all_with_timeout(dev->serial_fd, header, 3) < 0) {
|
|
if (errno == ETIMEDOUT) {
|
|
printc_err("flash_bsl: response timed out\n");
|
|
return -1;
|
|
} else {
|
|
perror("read response header");
|
|
printc_err("flash_bsl: read response failed\n");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (header[0] != 0x80) {
|
|
printc_err("flash_bsl: incorrect response header received\n");
|
|
return -1;
|
|
}
|
|
|
|
recv_len = header[2];
|
|
recv_len <<= 8;
|
|
recv_len |= header[1];
|
|
|
|
#if defined(FLASH_BSL_VERBOSE)
|
|
printc_dbg("flash_bsl: incoming message length %d\n", recv_len);
|
|
#endif
|
|
|
|
if (recv_len > buf_len) {
|
|
printc_err("flash_bsl: insufficient buffer to receive data\n");
|
|
return -1;
|
|
}
|
|
|
|
if (read_all_with_timeout(dev->serial_fd, recv_buf, recv_len) < 0) {
|
|
perror("receive message");
|
|
printc_err("flash_bsl: error receiving message\n");
|
|
return -1;
|
|
}
|
|
|
|
if (read_all_with_timeout(dev->serial_fd, crc_bytes, 2) < 0) {
|
|
perror("receive message CRC");
|
|
printc_err("flash_bsl: error receiving message CRC\n");
|
|
return -1;
|
|
}
|
|
|
|
crc_value = crc_bytes[1];
|
|
crc_value <<= 8;
|
|
crc_value |= crc_bytes[0];
|
|
|
|
if (crc_ccitt(recv_buf, recv_len) != crc_value) {
|
|
printc_err("flash_bsl: received message with bad CRC\n");
|
|
return -1;
|
|
}
|
|
|
|
#if defined(FLASH_BSL_VERBOSE)
|
|
debug_hexdump("received message", recv_buf, recv_len);
|
|
#endif
|
|
|
|
usleep(10000);
|
|
return recv_len;
|
|
}
|
|
|
|
static void flash_bsl_perror(uint8_t code) {
|
|
switch (code) {
|
|
case 0x00:
|
|
printc_err("flash_bsl: success\n");
|
|
break;
|
|
case 0x01:
|
|
printc_err("flash_bsl: FLASH verify failed\n");
|
|
break;
|
|
case 0x02:
|
|
printc_err("flash_bsl: FLASH operation failed\n");
|
|
break;
|
|
case 0x03:
|
|
printc_err("flash_bsl: voltage not constant during program\n");
|
|
break;
|
|
case 0x04:
|
|
printc_err("flash_bsl: BSL is locked\n");
|
|
break;
|
|
case 0x05:
|
|
printc_err("flash_bsl: incorrect password\n");
|
|
break;
|
|
case 0x06:
|
|
printc_err("flash_bsl: attempted byte write to FLASH\n");
|
|
break;
|
|
case 0x07:
|
|
printc_err("flash_bsl: unrecognized command\n");
|
|
break;
|
|
case 0x08:
|
|
printc_err("flash_bsl: command was too long\n");
|
|
break;
|
|
default:
|
|
printc_err("flash_bsl: unknown status message\n");
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
#define MAX_BLOCK 256
|
|
static int flash_bsl_readmem(device_t dev_base,
|
|
address_t addr, uint8_t *mem, address_t len)
|
|
{
|
|
struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base;
|
|
uint8_t recv_buf[MAX_BLOCK*2];
|
|
uint8_t send_buf[64];
|
|
uint16_t read_size;
|
|
int ret;
|
|
|
|
if (addr > 0xfffff || addr + len > 0xfffff) {
|
|
printc_err("flash_bsl: read exceeds possible range\n");
|
|
return -1;
|
|
}
|
|
|
|
while (len > 0) {
|
|
if (len > MAX_BLOCK) {
|
|
read_size = MAX_BLOCK;
|
|
} else {
|
|
read_size = len;
|
|
}
|
|
|
|
/* build command */
|
|
send_buf[0] = TX_DATA_BLOCK; /* command: transmit data block */
|
|
send_buf[1] = addr & 0xff;
|
|
send_buf[2] = (addr >> 8) & 0xff;
|
|
send_buf[3] = (addr >> 16) & 0xff;
|
|
send_buf[4] = read_size & 0xff;
|
|
send_buf[5] = (read_size >> 8) & 0xff;
|
|
|
|
if (flash_bsl_send(dev, send_buf, 6) < 0) {
|
|
printc_err("flash_bsl readmem: send failed\n");
|
|
return -1;
|
|
}
|
|
|
|
ret = flash_bsl_recv(dev, recv_buf, read_size + 1);
|
|
if (ret < 0) {
|
|
printc_err("flash_bsl readmem: receive failed\n");
|
|
return -1;
|
|
} else if (ret < read_size) {
|
|
printc_err("flash_bsl readmem: warning: not all requested data received\n");
|
|
}
|
|
|
|
if (recv_buf[0] == 0x3a) {
|
|
memcpy(mem, recv_buf + 1, ret - 1);
|
|
len -= ret - 1;
|
|
mem += ret - 1;
|
|
} else if (recv_buf[0] == 0x3b) {
|
|
flash_bsl_perror(recv_buf[1]);
|
|
} else {
|
|
printc_err("flash_bsl readmem: invalid response\n");
|
|
return -1;
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int flash_bsl_erase(struct flash_bsl_device *dev)
|
|
{
|
|
const uint8_t mass_erase_cmd[] = { MASS_ERASE };
|
|
uint8_t response_buffer[16];
|
|
int ret;
|
|
|
|
if (flash_bsl_send(dev, mass_erase_cmd, sizeof(mass_erase_cmd)) < 0) {
|
|
printc_err("flash_bsl_erase: failed to send erase command\n");
|
|
return -1;
|
|
}
|
|
|
|
ret = flash_bsl_recv(dev, response_buffer, sizeof(response_buffer));
|
|
if (ret < 2) {
|
|
printc_err("flash_bsl_erase: no response\n");
|
|
return -1;
|
|
}
|
|
|
|
if (response_buffer[0] != 0x3b) {
|
|
printc_err("flash_bsl_erase: incorrect response\n");
|
|
return -1;
|
|
}
|
|
|
|
if (response_buffer[1] != 0) {
|
|
flash_bsl_perror(response_buffer[1]);
|
|
printc_err("flash_bsl_erase: erase failed\n");
|
|
return -1;
|
|
} else {
|
|
#if defined(FLASH_BSL_VERBOSE)
|
|
printc_dbg("flash_bsl_erase: success\n");
|
|
#endif
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int flash_bsl_unlock(struct flash_bsl_device *dev)
|
|
{
|
|
/*
|
|
* after erase, the password will be 0xff * (16 or 32)
|
|
* (an empty interrupt vector table)
|
|
*/
|
|
uint8_t rx_password_cmd[33] = "\x11"
|
|
"\xff\xff\xff\xff\xff\xff\xff\xff"
|
|
"\xff\xff\xff\xff\xff\xff\xff\xff"
|
|
"\xff\xff\xff\xff\xff\xff\xff\xff"
|
|
"\xff\xff\xff\xff\xff\xff\xff\xff";
|
|
|
|
|
|
|
|
uint8_t response_buffer[16];
|
|
int ret;
|
|
|
|
/* mass erase (this might wipe Information Memory on some devices */
|
|
if (flash_bsl_erase(dev) < 0) {
|
|
printc_err("flash_bsl_unlock: warning: erase failed\n");
|
|
}
|
|
|
|
/* send password (which is now erased FLASH) */
|
|
if (dev->long_password) {
|
|
#if defined(FLASH_BSL_VERBOSE)
|
|
printc_dbg("flash_bsl_unlock: using long password\n");
|
|
#endif
|
|
}
|
|
|
|
if (flash_bsl_send(dev, rx_password_cmd, dev->long_password ? 33 : 17) < 0) {
|
|
printc_err("flash_bsl_unlock: send password failed\n");
|
|
return -1;
|
|
}
|
|
|
|
ret = flash_bsl_recv(dev, response_buffer, sizeof(response_buffer));
|
|
|
|
if (ret < 2) {
|
|
printc_err("flash_bsl_unlock: error receiving password response\n");
|
|
return -1;
|
|
}
|
|
|
|
if (response_buffer[0] != 0x3b) {
|
|
printc_err("flash_bsl_unlock: received invalid password response\n");
|
|
return -1;
|
|
}
|
|
|
|
if (response_buffer[1] != 0x00) {
|
|
flash_bsl_perror(response_buffer[1]);
|
|
printc_err("flash_bsl_unlock: password error\n");
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int flash_bsl_ctl(device_t dev_base, device_ctl_t type)
|
|
{
|
|
struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base;
|
|
|
|
switch (type) {
|
|
case DEVICE_CTL_ERASE:
|
|
return flash_bsl_erase(dev);
|
|
|
|
case DEVICE_CTL_HALT:
|
|
/* Ignore halt requests */
|
|
return 0;
|
|
|
|
case DEVICE_CTL_RESET:
|
|
/* Ignore reset requests */
|
|
return 0;
|
|
|
|
default:
|
|
printc_err("flash_bsl: CPU control is not possible\n");
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
static device_status_t flash_bsl_poll(device_t dev_base)
|
|
{
|
|
return DEVICE_STATUS_HALTED;
|
|
}
|
|
|
|
static int flash_bsl_getregs(device_t dev_base, address_t *regs)
|
|
{
|
|
printc_err("flash_bsl: register fetch is not implemented\n");
|
|
return -1;
|
|
}
|
|
|
|
static int flash_bsl_setregs(device_t dev_base, const address_t *regs)
|
|
{
|
|
printc_err("flash_bsl: register store is not implemented\n");
|
|
return -1;
|
|
}
|
|
|
|
static int flash_bsl_writemem(device_t dev_base,
|
|
address_t addr, const uint8_t *mem, address_t len)
|
|
{
|
|
struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base;
|
|
uint8_t send_buf[2*MAX_BLOCK];
|
|
uint8_t recv_buf[16];
|
|
uint16_t write_size;
|
|
int n_recv;
|
|
|
|
if (addr > 0xfffff || addr + len > 0xfffff) {
|
|
printc_err("flash_bsl: write exceeds possible range\n");
|
|
return -1;
|
|
}
|
|
|
|
while (len > 0) {
|
|
/* compute size of this write operation */
|
|
if (len > MAX_BLOCK) {
|
|
write_size = MAX_BLOCK;
|
|
} else {
|
|
write_size = len;
|
|
}
|
|
|
|
/* build write command */
|
|
/* command */
|
|
send_buf[0] = RX_DATA_BLOCK;
|
|
/* address */
|
|
send_buf[1] = addr & 0xff;
|
|
send_buf[2] = (addr >> 8) & 0xff;
|
|
send_buf[3] = (addr >> 16) & 0xff;
|
|
/* data */
|
|
memcpy(&send_buf[4], mem, write_size);
|
|
|
|
mem += write_size;
|
|
len -= write_size;
|
|
|
|
/* send command */
|
|
if (flash_bsl_send(dev, send_buf, write_size + 4) < 0) {
|
|
printc_err("flash_bsl: send failed\n");
|
|
return -1;
|
|
}
|
|
|
|
/* receive and check response */
|
|
n_recv = flash_bsl_recv(dev, recv_buf, sizeof(recv_buf));
|
|
|
|
if (n_recv < 0) {
|
|
printc_err("flash_bsl write: error occurred receiving response\n");
|
|
return -1;
|
|
} else if (n_recv < 2) {
|
|
printc_err("flash_bsl write: response too short\n");
|
|
return -1;
|
|
} else if (recv_buf[0] != 0x3b) {
|
|
printc_err("flash_bsl write: invalid response received\n");
|
|
return -1;
|
|
} else if (recv_buf[1] != 0x00) {
|
|
printc_err("flash_bsl write: BSL reported write error: ");
|
|
flash_bsl_perror(recv_buf[1]);
|
|
return -1;
|
|
}
|
|
/* else success! */
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void entry_delay(void)
|
|
{
|
|
usleep(1000);
|
|
}
|
|
|
|
static int set_serial_bits(int fd, int bits)
|
|
{
|
|
if (ioctl(fd, TIOCMSET, &bits) >= 0) {
|
|
return 0;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static int enter_via_dtr_rts(struct flash_bsl_device *dev)
|
|
{
|
|
/*
|
|
* Implement the sequence shown on page 8 of TI document SLAU319A,
|
|
* via serial-port control lines.
|
|
*/
|
|
|
|
int fd = dev->serial_fd;
|
|
|
|
|
|
/* drive RST# line low */
|
|
if (set_serial_bits(fd, TIOCM_RTS | TIOCM_DTR) != 0) {
|
|
return -1;
|
|
}
|
|
entry_delay( );
|
|
|
|
/* drive TEST line high then low again */
|
|
if (set_serial_bits(fd, TIOCM_DTR) != 0) {
|
|
return -1;
|
|
}
|
|
entry_delay( );
|
|
if (set_serial_bits(fd, TIOCM_RTS | TIOCM_DTR) != 0) {
|
|
return -1;
|
|
}
|
|
entry_delay( );
|
|
|
|
/* drive TEST line high followed by RST# line */
|
|
if (set_serial_bits(fd, TIOCM_DTR) != 0) {
|
|
return -1;
|
|
}
|
|
entry_delay( );
|
|
if (set_serial_bits(fd, 0) != 0) {
|
|
return -1;
|
|
}
|
|
entry_delay( );
|
|
if (set_serial_bits(fd, TIOCM_RTS) != 0) {
|
|
return -1;
|
|
}
|
|
entry_delay( );
|
|
|
|
|
|
/* BSL should now be running! */
|
|
return 0;
|
|
}
|
|
|
|
static void exit_via_dtr_rts(struct flash_bsl_device *dev)
|
|
{
|
|
int fd = dev->serial_fd;
|
|
|
|
/* RST# and TEST LOW */
|
|
set_serial_bits(fd, TIOCM_RTS | TIOCM_DTR);
|
|
|
|
/* wait a brief period */
|
|
entry_delay( );
|
|
|
|
/* RST# HIGH */
|
|
set_serial_bits(fd, TIOCM_DTR);
|
|
}
|
|
|
|
static void flash_bsl_destroy(device_t dev_base)
|
|
{
|
|
struct flash_bsl_device *dev = (struct flash_bsl_device *)dev_base;
|
|
|
|
exit_via_dtr_rts(dev);
|
|
|
|
close(dev->serial_fd);
|
|
free(dev);
|
|
}
|
|
|
|
device_t flash_bsl_open(const char *device, int long_password)
|
|
{
|
|
struct flash_bsl_device *dev = malloc(sizeof(*dev));
|
|
uint8_t tx_bsl_version_command[] = { TX_BSL_VERSION };
|
|
uint8_t tx_bsl_version_response[5];
|
|
|
|
if (!dev) {
|
|
pr_error("flash_bsl: can't allocate memory");
|
|
return NULL;
|
|
}
|
|
|
|
|
|
crc_selftest( );
|
|
|
|
memset(dev, 0, sizeof(*dev));
|
|
|
|
dev->base.destroy = flash_bsl_destroy;
|
|
dev->base.readmem = flash_bsl_readmem;
|
|
dev->base.writemem = flash_bsl_writemem;
|
|
dev->base.getregs = flash_bsl_getregs;
|
|
dev->base.setregs = flash_bsl_setregs;
|
|
dev->base.ctl = flash_bsl_ctl;
|
|
dev->base.poll = flash_bsl_poll;
|
|
|
|
dev->serial_fd = open_serial_even_parity(device, B9600);
|
|
if (dev->serial_fd < 0) {
|
|
printc_err("flash_bsl: can't open %s: %s\n",
|
|
device, strerror(errno));
|
|
free(dev);
|
|
return NULL;
|
|
}
|
|
|
|
dev->long_password = long_password;
|
|
|
|
/* enter bootloader */
|
|
if (enter_via_dtr_rts(dev) < 0) {
|
|
goto fail;
|
|
}
|
|
|
|
usleep(500000);
|
|
|
|
/* unlock device (erase then send password) */
|
|
if (flash_bsl_unlock(dev) < 0) {
|
|
goto fail;
|
|
}
|
|
|
|
|
|
if (flash_bsl_send(dev, tx_bsl_version_command, sizeof(tx_bsl_version_command)) < 0) {
|
|
printc_err("flash_bsl: failed to read BSL version");
|
|
goto fail;
|
|
}
|
|
|
|
if (flash_bsl_recv(dev, tx_bsl_version_response,
|
|
sizeof(tx_bsl_version_response)) < sizeof(tx_bsl_version_response)) {
|
|
|
|
printc_err("flash_bsl: BSL responded with invalid version");
|
|
goto fail;
|
|
}
|
|
|
|
debug_hexdump("BSL version", tx_bsl_version_response,
|
|
sizeof(tx_bsl_version_response));
|
|
|
|
return (device_t)dev;
|
|
|
|
fail:
|
|
close(dev->serial_fd);
|
|
free(dev);
|
|
return NULL;
|
|
}
|