363 lines
10 KiB
C
363 lines
10 KiB
C
/*
|
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
|
* avrdude is Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.com>
|
|
*
|
|
* This file: Copyright (C) 2005-2007 Colin O'Flynn <coflynn@newae.com>
|
|
*
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "ac_cfg.h"
|
|
#include "avrdude.h"
|
|
#include "avr.h"
|
|
#include "pgm.h"
|
|
#include "safemode.h"
|
|
|
|
/* This value from ac_cfg.h */
|
|
/*
|
|
* Writes the specified fuse in fusename (can be "lfuse", "hfuse", or
|
|
* "efuse") and verifies it. Will try up to tries amount of times
|
|
* before giving up
|
|
*/
|
|
int safemode_writefuse (unsigned char fuse, char * fusename, PROGRAMMER * pgm,
|
|
AVRPART * p, int tries, int verbose)
|
|
{
|
|
AVRMEM * m;
|
|
unsigned char fuseread;
|
|
int returnvalue = -1;
|
|
|
|
m = avr_locate_mem(p, fusename);
|
|
if (m == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
/* Keep trying to write then read back the fuse values */
|
|
while (tries > 0) {
|
|
if (avr_write_byte(pgm, p, m, 0, fuse) != 0)
|
|
{
|
|
continue;
|
|
}
|
|
if (pgm->read_byte(pgm, p, m, 0, &fuseread) != 0)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
/* Report information to user if needed */
|
|
if (verbose > 0) {
|
|
fprintf(stderr,
|
|
"%s: safemode: Wrote %s to %x, read as %x. %d attempts left\n",
|
|
progname, fusename, fuse, fuseread, tries-1);
|
|
}
|
|
|
|
/* If fuse wrote OK, no need to keep going */
|
|
if (fuse == fuseread) {
|
|
tries = 0;
|
|
returnvalue = 0;
|
|
}
|
|
tries--;
|
|
}
|
|
|
|
return returnvalue;
|
|
}
|
|
|
|
/*
|
|
* Reads the fuses three times, checking that all readings are the
|
|
* same. This will ensure that the before values aren't in error!
|
|
*/
|
|
int safemode_readfuses (unsigned char * lfuse, unsigned char * hfuse,
|
|
unsigned char * efuse, unsigned char * fuse,
|
|
PROGRAMMER * pgm, AVRPART * p, int verbose)
|
|
{
|
|
|
|
unsigned char value;
|
|
unsigned char fusegood = 0;
|
|
unsigned char allowfuseread = 1;
|
|
unsigned char safemode_lfuse;
|
|
unsigned char safemode_hfuse;
|
|
unsigned char safemode_efuse;
|
|
unsigned char safemode_fuse;
|
|
AVRMEM * m;
|
|
|
|
safemode_lfuse = *lfuse;
|
|
safemode_hfuse = *hfuse;
|
|
safemode_efuse = *efuse;
|
|
safemode_fuse = *fuse;
|
|
|
|
|
|
/* Read fuse three times */
|
|
fusegood = 2; /* If AVR device doesn't support this fuse, don't want
|
|
to generate a verify error */
|
|
m = avr_locate_mem(p, "fuse");
|
|
if (m != NULL) {
|
|
fusegood = 0; /* By default fuse is a failure */
|
|
if(pgm->read_byte(pgm, p, m, 0, &safemode_fuse) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 1, fuse value: %x\n",progname, safemode_fuse);
|
|
}
|
|
if(pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 2, fuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_fuse) {
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 3, fuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_fuse)
|
|
{
|
|
fusegood = 1; /* Fuse read OK three times */
|
|
}
|
|
}
|
|
}
|
|
|
|
//Programmer does not allow fuse reading.... no point trying anymore
|
|
if (allowfuseread == 0)
|
|
{
|
|
return -5;
|
|
}
|
|
|
|
if (fusegood == 0) {
|
|
fprintf(stderr,
|
|
"%s: safemode: Verify error - unable to read fuse properly. "
|
|
"Programmer may not be reliable.\n", progname);
|
|
return -1;
|
|
}
|
|
else if ((fusegood == 1) && (verbose > 0)) {
|
|
fprintf(stderr, "%s: safemode: fuse reads as %X\n", progname, safemode_fuse);
|
|
}
|
|
|
|
|
|
/* Read lfuse three times */
|
|
fusegood = 2; /* If AVR device doesn't support this fuse, don't want
|
|
to generate a verify error */
|
|
m = avr_locate_mem(p, "lfuse");
|
|
if (m != NULL) {
|
|
fusegood = 0; /* By default fuse is a failure */
|
|
if (pgm->read_byte(pgm, p, m, 0, &safemode_lfuse) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 1, lfuse value: %x\n",progname, safemode_lfuse);
|
|
}
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 2, lfuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_lfuse) {
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 3, lfuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_lfuse){
|
|
fusegood = 1; /* Fuse read OK three times */
|
|
}
|
|
}
|
|
}
|
|
|
|
//Programmer does not allow fuse reading.... no point trying anymore
|
|
if (allowfuseread == 0)
|
|
{
|
|
return -5;
|
|
}
|
|
|
|
|
|
if (fusegood == 0) {
|
|
fprintf(stderr,
|
|
"%s: safemode: Verify error - unable to read lfuse properly. "
|
|
"Programmer may not be reliable.\n", progname);
|
|
return -1;
|
|
}
|
|
else if ((fusegood == 1) && (verbose > 0)) {
|
|
fprintf(stderr, "%s: safemode: lfuse reads as %X\n", progname, safemode_lfuse);
|
|
}
|
|
|
|
/* Read hfuse three times */
|
|
fusegood = 2; /* If AVR device doesn't support this fuse, don't want
|
|
to generate a verify error */
|
|
m = avr_locate_mem(p, "hfuse");
|
|
if (m != NULL) {
|
|
fusegood = 0; /* By default fuse is a failure */
|
|
if (pgm->read_byte(pgm, p, m, 0, &safemode_hfuse) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 1, hfuse value: %x\n",progname, safemode_hfuse);
|
|
}
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 2, hfuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_hfuse) {
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 3, hfuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_hfuse){
|
|
fusegood = 1; /* Fuse read OK three times */
|
|
}
|
|
}
|
|
}
|
|
|
|
//Programmer does not allow fuse reading.... no point trying anymore
|
|
if (allowfuseread == 0)
|
|
{
|
|
return -5;
|
|
}
|
|
|
|
if (fusegood == 0) {
|
|
fprintf(stderr,
|
|
"%s: safemode: Verify error - unable to read hfuse properly. "
|
|
"Programmer may not be reliable.\n", progname);
|
|
return -2;
|
|
}
|
|
else if ((fusegood == 1) && (verbose > 0)){
|
|
fprintf(stderr, "%s: safemode: hfuse reads as %X\n", progname, safemode_hfuse);
|
|
}
|
|
|
|
/* Read efuse three times */
|
|
fusegood = 2; /* If AVR device doesn't support this fuse, don't want
|
|
to generate a verify error */
|
|
m = avr_locate_mem(p, "efuse");
|
|
if (m != NULL) {
|
|
fusegood = 0; /* By default fuse is a failure */
|
|
if (pgm->read_byte(pgm, p, m, 0, &safemode_efuse) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 1, efuse value: %x\n",progname, safemode_efuse);
|
|
}
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 2, efuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_efuse) {
|
|
if (pgm->read_byte(pgm, p, m, 0, &value) != 0)
|
|
{
|
|
allowfuseread = 0;
|
|
}
|
|
if (verbose > 2)
|
|
{
|
|
fprintf(stderr, "%s: safemode read 3, efuse value: %x\n",progname, value);
|
|
}
|
|
if (value == safemode_efuse){
|
|
fusegood = 1; /* Fuse read OK three times */
|
|
}
|
|
}
|
|
}
|
|
|
|
//Programmer does not allow fuse reading.... no point trying anymore
|
|
if (allowfuseread == 0)
|
|
{
|
|
return -5;
|
|
}
|
|
|
|
if (fusegood == 0) {
|
|
fprintf(stderr,
|
|
"%s: safemode: Verify error - unable to read efuse properly. "
|
|
"Programmer may not be reliable.\n", progname);
|
|
return -3;
|
|
}
|
|
else if ((fusegood == 1) && (verbose > 0)) {
|
|
fprintf(stderr, "%s: safemode: efuse reads as %X\n", progname, safemode_efuse);
|
|
}
|
|
|
|
*lfuse = safemode_lfuse;
|
|
*hfuse = safemode_hfuse;
|
|
*efuse = safemode_efuse;
|
|
*fuse = safemode_fuse;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* This routine will store the current values pointed to by lfuse,
|
|
* hfuse, and efuse into an internal buffer in this routine when save
|
|
* is set to 1. When save is 0 (or not 1 really) it will copy the
|
|
* values from the internal buffer into the locations pointed to be
|
|
* lfuse, hfuse, and efuse. This allows you to change the fuse bits if
|
|
* needed from another routine (ie: have it so if user requests fuse
|
|
* bits are changed, the requested value is now verified
|
|
*/
|
|
int safemode_memfuses (int save, unsigned char * lfuse, unsigned char * hfuse,
|
|
unsigned char * efuse, unsigned char * fuse)
|
|
{
|
|
static unsigned char safemode_lfuse = 0xff;
|
|
static unsigned char safemode_hfuse = 0xff;
|
|
static unsigned char safemode_efuse = 0xff;
|
|
static unsigned char safemode_fuse = 0xff;
|
|
|
|
switch (save) {
|
|
|
|
/* Save the fuses as safemode setting */
|
|
case 1:
|
|
safemode_lfuse = *lfuse;
|
|
safemode_hfuse = *hfuse;
|
|
safemode_efuse = *efuse;
|
|
safemode_fuse = *fuse;
|
|
|
|
break;
|
|
/* Read back the fuses */
|
|
default:
|
|
*lfuse = safemode_lfuse;
|
|
*hfuse = safemode_hfuse;
|
|
*efuse = safemode_efuse;
|
|
*fuse = safemode_fuse;
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|