349 lines
7.1 KiB
C

/*
NanoStack: MCU software and PC tools for IP-based wireless sensor networking.
Copyright (C) 2006-2007 Sensinode Ltd.
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 Street, Fifth Floor, Boston, MA 02110-1301 USA.
Address:
Sensinode Ltd.
Teknologiantie 6
90570 Oulu, Finland
E-mail:
info@sensinode.com
*/
/*
To build use the following gcc statement
(assuming you have the d2xx library in the /usr/local/lib directory).
gcc -o bitmode main.c -L. -lftd2xx -Wl,-rpath /usr/local/lib
*/
#include <unistd.h>
#include <getopt.h>
#include <strings.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#ifdef PLATFORM_WINDOWS
#include <windows.h>
#endif
#include "prog.h"
#define PRG_VERSION "1.3"
typedef enum
{
USAGE,
VERSION,
SCAN,
WRITE,
READ,
WRITE_MAC,
READ_MAC,
ERASE
}mode_t;
typedef struct opts_t
{
int port;
mode_t mode;
uint8_t write_mac[8];
char *filename;
}opts_t;
opts_t opts;
void usage(char *prg_name)
{
printf("\nUsage: %s [-p port] [-h] [-v] [-f file] [-r] [-w] [-m] [-M <mac>]\n", prg_name);
printf("General options:\n");
printf(" -p/--port [port] Select FTDI device\n");
printf(" -f/--file [filename] File to read/write\n");
printf("Operating modes:\n");
printf(" -d/--devices Scan available devices\n");
printf(" -v/--version Print program version\n");
printf(" -r/--read Read program code into ihex file (see -f)\n");
printf(" -w/--write Program firmware from ihex file (see -f)\n");
printf(" -m/--read-mac Read device MAC address\n");
printf(" -M/--write-mac xx:xx:xx:xx:xx:xx:xx:xx Write device MAC address\n");
printf(" -e/--erase Erase flash (erases MAC address!)\n");
printf("Defaults:\n");
printf("mode = usage\n");
printf("port = undefined\n");
printf("file = stdout\n");
}
int main(int argc, char *argv[])
{
DWORD dwBytesInQueue = 0;
FT_STATUS ftStatus;
FT_HANDLE ftHandle;
unsigned char ucMode = 0x00;
int i=0;
unsigned char wr[1] = { 0x30 };
unsigned char rd[1];
uint16_t chip_id;
if (opts_parse(argc, argv) < 0)
{
usage(argv[0]);
return -1;
}
switch(opts.mode)
{
case VERSION:
printf("Sensinode Nano USB Programmer version %s\n", PRG_VERSION);
return 0;
case USAGE:
usage(argv[0]);
return 0;
case SCAN:
prog_scan();
return 0;
case ERASE:
case READ:
case WRITE:
case READ_MAC:
case WRITE_MAC:
break;
}
printf("Opening programmer.\n");
ftHandle = prog_open(opts.port);
if (ftHandle == 0)
{
return (-1);
}
cdi_start(&chip_id);
printf("Chip ID = %4.4hX.\n", chip_id);
if ((chip_id & 0xFF00) == 0x8500)
{
printf("CC2430 chip found.\n");
}
else if ((chip_id & 0xFF00) == 0x8900)
{
printf("CC2431 chip found.\n");
}
else
{
printf("Unknown chip found.\n");
opts.mode = USAGE;
}
switch(opts.mode)
{
case VERSION:
case USAGE:
break;
case ERASE:
printf("Erase.\n");
break;
case READ:
printf("Read Flash.\n");
cdi_set_address(0);
break;
case WRITE:
{
int rval;
unsigned char page_table[64];
unsigned char page_buffer[128*1024];
printf("Write Flash.\n");
if((rval = hexfile_build_tables(opts.filename, page_buffer, page_table)) == -1)
{
printf("Error\n");
return(-1);
}
else if(rval == 0)
{
printf(".ihex file OK but nothing to write...\n");
return(1);
}
hexfile_program(page_buffer, page_table);
break;
}
case READ_MAC:
printf("Read MAC: ");
cdi_set_address(0x1FFF8);
{
uint8_t mac[8];
cdi_flash_read(mac, 8);
for (i=0; i<8; i++)
{
if (i) printf(":");
printf("%2.2X", mac[i]);
}
printf("\n");
}
break;
case WRITE_MAC:
printf("Write MAC: ");
/* cdi_set_address(0x1F800);
{
uint8_t block[2048];
memset(block, 0xFF, 2048);
for (i=0; i<8; i++)
{
block[2040+i] = opts.write_mac[i];
}
cdi_flash_write(block, 2048);
printf("\n");
}*/
cdi_flash_write_mac(opts.write_mac);
printf("\n");
break;
default:
printf("Duh\n");
break;
}
printf("Closing programmer.\n");
prog_close();
}
static int option_index = 0;
int do_exit = 0;
#define OPTIONS_STRING "p:vdhf:rwmM:e"
/* long option list */
static struct option long_options[] =
{
{"port", 1, NULL, 'p'},
{"version", 0, NULL, 'v'},
{"devices", 0, NULL, 'd'},
{"help", 0, NULL, 'h'},
{"file", 1, NULL, 'f'},
{"read", 0, NULL, 'r'},
{"write", 0, NULL, 'w'},
{"read-mac", 0, NULL, 'm'},
{"write-mac", 1, NULL, 'M'},
{"erase", 0, NULL, 'e'},
{0, 0, 0, 0}
};
int opts_parse(int count, char* param[])
{
int opt;
int error=0;
opts.mode = USAGE;
opts.filename = 0;
while ((opt = getopt_long(count, param, OPTIONS_STRING,
long_options, &option_index)) != -1)
{
fflush(stdout);
switch(opt)
{
case 'p':
opts.port = 0;
if (sscanf(optarg, "%d", &(opts.port)) != 1)
{
if (sscanf(optarg, "/dev/ttyUSB%d", &(opts.port)) != 1)
{
printf("Invalid port.\n");
opts.mode = USAGE;
return 0;
}
}
printf("Port %d.\n", opts.port);
break;
case 'v':
opts.mode = VERSION;
return 0;
case 'd':
opts.mode = SCAN;
return 0;
case 'h':
opts.mode = USAGE;
return 0;
case 'e':
opts.mode = ERASE;
break;
case 'f':
printf("Filename: %s\n", optarg);
opts.filename = malloc(strlen(optarg)+1);
strcpy(opts.filename, optarg);
break;
case 'r':
opts.mode = READ;
break;
case 'w':
opts.mode = WRITE;
break;
case 'm':
opts.mode = READ_MAC;
break;
case 'M':
opts.mode = WRITE_MAC;
if (sscanf(optarg, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx",
&opts.write_mac[0], &opts.write_mac[1],
&opts.write_mac[2], &opts.write_mac[3],
&opts.write_mac[4], &opts.write_mac[5],
&opts.write_mac[6], &opts.write_mac[7]) != 8)
{
printf("Invalid MAC.\n");
opts.mode = USAGE;
}
else
{
printf("MAC to write: %2.2hhX:%2.2hhX:%2.2hhX:%2.2hhX:%2.2hhX:%2.2hhX:%2.2hhX:%2.2hhX\n",
opts.write_mac[0], opts.write_mac[1],
opts.write_mac[2], opts.write_mac[3],
opts.write_mac[4], opts.write_mac[5],
opts.write_mac[6], opts.write_mac[7]);
}
break;
case '?':
printf("Duh\n");
error = -1;
break;
}
}
return error;
}