Copy 5.11.0 code over to branch.
git-svn-id: svn://svn.savannah.nongnu.org/avrdude/branches/RELEASE_5_11@1011 81a1dc3b-b13d-400b-aceb-764788c761c2
This commit is contained in:
parent
43e9133063
commit
1339ff76ee
|
@ -0,0 +1,33 @@
|
||||||
|
*.diff
|
||||||
|
*.patch
|
||||||
|
y.output
|
||||||
|
y.tab.h
|
||||||
|
lexer.c
|
||||||
|
config_gram.c
|
||||||
|
config_gram.h
|
||||||
|
.cvsignore
|
||||||
|
.depend
|
||||||
|
.deps
|
||||||
|
INSTALL
|
||||||
|
Makefile.in
|
||||||
|
Makefile
|
||||||
|
ac_cfg.h.in
|
||||||
|
aclocal.m4
|
||||||
|
autom4te.cache
|
||||||
|
configure
|
||||||
|
depcomp
|
||||||
|
install-sh
|
||||||
|
compile
|
||||||
|
missing
|
||||||
|
mkinstalldirs
|
||||||
|
stamp-h.in
|
||||||
|
stamp-h1
|
||||||
|
ac_cfg.h
|
||||||
|
avrdude.conf
|
||||||
|
avrdude.conf.tmp
|
||||||
|
avrdude.spec
|
||||||
|
config.guess
|
||||||
|
config.log
|
||||||
|
config.status
|
||||||
|
config.sub
|
||||||
|
avrdude
|
|
@ -0,0 +1,26 @@
|
||||||
|
AVRDUDE was written by:
|
||||||
|
|
||||||
|
Brian S. Dean <bsd@bdmicro.com>
|
||||||
|
|
||||||
|
Contributors:
|
||||||
|
|
||||||
|
Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
Eric Weddington <ericw@evcohs.com>
|
||||||
|
Jan-Hinnerk Reichert <hinni@despammed.com>
|
||||||
|
Alex Shepherd <maillists@ajsystems.co.nz>
|
||||||
|
Martin Thomas <mthomas@rhrk.uni-kl.de>
|
||||||
|
Theodore A. Roth <troth@openavr.org>
|
||||||
|
Michael Holzt <kju-avr@fqdn.org>
|
||||||
|
Colin O'Flynn <coflynn@newae.com>
|
||||||
|
Thomas Fischl <tfischl@gmx.de>
|
||||||
|
David Hoerl <dhoerl@mac.com>
|
||||||
|
Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
Darell Tan <darell.tan@gmail.com>
|
||||||
|
Wolfgang Moser
|
||||||
|
Ville Voipio
|
||||||
|
Hannes Weisbach
|
||||||
|
Doug Springer
|
||||||
|
Brett Hagman <bhagman@roguerobotics.com>
|
||||||
|
|
||||||
|
For minor contributions, please see the ChangeLog files.
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
$Id$
|
||||||
|
|
||||||
|
How to build avrdude from SVN:
|
||||||
|
|
||||||
|
1. svn co svn://svn.savannah.nongnu.org/avrdude/trunk
|
||||||
|
|
||||||
|
2. cd trunk/avrdude
|
||||||
|
|
||||||
|
3. ./bootstrap
|
||||||
|
|
||||||
|
4. ./configure
|
||||||
|
|
||||||
|
5. make
|
|
@ -0,0 +1,340 @@
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 2, June 1991
|
||||||
|
|
||||||
|
Copyright (C) 1989, 1991 Free Software Foundation, Inc.
|
||||||
|
59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
License is intended to guarantee your freedom to share and change free
|
||||||
|
software--to make sure the software is free for all its users. This
|
||||||
|
General Public License applies to most of the Free Software
|
||||||
|
Foundation's software and to any other program whose authors commit to
|
||||||
|
using it. (Some other Free Software Foundation software is covered by
|
||||||
|
the GNU Library General Public License instead.) You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
this service if you wish), that you receive source code or can get it
|
||||||
|
if you want it, that you can change the software or use pieces of it
|
||||||
|
in new free programs; and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
anyone to deny you these rights or to ask you to surrender the rights.
|
||||||
|
These restrictions translate to certain responsibilities for you if you
|
||||||
|
distribute copies of the software, or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must give the recipients all the rights that
|
||||||
|
you have. You must make sure that they, too, receive or can get the
|
||||||
|
source code. And you must show them these terms so they know their
|
||||||
|
rights.
|
||||||
|
|
||||||
|
We protect your rights with two steps: (1) copyright the software, and
|
||||||
|
(2) offer you this license which gives you legal permission to copy,
|
||||||
|
distribute and/or modify the software.
|
||||||
|
|
||||||
|
Also, for each author's protection and ours, we want to make certain
|
||||||
|
that everyone understands that there is no warranty for this free
|
||||||
|
software. If the software is modified by someone else and passed on, we
|
||||||
|
want its recipients to know that what they have is not the original, so
|
||||||
|
that any problems introduced by others will not reflect on the original
|
||||||
|
authors' reputations.
|
||||||
|
|
||||||
|
Finally, any free program is threatened constantly by software
|
||||||
|
patents. We wish to avoid the danger that redistributors of a free
|
||||||
|
program will individually obtain patent licenses, in effect making the
|
||||||
|
program proprietary. To prevent this, we have made it clear that any
|
||||||
|
patent must be licensed for everyone's free use or not licensed at all.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License applies to any program or other work which contains
|
||||||
|
a notice placed by the copyright holder saying it may be distributed
|
||||||
|
under the terms of this General Public License. The "Program", below,
|
||||||
|
refers to any such program or work, and a "work based on the Program"
|
||||||
|
means either the Program or any derivative work under copyright law:
|
||||||
|
that is to say, a work containing the Program or a portion of it,
|
||||||
|
either verbatim or with modifications and/or translated into another
|
||||||
|
language. (Hereinafter, translation is included without limitation in
|
||||||
|
the term "modification".) Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running the Program is not restricted, and the output from the Program
|
||||||
|
is covered only if its contents constitute a work based on the
|
||||||
|
Program (independent of having been made by running the Program).
|
||||||
|
Whether that is true depends on what the Program does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Program's
|
||||||
|
source code as you receive it, in any medium, provided that you
|
||||||
|
conspicuously and appropriately publish on each copy an appropriate
|
||||||
|
copyright notice and disclaimer of warranty; keep intact all the
|
||||||
|
notices that refer to this License and to the absence of any warranty;
|
||||||
|
and give any other recipients of the Program a copy of this License
|
||||||
|
along with the Program.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy, and
|
||||||
|
you may at your option offer warranty protection in exchange for a fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Program or any portion
|
||||||
|
of it, thus forming a work based on the Program, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) You must cause the modified files to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
b) You must cause any work that you distribute or publish, that in
|
||||||
|
whole or in part contains or is derived from the Program or any
|
||||||
|
part thereof, to be licensed as a whole at no charge to all third
|
||||||
|
parties under the terms of this License.
|
||||||
|
|
||||||
|
c) If the modified program normally reads commands interactively
|
||||||
|
when run, you must cause it, when started running for such
|
||||||
|
interactive use in the most ordinary way, to print or display an
|
||||||
|
announcement including an appropriate copyright notice and a
|
||||||
|
notice that there is no warranty (or else, saying that you provide
|
||||||
|
a warranty) and that users may redistribute the program under
|
||||||
|
these conditions, and telling the user how to view a copy of this
|
||||||
|
License. (Exception: if the Program itself is interactive but
|
||||||
|
does not normally print such an announcement, your work based on
|
||||||
|
the Program is not required to print an announcement.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Program,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Program, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Program.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Program
|
||||||
|
with the Program (or with a work based on the Program) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may copy and distribute the Program (or a work based on it,
|
||||||
|
under Section 2) in object code or executable form under the terms of
|
||||||
|
Sections 1 and 2 above provided that you also do one of the following:
|
||||||
|
|
||||||
|
a) Accompany it with the complete corresponding machine-readable
|
||||||
|
source code, which must be distributed under the terms of Sections
|
||||||
|
1 and 2 above on a medium customarily used for software interchange; or,
|
||||||
|
|
||||||
|
b) Accompany it with a written offer, valid for at least three
|
||||||
|
years, to give any third party, for a charge no more than your
|
||||||
|
cost of physically performing source distribution, a complete
|
||||||
|
machine-readable copy of the corresponding source code, to be
|
||||||
|
distributed under the terms of Sections 1 and 2 above on a medium
|
||||||
|
customarily used for software interchange; or,
|
||||||
|
|
||||||
|
c) Accompany it with the information you received as to the offer
|
||||||
|
to distribute corresponding source code. (This alternative is
|
||||||
|
allowed only for noncommercial distribution and only if you
|
||||||
|
received the program in object code or executable form with such
|
||||||
|
an offer, in accord with Subsection b above.)
|
||||||
|
|
||||||
|
The source code for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For an executable work, complete source
|
||||||
|
code means all the source code for all modules it contains, plus any
|
||||||
|
associated interface definition files, plus the scripts used to
|
||||||
|
control compilation and installation of the executable. However, as a
|
||||||
|
special exception, the source code distributed need not include
|
||||||
|
anything that is normally distributed (in either source or binary
|
||||||
|
form) with the major components (compiler, kernel, and so on) of the
|
||||||
|
operating system on which the executable runs, unless that component
|
||||||
|
itself accompanies the executable.
|
||||||
|
|
||||||
|
If distribution of executable or object code is made by offering
|
||||||
|
access to copy from a designated place, then offering equivalent
|
||||||
|
access to copy the source code from the same place counts as
|
||||||
|
distribution of the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
4. You may not copy, modify, sublicense, or distribute the Program
|
||||||
|
except as expressly provided under this License. Any attempt
|
||||||
|
otherwise to copy, modify, sublicense or distribute the Program is
|
||||||
|
void, and will automatically terminate your rights under this License.
|
||||||
|
However, parties who have received copies, or rights, from you under
|
||||||
|
this License will not have their licenses terminated so long as such
|
||||||
|
parties remain in full compliance.
|
||||||
|
|
||||||
|
5. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Program or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Program (or any work based on the
|
||||||
|
Program), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Program or works based on it.
|
||||||
|
|
||||||
|
6. Each time you redistribute the Program (or any work based on the
|
||||||
|
Program), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute or modify the Program subject to
|
||||||
|
these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties to
|
||||||
|
this License.
|
||||||
|
|
||||||
|
7. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Program at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Program by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Program.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under
|
||||||
|
any particular circumstance, the balance of the section is intended to
|
||||||
|
apply and the section as a whole is intended to apply in other
|
||||||
|
circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system, which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
8. If the distribution and/or use of the Program is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Program under this License
|
||||||
|
may add an explicit geographical distribution limitation excluding
|
||||||
|
those countries, so that distribution is permitted only in or among
|
||||||
|
countries not thus excluded. In such case, this License incorporates
|
||||||
|
the limitation as if written in the body of this License.
|
||||||
|
|
||||||
|
9. The Free Software Foundation may publish revised and/or new versions
|
||||||
|
of the General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Program
|
||||||
|
specifies a version number of this License which applies to it and "any
|
||||||
|
later version", you have the option of following the terms and conditions
|
||||||
|
either of that version or of any later version published by the Free
|
||||||
|
Software Foundation. If the Program does not specify a version number of
|
||||||
|
this License, you may choose any version ever published by the Free Software
|
||||||
|
Foundation.
|
||||||
|
|
||||||
|
10. If you wish to incorporate parts of the Program into other free
|
||||||
|
programs whose distribution conditions are different, write to the author
|
||||||
|
to ask for permission. For software which is copyrighted by the Free
|
||||||
|
Software Foundation, write to the Free Software Foundation; we sometimes
|
||||||
|
make exceptions for this. Our decision will be guided by the two goals
|
||||||
|
of preserving the free status of all derivatives of our free software and
|
||||||
|
of promoting the sharing and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
|
||||||
|
FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
|
||||||
|
OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
|
||||||
|
PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
|
||||||
|
OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
|
||||||
|
TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
|
||||||
|
PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
|
||||||
|
REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
|
||||||
|
REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
|
||||||
|
INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
|
||||||
|
OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
|
||||||
|
TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
|
||||||
|
YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
|
||||||
|
PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program is interactive, make it output a short notice like this
|
||||||
|
when it starts in an interactive mode:
|
||||||
|
|
||||||
|
Gnomovision version 69, Copyright (C) year name of author
|
||||||
|
Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, the commands you use may
|
||||||
|
be called something other than `show w' and `show c'; they could even be
|
||||||
|
mouse-clicks or menu items--whatever suits your program.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the program, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the program
|
||||||
|
`Gnomovision' (which makes passes at compilers) written by James Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1989
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
This General Public License does not permit incorporating your program into
|
||||||
|
proprietary programs. If your program is a subroutine library, you may
|
||||||
|
consider it more useful to permit linking proprietary applications with the
|
||||||
|
library. If this is what you want to do, use the GNU Library General
|
||||||
|
Public License instead of this License.
|
|
@ -0,0 +1,272 @@
|
||||||
|
2011-08-27 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Bump version for releasing AVRDUDE 5.11.
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.1: Update the list of supported AVR devices.
|
||||||
|
* doc/avrdude.texi: (Ditto).
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: add -lusb as "other libraries" when checking
|
||||||
|
for libftdi.
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Juergen Weigert:
|
||||||
|
patch #7056: adding support for mikrokopter bootloader to butterfly
|
||||||
|
* butterfly.c: Add some specific logic to handle the
|
||||||
|
mikrokopter.de butterfly bootloader.
|
||||||
|
* butterfly.h: Add one related function declaration.
|
||||||
|
* config_gram.y: Add butterfly_mk keyword.
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* avrdude.conf.in: Add entry for butterfly_mk.
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Stefan Tomanek:
|
||||||
|
patch #7542: add default_bitclock to configuration files
|
||||||
|
* config.c: Add the new keyword and its handling.
|
||||||
|
* config.h: (Ditto.)
|
||||||
|
* config_gram.y: (Ditto.)
|
||||||
|
* avrdude.conf.in: (Ditto.)
|
||||||
|
* main.c: (Ditto.)
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* avrdude.1: Document the change.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Brett Hagman:
|
||||||
|
patch #7603: wiring - programmer type for Wiring boards
|
||||||
|
(based on STK500v2)
|
||||||
|
* wiring.c: New file.
|
||||||
|
* wiring.h: (Ditto.)
|
||||||
|
* Makefile.am: Add new files.
|
||||||
|
* stk500v2_private.h: Reorganize so some functions and struct
|
||||||
|
pdata are globally known.
|
||||||
|
* stk500v2.c: (Ditto.)
|
||||||
|
* stk500v2.h: (Ditto.)
|
||||||
|
* lexer.l: Add new programmer keywords.
|
||||||
|
* config_gram.y: (Ditto.)
|
||||||
|
* avrdude.conf.in: Add "wiring" programmer entry.
|
||||||
|
* avrdude.1: Document the new programmer.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
* AUTHORS: Add Brett Hagman.
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by an anonymous contributor on the mailinglist:
|
||||||
|
* avrdude.conf (jtagkey): Add a definition for the Amontec
|
||||||
|
JTAGKey
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Juergen Weigert:
|
||||||
|
bug #22720: avrdude-5.5 ignores buff settings in avrdude.conf
|
||||||
|
(Note that the actual bug the subject is about has been fixed
|
||||||
|
long ago.)
|
||||||
|
* update.c (do_op): fix a diagnostic message
|
||||||
|
* pgm.h: add exit_datahigh field
|
||||||
|
* par.c: set and act upon the exit_datahigh field
|
||||||
|
* avrdude.1: document the new -E options
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #33811: Parallel make fails
|
||||||
|
* Makefile.am (BUILT_SOURCES): Add this macro.
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #33114: Segfault after setting the DWEN fuse with Dragon
|
||||||
|
* jtagII.c (jtagmkII_getsync): Instead of exit()ing from
|
||||||
|
deep within the tree when detecting the "need debugWIRE"
|
||||||
|
situation, properly pass this up as a return code.
|
||||||
|
* jtagII_private.h (JTAGII_GETSYNC_FAIL_GRACEFUL): New constant.
|
||||||
|
* stk500v2.c (stk500v2_jtagmkII_open): Don't tell anything
|
||||||
|
anymore when receiving a JTAGII_GETSYNC_FAIL_GRACEFUL from
|
||||||
|
jtagmkII_getsync(); silently give up (all necessary has been
|
||||||
|
said already).
|
||||||
|
|
||||||
|
2011-08-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Reported by Jason Hecker:
|
||||||
|
* usbasp.c (libusb_to_errno): Conditionalize some error codes
|
||||||
|
that apparently are lacking on MinGW.
|
||||||
|
|
||||||
|
2011-08-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Fix warnings.
|
||||||
|
* ser_avrdoper.c: add <stdlib.h> so exit() is declared.
|
||||||
|
* usbtiny.c (usbtiny_open): provide an initializer to a
|
||||||
|
"may be used uninitialized" variable (since GCC could not
|
||||||
|
fully detect the logic behind).
|
||||||
|
|
||||||
|
2011-08-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Add a check for FreeBSD's libusb-1.0
|
||||||
|
compatible library that is found in libusb.a/.so on
|
||||||
|
FreeBSD 8+.
|
||||||
|
|
||||||
|
2011-08-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Doug Springer, based on work by
|
||||||
|
Wolfgang Moser, Ville Voipio, Hannes Weisbach
|
||||||
|
patch #7486: Patch to add FT2232C/D, FT2232H, FT4232H,
|
||||||
|
usbvid, usbpid, usbdev for USB support - Based on #7062
|
||||||
|
* avrftdi.c: New file.
|
||||||
|
* avrftdi.h: (Ditto.)
|
||||||
|
* configure.ac: Add check for libftdi.
|
||||||
|
* config_gram.y: Add AVRFTDI and per-programmer USB string
|
||||||
|
keywords.
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* avrdude.conf.in: Add avrftdi and 2232HIO programmers.
|
||||||
|
* pgm.h: Add USB parameters.
|
||||||
|
* Makefile.am: Add avrftdi.c and avrftdi.h.
|
||||||
|
* AUTHORS: Mention the new authors.
|
||||||
|
* avrdude.1: Document the changes.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2011-08-23 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #29585: Fix license
|
||||||
|
* doc/avrdude.texi: Add FDL as an option to the licensing
|
||||||
|
statement, as the savannah administration would like it
|
||||||
|
that way.
|
||||||
|
|
||||||
|
2011-08-23 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Darell Tan:
|
||||||
|
patch #7244: TPI bitbang implementation
|
||||||
|
* bitbang.c: Add TPI bitbang stuff.
|
||||||
|
* bitbang.h: (Ditto.)
|
||||||
|
* avr.c: (Ditto.)
|
||||||
|
* avr.h: (Ditto.)
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
* serbb_posix.c: Wire bitbang_cmd_tpi into the struct pgm.
|
||||||
|
* serbb_win32.c: (Ditto.)
|
||||||
|
* par.c: (Ditto.)
|
||||||
|
* doc/avrdude.texi: Document the TPI bitbang support.
|
||||||
|
|
||||||
|
2011-08-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Grygoriy Fuchedzhy:
|
||||||
|
bug #31779: Add support for addressing usbtinyisp with -P option
|
||||||
|
* usbtiny.c (usbtiny_open): Add logic to distinguish multiple USBtinyISP
|
||||||
|
programmers by their bus:device tuple.
|
||||||
|
* doc/avrdude.texi: Document the new functionality.
|
||||||
|
* avrdude.1: (Ditto.)
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Timon Van Overveldt:
|
||||||
|
bug #30268: Debugwire broken in avrdude-5.10
|
||||||
|
* jtagmkII.c (jtagmkII_initialize): only try setting up a JTAG chain when
|
||||||
|
the programmer is using JTAG.
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #29636: AVRDude issues invalid CMD_CHECK_TARGET_CONNECTION
|
||||||
|
on the AVRISP-MKII
|
||||||
|
* stk500v2.c (stk500v2_program_enable): Rewrite the logic to
|
||||||
|
explain ISP activation failures.
|
||||||
|
* stk500v2_private.h: Fix the various STATUS_* constants;
|
||||||
|
AVR069 and AVR079 disagreed in their values, even though they
|
||||||
|
are apparently implementing the same logic behind.
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #29650: Programming timeouts in ATmega128RFA1 are too slow
|
||||||
|
* avrdude.conf.in (ATmega128RFA1): Bump write delay values for flash and
|
||||||
|
EEPROM to 50 ms.
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in (ATmega8515, ATmega8535, ATmega48, ATmega88, ATmega88P,
|
||||||
|
ATtiny88, ATmega168, ATmega168P, ATmega328P): Bump delay value for STK500v2
|
||||||
|
EEPROM write operation to 5, according to the respective XML files.
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Darcy Houlahan:
|
||||||
|
bug #29694: error in avrdude.conf for attiny84 eeprom
|
||||||
|
* avrdude.conf.in (ATtiny84, ATtiny85): fix A7 bit in EEPROM write
|
||||||
|
command.
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Durant Gilles:
|
||||||
|
* avrdude.conf.in (ATtiny4313): Fix flash addressing bits for manual ISP
|
||||||
|
algorithm.
|
||||||
|
|
||||||
|
2011-08-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Philip:
|
||||||
|
bug #31386: A "BUILD.svn" or similar "how to get started" doc would be helpful
|
||||||
|
* BUILD-FROM-SVN: New file.
|
||||||
|
|
||||||
|
2011-08-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Nic Jones:
|
||||||
|
bug #32539: [Documentation][Patch] Man page is misleading
|
||||||
|
re: Dragon & PDI
|
||||||
|
* doc/avrdude.texi: Update information about PDI connections
|
||||||
|
on AVR Dragon
|
||||||
|
|
||||||
|
2011-08-12 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* usbasp.c: Add <stdint.h> so this actually compiles
|
||||||
|
again.
|
||||||
|
|
||||||
|
2011-08-12 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by tixiv@gmx.net:
|
||||||
|
bug #33345: File auto detection as binary doesn't open
|
||||||
|
file in binary mode on Windows
|
||||||
|
* fileio.c: Move the decision about opening files in
|
||||||
|
binary mode until before the fopen() call.
|
||||||
|
|
||||||
|
2011-06-16 Thomas Fischl <tfischl@gmx.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Fix part id of ATtiny9.
|
||||||
|
|
||||||
|
2011-05-28 Thomas Fischl <tfischl@gmx.de>
|
||||||
|
|
||||||
|
Based on patch #7440 commited by Slawomir Fraś:
|
||||||
|
* usbasp.c: added TPI support for USBasp
|
||||||
|
* usbasp.h: (Ditto.)
|
||||||
|
|
||||||
|
2011-05-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Add support for ATmega168P.
|
||||||
|
|
||||||
|
2011-05-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Fix abbreviated name for ATmega324PA.
|
||||||
|
|
||||||
|
2011-05-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Lech Perczak:
|
||||||
|
bug #30946: Added support for ATmega8/16/32U2
|
||||||
|
* avrdude.conf.in: Add ATmega8/16/32U2 entries.
|
||||||
|
|
||||||
|
2011-05-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by David A Lyons:
|
||||||
|
patch #7393: Adding ATtiny4313 Device to avrdude.conf.in
|
||||||
|
* avrdude.conf.in: Add ATtiny4313 data.
|
||||||
|
|
||||||
|
2011-05-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* usb_libusb.c: Bump timeout values to allow for slow clock
|
||||||
|
speeds.
|
||||||
|
* jtagmkII.c: (Ditto.)
|
||||||
|
|
||||||
|
2011-03-04 Eric B. Weddington <eric.weddington@atmel.com>
|
||||||
|
|
||||||
|
Thanks to Vitaly Chernookiy for the patch.
|
||||||
|
* avrdude.conf.in: Add support for atmega324pa.
|
||||||
|
* ChangeLog-2010: New file, rotate ChangeLog for new year.
|
|
@ -0,0 +1,598 @@
|
||||||
|
2001-12-30 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c: Update version.
|
||||||
|
|
||||||
|
* avrdude.conf.sample: Clarify a comment.
|
||||||
|
|
||||||
|
* avrdude.conf.sample: fix address bits
|
||||||
|
|
||||||
|
* avrdude.1: Bring up to date.
|
||||||
|
|
||||||
|
2001-12-29 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample: Add the AVR3 progammer.
|
||||||
|
|
||||||
|
* avr.c, avrdude.conf.sample, config_gram.y, main.c, pindefs.h:
|
||||||
|
Fix VCC assertion.
|
||||||
|
|
||||||
|
Make the BUFF pin a mask like VCC to allow multiple pins to be
|
||||||
|
asserted at the same time (STK200 has two buffer enable lines).
|
||||||
|
|
||||||
|
Add the STK200 programmer.
|
||||||
|
|
||||||
|
Fix EEPROM address line selection for several parts.
|
||||||
|
|
||||||
|
2001-12-15 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample: fix spelling error
|
||||||
|
|
||||||
|
2001-11-24 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile:
|
||||||
|
Change "WARNING" to "NOTE" when overwriting the avrprog.conf file.
|
||||||
|
|
||||||
|
* avrdude.1: Add my e-mail address.
|
||||||
|
|
||||||
|
* avrdude.conf.sample:
|
||||||
|
Add comments about instruction formats. Correct an instruction
|
||||||
|
specification (cut&paste error).
|
||||||
|
|
||||||
|
2001-11-21 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, config_gram.y, lexer.l, term.c:
|
||||||
|
In interactive mode, reset the address and length if we start dumping
|
||||||
|
a memory type different than the previous one.
|
||||||
|
|
||||||
|
* avr.c, avrdude.conf.sample, config_gram.y:
|
||||||
|
Allow instruction data to be specified more flexibly, which can be
|
||||||
|
used to make the instruction input more readable in the config file.
|
||||||
|
|
||||||
|
* main.c: Bump version number.
|
||||||
|
|
||||||
|
* Makefile, avr.c, avr.h, avrdude.conf.sample, config.c, config.h:
|
||||||
|
* config_gram.y, fileio.c, fileio.h, lexer.l, main.c, term.c:
|
||||||
|
This is a major re-write of the programming algorithms. The Atmel
|
||||||
|
serial programming instructions are not very orthoganal, i.e., the
|
||||||
|
"read fuse bits" instruction on an ATMega103 is an entirely different
|
||||||
|
opcode and data format from the _same_ instruction for an ATMega163!
|
||||||
|
Thus, it becomes impossible to have a single instruction encoding
|
||||||
|
(varying the data) across the chip lines.
|
||||||
|
|
||||||
|
This set of changes allows and requires instruction encodings to be
|
||||||
|
defined on a per-part basis within the configuration file. Hopefully
|
||||||
|
I've defined the encoding scheme in a general enough way so it is
|
||||||
|
useful in describing the instruction formats for yet-to-be invented
|
||||||
|
Atmel chips. I've tried hard to make it match very closely with the
|
||||||
|
specification in Atmel's data sheets for their parts. It's a little
|
||||||
|
more verbose than what I initially hoped for, but I've tried to keep
|
||||||
|
it as concise as I could, while still remaining reasonably flexible.
|
||||||
|
|
||||||
|
2001-11-19 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, avr.h, avrdude.conf.sample, main.c, ppi.c, term.c:
|
||||||
|
Add support for ATMega163.
|
||||||
|
|
||||||
|
Add support for reading/writing ATMega163 lock and fuse bits.
|
||||||
|
Unfortunately, in looking at the specs for other ATMega parts, they
|
||||||
|
use entirely different instruction formats for these commands. Thus,
|
||||||
|
these routines won't work for the ATMega103, for example.
|
||||||
|
|
||||||
|
Add support for sending raw command bytes via the interactive terminal
|
||||||
|
interface. This allows one to execute any programming instruction on
|
||||||
|
the target device, whether or not avrprog supports it explicitly or
|
||||||
|
not. Thus, one can use this feature to program fuse / lock bits, or
|
||||||
|
access any other feature of a current or future device that avrprog
|
||||||
|
does not know how to do.
|
||||||
|
|
||||||
|
Add in comments, an experimental instruction format in the
|
||||||
|
configuration file. If this works out, it would allow supporting new
|
||||||
|
parts and non-orthoganal instructions across existing parts without
|
||||||
|
making avrprog code changes.
|
||||||
|
|
||||||
|
2001-11-17 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample: Add ATMEGA163 part.
|
||||||
|
|
||||||
|
2001-11-11 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c: output formatting
|
||||||
|
|
||||||
|
2001-11-05 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* ppi.c: Get ppi.h from /usr/include, not /sys.
|
||||||
|
|
||||||
|
2001-10-31 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, avrdude.conf.sample, main.c: Correct version string.
|
||||||
|
Update read/write status more frequently.
|
||||||
|
Prefix ATMega parts with an 'm'.
|
||||||
|
|
||||||
|
2001-10-16 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c: Change ording for memory display.
|
||||||
|
|
||||||
|
* config_gram.y: comment
|
||||||
|
|
||||||
|
* avr.c, avr.h, avrdude.conf.sample, config_gram.y, lexer.l, term.c:
|
||||||
|
Fix (again, hopefully) page addressing for the ATMega parts.
|
||||||
|
|
||||||
|
Rename the poorly chosen name "bank" to "page" for page addressing.
|
||||||
|
Atmel calls it "page" in their documentation.
|
||||||
|
|
||||||
|
* config_gram.y, main.c: Fix an (non)exit.
|
||||||
|
Silence a couple of compiler warnings.
|
||||||
|
|
||||||
|
* avr.c, avr.h, avrdude.conf.sample, config_gram.y, main.c:
|
||||||
|
Fix ATMega flash addressing. Add an ATMEGA16 part. Perform sanity
|
||||||
|
checking on the memory parameters for parts that do bank addressing.
|
||||||
|
|
||||||
|
2001-10-15 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* config.c, config.h, lists.h: Add copyright.
|
||||||
|
|
||||||
|
* config_gram.y, lexer.l, lists.c: Add copyrights.
|
||||||
|
|
||||||
|
* Makefile: Attempt to install avrprog.conf.
|
||||||
|
|
||||||
|
* avrdude.conf.sample: Correct dt006 pinout.
|
||||||
|
|
||||||
|
* Makefile, lexer.l:
|
||||||
|
Try and detect an old-style config file and print an appropriate error
|
||||||
|
message and a suggestion for correcting it.
|
||||||
|
|
||||||
|
* Makefile, avr.c, avrdude.1, avrdude.conf.sample: Update the man page.
|
||||||
|
|
||||||
|
Miscellaneous minor cleanups.
|
||||||
|
|
||||||
|
2001-10-14 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, Makefile.inc, avr.c, avr.h, avrdude.conf.sample:
|
||||||
|
* config.c, config.h, config_gram.y, lexer.l, lists.c, lists.h:
|
||||||
|
* main.c, pindefs.h, term.c:
|
||||||
|
Use lex/yacc for parsing the config file. Re-work the config file
|
||||||
|
format using a more human-readable format.
|
||||||
|
|
||||||
|
Read part descriptions from the config file now instead of hard-coding
|
||||||
|
them.
|
||||||
|
|
||||||
|
Update usage().
|
||||||
|
|
||||||
|
Cleanup unused code.
|
||||||
|
|
||||||
|
* Makefile, avr.c, avr.h, fileio.c, term.c:
|
||||||
|
First cut at supporting the ATmega 103 which uses bank addressing and
|
||||||
|
has a 128K flash.
|
||||||
|
|
||||||
|
Due to the bank addressing required, interactive update of the flash
|
||||||
|
is not supported, though the eeprom can be updated interactively.
|
||||||
|
Both memories can be programmed via non-interactive mode.
|
||||||
|
|
||||||
|
Intel Hex Record type '04' is now generated as required for outputing
|
||||||
|
memory contents that go beyond 64K.
|
||||||
|
|
||||||
|
2001-10-13 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, avr.h, fileio.c, fileio.h, main.c, ppi.c, ppi.h, term.c:
|
||||||
|
* term.h:
|
||||||
|
Style fixes.
|
||||||
|
|
||||||
|
* avr.c, avr.h, fileio.c, fileio.h, main.c, term.c:
|
||||||
|
Commit changes in preparation for support the ATMega line.
|
||||||
|
|
||||||
|
2001-10-01 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile: Don't override CFLAGS.
|
||||||
|
|
||||||
|
* avrdude.1: Correct default pin assignment.
|
||||||
|
|
||||||
|
* avr.c, fileio.c, main.c, ppi.c, term.c:
|
||||||
|
Remove debugging code - it served its purpose.
|
||||||
|
|
||||||
|
Update copyrights.
|
||||||
|
|
||||||
|
2001-09-21 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c:
|
||||||
|
Be sure to read the exit specs after the pin configuration has been
|
||||||
|
assigned, otherwise, we may apply the exit specs to the wrong pins.
|
||||||
|
|
||||||
|
* main.c: debugging
|
||||||
|
|
||||||
|
2001-09-20 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.1, avrdude.conf.sample, main.c:
|
||||||
|
Prefix pin config entries in the config file with a "c:". Later, I
|
||||||
|
might make part descriptions read in this way and we can use a
|
||||||
|
different letter for those (p). This will make the parsing easier to
|
||||||
|
distinguish between the entry types.
|
||||||
|
|
||||||
|
* main.c: Initialize pin configuration description.
|
||||||
|
|
||||||
|
2001-09-19 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* AVRprog.pdf, Makefile, avr.c, avrdude.1, avrdude.conf.sample:
|
||||||
|
* avrdude.pdf, fileio.c, fileio.h, main.c, pindefs.h, term.c:
|
||||||
|
Make the pin definitions configurable based on entries in a config
|
||||||
|
file. This makes supporting other programmers much easier.
|
||||||
|
|
||||||
|
Rename AVRprog.pdf to avrprog.pdf.
|
||||||
|
|
||||||
|
2001-04-29 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog-programmer.jpg: Remove this image file from the repository.
|
||||||
|
|
||||||
|
2001-04-26 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog-schematic.jpg:
|
||||||
|
Remove this image, use AVRprog.pdf as the preferred schematic for the
|
||||||
|
programmer.
|
||||||
|
|
||||||
|
2001-04-25 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* AVRprog.pdf, Makefile, avrdude.1:
|
||||||
|
Add a schematic provided by Joerg Wunch and also update the manual
|
||||||
|
page (also updated by Joerg) to reference the schematic.
|
||||||
|
|
||||||
|
2001-02-25 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, Makefile.inc: Automate dependency generation.
|
||||||
|
|
||||||
|
2001-02-08 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c: Turn off ready led when finished programming.
|
||||||
|
|
||||||
|
* main.c: update version
|
||||||
|
|
||||||
|
* avr.c, main.c: Correct a few comments.
|
||||||
|
|
||||||
|
* Makefile, avr.c, term.c: Makefile : update dependencies
|
||||||
|
|
||||||
|
avr.c : correct status led updates
|
||||||
|
|
||||||
|
term.c : update status leds on write, make the address and length
|
||||||
|
arguments for dump optional.
|
||||||
|
|
||||||
|
2001-01-26 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c: Version 1.1
|
||||||
|
|
||||||
|
* main.c:
|
||||||
|
Hmmm ... cvs co -D <timestamp> does not work. Change the revision
|
||||||
|
timestamp to a full date/time value.
|
||||||
|
|
||||||
|
* avr.c, fileio.c, main.c, ppi.c, term.c:
|
||||||
|
Add a -V option to display the version information about each
|
||||||
|
component module. This is intended for support purposes, so that I
|
||||||
|
can tell unambiguously what version a binary out in the field is.
|
||||||
|
|
||||||
|
Additionally, display a revision timestamp along with the version
|
||||||
|
number. This also is intended for aiding in support and is the Unix
|
||||||
|
time of the latest component module. Having this, should allow me to
|
||||||
|
do a "cvs co -D timestamp avrprog" and get exactly the source of the
|
||||||
|
version that is being reported.
|
||||||
|
|
||||||
|
* fileio.c:
|
||||||
|
Return the maximum address (+1) written as opposed to the actual
|
||||||
|
number of bytes written. The presence of an Intel Hex address
|
||||||
|
record can cause these two number to be different; but the callers
|
||||||
|
of this routine need the former.
|
||||||
|
|
||||||
|
* main.c:
|
||||||
|
Fix a place where we were exiting without applying the exit-specs.
|
||||||
|
|
||||||
|
Wrap a long line.
|
||||||
|
|
||||||
|
* avr.c, fileio.c: avr.c: Update a comment.
|
||||||
|
|
||||||
|
fileio.c: Properly handle all the Intel Hex record types that I can
|
||||||
|
find information about.
|
||||||
|
|
||||||
|
2001-01-25 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Usage, avr.h: Get rid of the Usage file.
|
||||||
|
|
||||||
|
2001-01-24 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, avr.c, avr.h, main.c, pindefs.h, ppi.c:
|
||||||
|
Move pin definitions to their own file.
|
||||||
|
|
||||||
|
First pass at providing feedback via the optionally connected leds. I
|
||||||
|
don't actually have any of these attached to my programmer, so I can
|
||||||
|
only guess as whether this is toggling them on and off correctly.
|
||||||
|
|
||||||
|
Also, enable and disable the optional 74367 buffer.
|
||||||
|
|
||||||
|
* avr.h, main.c, ppi.c, ppi.h, avr.c:
|
||||||
|
Rearrange the pinout for the programmer to be a little more logical.
|
||||||
|
Provide hooks to support a buffered programmer, pin 6 is now used to
|
||||||
|
enable a buffer that can be used to isolate the target system from the
|
||||||
|
parallel port pins. This is important when programming the target
|
||||||
|
in-system.
|
||||||
|
|
||||||
|
Totally change the way the pin definitions are defined. Actually
|
||||||
|
set/clear pins based on the way more intuitive pin number, instead of
|
||||||
|
PPI data register, bit number combination. A table of pin data is
|
||||||
|
used so that any hardware inversion done by the parallel port is
|
||||||
|
accounted for, what you set is actually what appears at the pin.
|
||||||
|
Retain the old method for handling Vcc, however, because the hold
|
||||||
|
method is much easier to use when setting / retrieving multiple pins
|
||||||
|
simultaneously.
|
||||||
|
|
||||||
|
2001-01-22 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile: Don't gzip the man page.
|
||||||
|
|
||||||
|
* avrdude.1: .Nm macro fix. Submitted by Joerg.
|
||||||
|
|
||||||
|
* main.c: Cosmetic, don't output a preceding linefeed for usage().
|
||||||
|
|
||||||
|
* Makefile, avr.c, avr.h, fileio.c, term.c:
|
||||||
|
Makefile : use gzip -f for man page installation so that we don't get
|
||||||
|
prompted.
|
||||||
|
|
||||||
|
avr.c avr.h fileio.c term.c :
|
||||||
|
|
||||||
|
Change the avrpart data structure so that the typedef AVRMEM is
|
||||||
|
used as an index into an array for the sizes of the memory types
|
||||||
|
and also for pointers to buffers that represent the chip data for
|
||||||
|
that memory type. This removes a lot of conditional code of the
|
||||||
|
form:
|
||||||
|
|
||||||
|
switch (memtype) {
|
||||||
|
case AVR_FLASH :
|
||||||
|
...
|
||||||
|
}
|
||||||
|
|
||||||
|
Also, re-code avr_read_byte() and avr_write_byte() to properly
|
||||||
|
handle the flash memory type without having to tell them whether
|
||||||
|
they should program the high byte or the low byte - figure that
|
||||||
|
out from the address itself. For flash memory type, these
|
||||||
|
routines now take the actual byte address instead of the word
|
||||||
|
address. This _greatly_ simplifies many otherwise simple
|
||||||
|
operations, such a reading or writing a range of memory, by not
|
||||||
|
having to worry about whether the address starts on an odd byte
|
||||||
|
or an even byte.
|
||||||
|
|
||||||
|
2001-01-20 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, avr.h, fileio.c, fileio.h, main.c:
|
||||||
|
Return error codes instead of exiting, thus making sure that we exit
|
||||||
|
only via main() so that the exitspecs are properly applied.
|
||||||
|
|
||||||
|
When reading input data from a file, remember how many bytes were read
|
||||||
|
and write and verify only that many bytes.
|
||||||
|
|
||||||
|
Don't complain when an input file size is smaller than the memory size
|
||||||
|
we are programming. This is normal.
|
||||||
|
|
||||||
|
* fileio.c:
|
||||||
|
Correct checksum calculation; failure to account for the value of the
|
||||||
|
record type was causing non-zero record types to be calculated
|
||||||
|
incorrectly.
|
||||||
|
|
||||||
|
* Makefile, main.c: Makefile : install the man page
|
||||||
|
|
||||||
|
main.c : drop the giant usage text now that we have a man page.
|
||||||
|
|
||||||
|
* avrdude.1:
|
||||||
|
Add initial man page graciously contributed by Joerg Wunsch. Thanks
|
||||||
|
Joerg!
|
||||||
|
|
||||||
|
2001-01-19 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* term.c:
|
||||||
|
Accept abbreviations for eeprom and flash for the dump and write
|
||||||
|
commands.
|
||||||
|
|
||||||
|
Fix small bug keeping 1 character command lines from being added to
|
||||||
|
the history.
|
||||||
|
|
||||||
|
* term.c:
|
||||||
|
Implement enough state in cmd_dump so that if it is called with no
|
||||||
|
arguments, it successively dumps the next chunk of data of the same
|
||||||
|
previously specified length.
|
||||||
|
|
||||||
|
* term.c, term.h, fileio.c, fileio.h, main.c, ppi.c, ppi.h:
|
||||||
|
* Makefile, avr.c, avr.h, avrprog.c:
|
||||||
|
The program was getting too large for a single file. Split it up into
|
||||||
|
more modular pieces.
|
||||||
|
|
||||||
|
Also, accept command abbreviations as long as they are not ambiguous.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Add ability to specify the state of the power and reset pins on
|
||||||
|
program exit. Default to leaving the pins in the state they were when
|
||||||
|
we found them.
|
||||||
|
|
||||||
|
Contributed by: Joerg Wunsch
|
||||||
|
|
||||||
|
2001-01-18 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, avrprog.c:
|
||||||
|
Switch to using readline() for getting terminal input. I can't seem
|
||||||
|
to get the history capabilities working yet, but even so, it does
|
||||||
|
better handling of the prompt and strips newlines for us, so it's
|
||||||
|
still a win.
|
||||||
|
|
||||||
|
Add a few new commands for terminal mode: help, sig, part, erase.
|
||||||
|
Display rudimentory help using the help command.
|
||||||
|
|
||||||
|
Add some function prototypes.
|
||||||
|
|
||||||
|
* Usage, avrprog.c:
|
||||||
|
Change -c (interactive command mode) to the more intuitive -t
|
||||||
|
(terminal mode).
|
||||||
|
|
||||||
|
Make binary format the default for output.
|
||||||
|
|
||||||
|
Update the parts table with corrections for old values and add some
|
||||||
|
new values.
|
||||||
|
|
||||||
|
2001-01-15 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Automatically verify on-chip data with what we just programmed.
|
||||||
|
|
||||||
|
* avrprog.c, Makefile:
|
||||||
|
Prepare the Makefile for integration into the FreeBSD ports tree.
|
||||||
|
|
||||||
|
Fix a few "may be used uninitialized" bugs found by -Wall.
|
||||||
|
|
||||||
|
2001-01-14 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c: Free a buffer.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Use a smarter programming algorithm - read the existing data byte
|
||||||
|
first and only write the new one if it is different.
|
||||||
|
|
||||||
|
Add -n option which is a test mode in which the chip is not actually
|
||||||
|
updated. This option does not affect writes in interactive mode.
|
||||||
|
|
||||||
|
* avrprog.c: Add the "dump" and "write" interactive commands.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Correctly produce and handle "end of record" for intel hex files.
|
||||||
|
|
||||||
|
2001-01-13 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Re-enable writing to the chip. I should probably should make this a
|
||||||
|
command-line selectable option so that I don't keep forgetting and
|
||||||
|
committing it with it disabled.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Add a newline before exiting due to command line errors. Perform a
|
||||||
|
bit more option compatibility testing between -c, -i, and -o.
|
||||||
|
|
||||||
|
* avrprog.c: Add input file format auto-detection support.
|
||||||
|
|
||||||
|
* Usage, avrprog.c: Say what the defaults are.
|
||||||
|
|
||||||
|
* avrprog-programmer.jpg, Usage, avrprog-schematic.jpg: New files.
|
||||||
|
|
||||||
|
* avrprog.c: Correct usage text.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Parameterize a few additional items per chip. Print out all per-chip
|
||||||
|
parameters on startup. Use the per-chip parameters in the code
|
||||||
|
instead of hard-coded values for the 2313.
|
||||||
|
|
||||||
|
* avrprog.c: Fix filename assignment error.
|
||||||
|
|
||||||
|
Clean up debugging code a little, utilize fileio() instead of making
|
||||||
|
direct calls to b2ihex().
|
||||||
|
|
||||||
|
* avrprog.c: A lot of general code cleanup.
|
||||||
|
|
||||||
|
Re-work command line options to be more intuitive.
|
||||||
|
|
||||||
|
Support Intel Hex input and output file formats. Provide hooks to
|
||||||
|
support Motorola S-Record as well.
|
||||||
|
|
||||||
|
Add a few more part-specific parameters to the avrpart structure.
|
||||||
|
|
||||||
|
Only write the flash or eeprom if the data to be written is not 0xff.
|
||||||
|
|
||||||
|
2000-12-31 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c: Update a comment.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Provide the ability to tie additionally tie pins 6-9 of the parallel
|
||||||
|
port to Vcc in order to supply more current.
|
||||||
|
|
||||||
|
Fix a typo on the size of the S1200's Flash.
|
||||||
|
|
||||||
|
Bring RESET low when programming is completed.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Correct pin connection comments. Elaborate a bit on Vcc connection.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Update after receiving some good feedback from Joerg Wunsch. We
|
||||||
|
should now be able to program AT90S1200's.
|
||||||
|
|
||||||
|
2000-12-30 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c: Don't limit eeprom addresses.
|
||||||
|
|
||||||
|
2000-12-20 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, avrprog.c:
|
||||||
|
Add support for the 8515. Make the addition for other devices easier.
|
||||||
|
|
||||||
|
2000-08-27 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Clear all bits except AVR_RESET when finished reading or programming
|
||||||
|
the Atmel device.
|
||||||
|
|
||||||
|
2000-08-07 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c: update announcement message
|
||||||
|
|
||||||
|
* avrprog.c: Update announcement message.
|
||||||
|
|
||||||
|
* avrprog.c: Return the correct return code from 'main()'.
|
||||||
|
|
||||||
|
* avrprog.c:
|
||||||
|
Add ppi_pulse() function and fix ppi_toggle() to actully toggle
|
||||||
|
instead of pulse.
|
||||||
|
|
||||||
|
Make all abnormal returns after the parallel port has been opened go
|
||||||
|
through a single exit point at the bottom of 'main()'.
|
||||||
|
|
||||||
|
2000-08-06 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, avrprog.c: Makefile: add --pedantic compiler option
|
||||||
|
|
||||||
|
avrprog.c:
|
||||||
|
|
||||||
|
Add lots of comments, move getop() variable declarations to
|
||||||
|
the top of the program.
|
||||||
|
|
||||||
|
Add a typedef name to the AVR memory type and use it for
|
||||||
|
function declarations.
|
||||||
|
|
||||||
|
Add a usleep() delay in the sense loop to avoid becoming a cpu
|
||||||
|
hog.
|
||||||
|
|
||||||
|
Print out a version string so that folks know what version of
|
||||||
|
the software they are running.
|
||||||
|
|
||||||
|
Be sure and close the parallel device and the i/o file when
|
||||||
|
terminating abnormally.
|
||||||
|
|
||||||
|
* avrprog.c: Print out version information when invoked.
|
||||||
|
|
||||||
|
* Makefile, avrprog.c: Makefile: Add an install target.
|
||||||
|
|
||||||
|
avrprog.c:
|
||||||
|
|
||||||
|
Add license.
|
||||||
|
|
||||||
|
Document the header a bit better.
|
||||||
|
|
||||||
|
Add capability to read out and display the device signature bytes.
|
||||||
|
|
||||||
|
Add capability to power the device from the parallel port.
|
||||||
|
|
||||||
|
Eliminate debug print facility.
|
||||||
|
|
||||||
|
Provide 'avr_cmd()' function.
|
||||||
|
|
||||||
|
When memory locations don't program, generate a newline so that the
|
||||||
|
information is not overwritten and lost.
|
||||||
|
|
||||||
|
Don't print out the message about needing to specify a file if the
|
||||||
|
user is not requesting an operation that requires the file.
|
||||||
|
|
||||||
|
2000-08-05 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrprog.c: Pring usage when no arguments are supplied.
|
||||||
|
|
||||||
|
* Makefile, avrprog.c: Initial check-in
|
||||||
|
|
||||||
|
* Makefile, avrprog.c: New file.
|
||||||
|
|
|
@ -0,0 +1,237 @@
|
||||||
|
2002-12-12 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c: minor cleanup
|
||||||
|
|
||||||
|
2002-12-07 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.1, main.c:
|
||||||
|
If the stk500 is being used, default to using the first serial port.
|
||||||
|
|
||||||
|
2002-12-03 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.1: Mention STK500 support.
|
||||||
|
|
||||||
|
2002-12-01 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* stk500.c: Remove unused code.
|
||||||
|
|
||||||
|
* CHANGELOG, stk500.c:
|
||||||
|
Document changes since the previous version in the CHANGELOG.
|
||||||
|
|
||||||
|
Cleanup stk500.c a bit.
|
||||||
|
|
||||||
|
* stk500.c: Fix cut and paste braino.
|
||||||
|
|
||||||
|
* avr.c, avrdude.conf.sample, main.c, pgm.h, stk500.c:
|
||||||
|
The STK500 can perform paged read/write operations even on standard
|
||||||
|
"non-paged" parts. Take advantage of that and use the faster internal
|
||||||
|
routines of the STK500 for those parts as well.
|
||||||
|
|
||||||
|
* avr.c, avr.h, avrpart.h, main.c, pgm.c, pgm.h, stk500.c:
|
||||||
|
Optimize reading and writing for the STK500 programmer if the part
|
||||||
|
supports paged reads and writes. This greatly decreases the
|
||||||
|
program/verify time from about 4.5 minutes down to about 10 seconds in
|
||||||
|
a 12K program size test case.
|
||||||
|
|
||||||
|
Print out the hardware and firmware version for the STK500 if verbose
|
||||||
|
is enabled.
|
||||||
|
|
||||||
|
* avrdude.conf.sample, avrpart.h, config_gram.y, lexer.l, pgm.h:
|
||||||
|
* ppi.c, ppi.h, stk500.c, stk500.h, stk500_private.h:
|
||||||
|
Add basic support for STK500.
|
||||||
|
|
||||||
|
2002-11-30 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample, config.c, config.h, config_gram.y, lexer.l:
|
||||||
|
* main.c, pgm.c, pgm.h, ppi.c, ppi.h, term.c, term.h, Makefile:
|
||||||
|
* avr.c, avr.h:
|
||||||
|
Seperate programmer operations out into a driver-like interface so
|
||||||
|
that programmers other than the direct parallel port connection can be
|
||||||
|
supported.
|
||||||
|
|
||||||
|
2002-11-23 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG, main.c, term.c:
|
||||||
|
term.c - when in interactive terminal mode and dumping memory using
|
||||||
|
the 'dump <memtype>' command without any address information,
|
||||||
|
and the end of memory is reached, wrap back around to zero on
|
||||||
|
the next invocation.
|
||||||
|
|
||||||
|
CHANGELOG - describe changes
|
||||||
|
|
||||||
|
main.c - update version number
|
||||||
|
|
||||||
|
* main.c:
|
||||||
|
When getting ready to initiate communications with the AVR device,
|
||||||
|
first pull /RESET low for a short period of time before enabling the
|
||||||
|
buffer chip. This sequence allows the AVR to be reset before the
|
||||||
|
buffer is enabled to avoid a short period of time where the AVR may be
|
||||||
|
driving the programming lines at the same time the programmer tries
|
||||||
|
to. Of course, if a buffer is being used, then the /RESET line from
|
||||||
|
the programmer needs to be directly connected to the AVR /RESET line
|
||||||
|
and not via the buffer chip.
|
||||||
|
|
||||||
|
2002-11-06 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG: Update changelog.
|
||||||
|
|
||||||
|
* avr.c, avr.h, main.c: Fix -Y option. Reported by Joerg Wunsch.
|
||||||
|
|
||||||
|
2002-11-01 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG, main.c: Version update and CHANGELOG entry.
|
||||||
|
|
||||||
|
* avr.c:
|
||||||
|
Be backward compatible with the 2-byte rewrite cycle counter which
|
||||||
|
appeared in version 2.1.0, but was changed to a 4 byte counter in
|
||||||
|
version 2.1.1. Reminded by Joerg Wunsch.
|
||||||
|
|
||||||
|
2002-10-29 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG, avrdude.1, main.c:
|
||||||
|
Add '-V' (no verify) flag requested by Joerg Wunsch. Update the man
|
||||||
|
page.
|
||||||
|
|
||||||
|
2002-10-13 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG, avrdude.1: Update man page and changelog.
|
||||||
|
|
||||||
|
* main.c: Update version number.
|
||||||
|
|
||||||
|
2002-10-12 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile: Remove --pedantic and -g from the compiler options.
|
||||||
|
|
||||||
|
2002-10-11 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, term.c:
|
||||||
|
Use a four byte value instead of a two byte value for the programming
|
||||||
|
cycle count stored at the end of EEPROM. It seems as though Atmel was
|
||||||
|
greatly conservative in claiming a 1000 count reliability for the
|
||||||
|
FLASH. I current have a part that has been reprogrammed 173330 times,
|
||||||
|
and counting.
|
||||||
|
|
||||||
|
Fix a compiler warning.
|
||||||
|
|
||||||
|
* avrdude.conf.sample:
|
||||||
|
Fix ATMega128 instruction encoding for reading the low and high fuse
|
||||||
|
bits. Thanks to Joerg Wunsch for tripping over this.
|
||||||
|
|
||||||
|
2002-08-01 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, avrdude.1, main.c:
|
||||||
|
Move erase-rewrite cycle increment to within the chip erase routine so
|
||||||
|
that it is tracked no matter where the erase was initiated: command
|
||||||
|
line mode or interactive mode, without code duplicaiton.
|
||||||
|
|
||||||
|
* CHANGELOG: Recent updates.
|
||||||
|
|
||||||
|
* avr.c: Eliminate unused variables.
|
||||||
|
|
||||||
|
* avr.c, avr.h, avrdude.1, fileio.c, main.c:
|
||||||
|
Implement a way of tracking how many erase-rewrite cycles a part has
|
||||||
|
undergone. This utilizes the last two bytes of EEPROM to maintain a
|
||||||
|
counter that is incremented each time the part is erased.
|
||||||
|
|
||||||
|
2002-07-27 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avr.c, main.c:
|
||||||
|
Fix a typo in a comment. Display the size of memory being written.
|
||||||
|
Display the correct memory name in an error message (previously
|
||||||
|
hardcoded).
|
||||||
|
|
||||||
|
2002-06-22 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG, avrdude.conf.sample:
|
||||||
|
Add support for ATtiny15 - contributed by Asher Hoskins
|
||||||
|
<asher@crumbly.freeserve.co.uk>
|
||||||
|
|
||||||
|
2002-04-23 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG: Say what changed.
|
||||||
|
|
||||||
|
2002-04-07 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* Makefile, avrdude.conf.sample:
|
||||||
|
Backup the config file to a timestamped name to keep from possibly
|
||||||
|
overwriting user-modified configs.
|
||||||
|
|
||||||
|
Add read/write instructions for all memory types for ATMEGA103,
|
||||||
|
ATMEGA128, ATMEGA16, and ATMEGA8.
|
||||||
|
|
||||||
|
2002-04-05 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample:
|
||||||
|
Add support for ATMEGA128; untested; requested by Jeff Gardner
|
||||||
|
<gardner@journey.com>.
|
||||||
|
|
||||||
|
2002-02-15 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample: Minor ordering.
|
||||||
|
|
||||||
|
* CHANGELOG, main.c: Update version numbers.
|
||||||
|
|
||||||
|
2002-02-14 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG: Summarize latest updates.
|
||||||
|
|
||||||
|
* avrdude.conf.sample, config_gram.y:
|
||||||
|
Make pwroff_after_write a yes/no field instead of a numeric.
|
||||||
|
|
||||||
|
* avrdude.conf.sample: Document the pwroff_after_write flag.
|
||||||
|
|
||||||
|
* avr.c: Enable the extra part verbosity when verbosity >= 3.
|
||||||
|
|
||||||
|
* avr.c, avr.h, avrdude.conf.sample, config_gram.y, lexer.l:
|
||||||
|
* main.c, term.c:
|
||||||
|
Fix error reporting by avr_write_byte().
|
||||||
|
|
||||||
|
Fix setting of status LEDs under various write-fail conditions.
|
||||||
|
|
||||||
|
Add a flag to indicate that a memory type requires the device to
|
||||||
|
possibly be powered off and back on after a write to it. This is due
|
||||||
|
to a hardware problem on some Atmel devices, see:
|
||||||
|
|
||||||
|
http://www.atmel.com/atmel/acrobat/doc1280.pdf
|
||||||
|
|
||||||
|
Add greater verbosity to the part-display code when verbose>1 to
|
||||||
|
display avrprog's encoding of the defined programming instructions.
|
||||||
|
This is primarily for debugging purposes.
|
||||||
|
|
||||||
|
|
||||||
|
Part updates:
|
||||||
|
|
||||||
|
* add the AT90S4414 part
|
||||||
|
|
||||||
|
* add fuse and lock bit access instructions for the AT90S1200,
|
||||||
|
AT90S4434, and AT90S8515.
|
||||||
|
|
||||||
|
* add the pwroff_after_write flag to the fuse bits for the AT90S2333
|
||||||
|
and AT90S4433 parts
|
||||||
|
|
||||||
|
2002-02-09 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* avrdude.conf.sample:
|
||||||
|
Updates to the 2333 and 4433 parts, contributed by Joerg Wunsh.
|
||||||
|
|
||||||
|
2002-01-18 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* CHANGELOG: Add changelog.
|
||||||
|
|
||||||
|
2002-01-12 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
|
||||||
|
* main.c: Add (c) to copyright.
|
||||||
|
|
||||||
|
* fileio.c, fileio.h, lexer.l, lists.c, lists.h, main.c:
|
||||||
|
* pindefs.h, ppi.c, ppi.h, term.c, term.h, avr.c, avr.h:
|
||||||
|
* config.c, config.h, config_gram.y:
|
||||||
|
Update version number. Update copyright.
|
||||||
|
|
||||||
|
* avrdude.1: Update copyright and add description of "default".
|
||||||
|
|
||||||
|
Submitted by: Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avr.c, term.c:
|
||||||
|
Fix programming of write-only memories (such as lock bits on the
|
||||||
|
2313).
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,364 @@
|
||||||
|
2007-11-08 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* main.c: Partially revert the line buffered output change,
|
||||||
|
and turn stderr into unbuffered output while producing the
|
||||||
|
progress report.
|
||||||
|
|
||||||
|
2007-11-07 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* main.c: Add setup and teardown hooks to the programmer
|
||||||
|
definition. If present, call the setup hook immediately after
|
||||||
|
finding the respective programmer object, and schedule the
|
||||||
|
teardown hook to be called upon exit. This allows the
|
||||||
|
programmer implementation to dynamically allocate private
|
||||||
|
programmer data.
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
* avr910.c: Convert static programmer data into dynamically
|
||||||
|
allocated data.
|
||||||
|
* butterfly.c: (Ditto.)
|
||||||
|
* jtagmkI.c: (Ditto.)
|
||||||
|
* jtagmkII.c: (Ditto.)
|
||||||
|
* stk500v2.c: (Ditto.)
|
||||||
|
* usbasp.c: (Ditto.)
|
||||||
|
* usbtiny.c: (Ditto.)
|
||||||
|
|
||||||
|
2007-11-06 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* butterfly.c: Remove the no_show_func_info() calls, as Brian
|
||||||
|
promised some 4 years ago.
|
||||||
|
|
||||||
|
2007-11-06 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* main.c: Add the -x option to pass extended parameters to
|
||||||
|
the programmer backend.
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
* jtagmkII.c: Implement the extended parameter jtagchain=
|
||||||
|
to support JTAG daisy-chains.
|
||||||
|
* avrdude.1: Document all of the above.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2007-10-30 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac (AC_INIT): Bump version for post-release.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac (AC_INIT): Bump version, releasing avrdude-5.5.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by <bikenomad@gmail.com>:
|
||||||
|
patch #5007: Patch for line-buffering of stdout and stderr
|
||||||
|
* main.c: call setvbuf() for stdout and stderr.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by <graceindustries@gmail.com>:
|
||||||
|
patch #5953: Add AT90CAN64 and AT90CAN32 to avrdude.conf
|
||||||
|
* avrdude.conf.in: Add entry for AT90CAN64 and AT90CAN32.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Wolfgang Moser:
|
||||||
|
patch #6121: ISP support for the C2N232I device (serial port
|
||||||
|
bitbanging)
|
||||||
|
* avrdude.conf.in: Add entry for c2n232i.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by <karl.yerkes@gmail.com>:
|
||||||
|
patch #6141: accept binary format immediate values
|
||||||
|
* fileio.c: Detect a 0b prefix, and call strtoul() differently
|
||||||
|
in that case.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #21076: -vvvv serial receive prints are empty in Win32 build
|
||||||
|
* ser_win32.c (ser_recv): Drop the essentially unused variable
|
||||||
|
"len", and use the variable "read" in order to track how many
|
||||||
|
bytes have just been read in.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #21145: atmega329p not recognized
|
||||||
|
* avrdude.conf.in: Add definitions for the ATmega329P/3290P.
|
||||||
|
Same as ATmega329/3290 except of the different signature.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #21152: Unable to program atmega324p with avrdude 5.4 and AVRISP
|
||||||
|
using default configuration file.
|
||||||
|
* avrdude.conf.in: Uncomment the (bogus) stk500_devcode lines for
|
||||||
|
the ATmega164P, ATmega324P, ATmega644, and ATmega644P definitions.
|
||||||
|
This only affects users of STK500v1 firmware.
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by <ladyada@gmail.com>:
|
||||||
|
Patch #6233: Add support for USBtinyISP programmer
|
||||||
|
* usbtiny.c: New file.
|
||||||
|
* usbtiny.h: (Ditto.)
|
||||||
|
* Makefile.am: Include usbtiny into the build.
|
||||||
|
* avrdude.conf.in: (Ditto.)
|
||||||
|
* config_gram.y: (Ditto.)
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* avrdude.1: Document the usbtiny support.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2007-10-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* doc/avrdude.texi: Sort list of supported programmers into
|
||||||
|
alphabetical order, add all missing programmers.
|
||||||
|
|
||||||
|
2007-07-24 Thomas Fischl <tfischl@gmx.de>
|
||||||
|
|
||||||
|
* usbasp.c: Added long addresses to support devices with more
|
||||||
|
than 64kB flash. Closes bug #20558: Long address problem with
|
||||||
|
USBasp.
|
||||||
|
|
||||||
|
2007-06-27 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* Makefile.am (EXTRA_DIST): Add ChangeLog-2004-2006.
|
||||||
|
|
||||||
|
2007-05-16 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac (AC_INIT): Bump version for post-release.
|
||||||
|
|
||||||
|
2007-05-16 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac (AC_INIT): Bump version, releasing avrdude-5.4.
|
||||||
|
|
||||||
|
2007-05-16 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Fix AVR910 devcodes. It seems that the AVR109
|
||||||
|
listing refers to "BOOT"-type code, while the standard codes are
|
||||||
|
different (usually one below).
|
||||||
|
|
||||||
|
2007-05-16 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avr.c (avr_read, avr_write): only use the paged_load and
|
||||||
|
paged_write backend functions iff the memory area in question has
|
||||||
|
a page_size != 0.
|
||||||
|
This is supposed to fix bug #19234: avrdude-5.3.1 segfaults when
|
||||||
|
stk500v1 tries to program an ATtiny15
|
||||||
|
|
||||||
|
2007-05-15 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avr910.c: Fall back to avr_{read,write}_byte_default(). Fixes
|
||||||
|
bug #18803: Fuse reading regression in avrdude 5.3.1 with avr910
|
||||||
|
programmer
|
||||||
|
|
||||||
|
2007-05-15 Colin O'Flynn <coflynn@newae.com>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Rename the ATmega164 and ATmega324 into
|
||||||
|
ATmega164P and ATmega324P, resp. Add an entry for the ATmega644P.
|
||||||
|
Fixes bug #19769: ATmega164p not recognized
|
||||||
|
|
||||||
|
2007-05-15 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* ser_posix.c (ser_send): Don't select() on the output fd before
|
||||||
|
trying to write something to the serial line. That kind of
|
||||||
|
polling isn't very useful anyway, and it seems it breaks for the
|
||||||
|
Linux CP210x USB<->RS-232 bridge driver which is certainly a bug
|
||||||
|
in the driver, but we can just avoid that bug alltogether.
|
||||||
|
|
||||||
|
2007-05-15 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Fix the STK500v2 ISP delay parameter for
|
||||||
|
ATmega640/1280/1281/2560/2561. Atmel has changed the XML
|
||||||
|
files after the initial release.
|
||||||
|
|
||||||
|
2007-05-01 Colin O'Flynn <coflynn@newae.com>
|
||||||
|
|
||||||
|
* safemode.c: -Oops - bug in verbose output. Fixed.
|
||||||
|
-Fixed handling of cases where programmer cannot read fuses (AVR910)
|
||||||
|
* main.c: -Also fixing handling of cases where programmer cannot
|
||||||
|
read fuses
|
||||||
|
This should close one or more bugs (18803, 19570)
|
||||||
|
|
||||||
|
2007-05-01 Colin O'Flynn <coflynn@newae.com>
|
||||||
|
|
||||||
|
* safemode.c: Added verbose output from safemode routines.
|
||||||
|
|
||||||
|
2007-03-25 Colin O'Flynn <coflynn@newae.com>
|
||||||
|
|
||||||
|
* stk500generic.c: Forgot to close the serial port before trying to
|
||||||
|
open it again, caused problems on Windows machines.
|
||||||
|
Closes bug #19411
|
||||||
|
|
||||||
|
2007-02-26 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Add the AT90PWM2/3B devices.
|
||||||
|
|
||||||
|
2007-02-02 Thomas Fischl <tfischl@gmx.de>
|
||||||
|
|
||||||
|
* usbasp.c: Changed return value of function usbasp_initialize to stop
|
||||||
|
avrdude on communication errors between programmer and target.
|
||||||
|
Closes bug #18581: safemode destroys fuse bits
|
||||||
|
|
||||||
|
2007-02-01 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* config_gram.y: Remove duplicate definition of token K_WRITEPAGE
|
||||||
|
|
||||||
|
2007-01-30 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* butterfly.c: Implement ATmega256x support for butterfly/avr109.
|
||||||
|
|
||||||
|
2007-01-30 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Fix subdir handling. Now finally, "make
|
||||||
|
distcheck" will include the documentation into the tarball even if
|
||||||
|
the configure had been run without the --enable-doc.
|
||||||
|
|
||||||
|
2007-01-30 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* safemode.c: Obtain progname from avrdude.h rather than trying to
|
||||||
|
roll our own (duplicate) copy of it.
|
||||||
|
* avr910.c: Constify char pointers.
|
||||||
|
* avrpart.c: (Ditto.)
|
||||||
|
* avrpart.h: (Ditto.)
|
||||||
|
* butterfly.c: (Ditto.)
|
||||||
|
* config.c: (Ditto.)
|
||||||
|
* config.h: (Ditto.)
|
||||||
|
* jtagmkI.c: (Ditto.)
|
||||||
|
* jtagmkII.c: (Ditto.)
|
||||||
|
* par.c: (Ditto.)
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
* serbb_posix.c: (Ditto.)
|
||||||
|
* serbb_win32.c: (Ditto.)
|
||||||
|
* stk500.c: (Ditto.)
|
||||||
|
* stk500v2.c: (Ditto.)
|
||||||
|
* usbasp.c: (Ditto.)
|
||||||
|
|
||||||
|
2007-01-29 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrpart.c: More backend/library abstraction and generalization:
|
||||||
|
turn the list_parts() and list_programmers() functions into
|
||||||
|
general list iteration functions that call a caller-supplied
|
||||||
|
callback for each element. Implement list_parts() and
|
||||||
|
list_programmers() as private functions in main.c based on that
|
||||||
|
approach.
|
||||||
|
* avrpart.h: (Ditto.)
|
||||||
|
* main.c: (Ditto.)
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
|
||||||
|
2007-01-25 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* Makefile.am: Rearrange everything so it is now built into a
|
||||||
|
libavrdude.a library, and link main.c against that library.
|
||||||
|
* configure.ac: Add AC_PROG_RANLIB as we are building a library
|
||||||
|
now.
|
||||||
|
|
||||||
|
2007-01-24 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Major code cleanup.
|
||||||
|
- Make all internal functions "static".
|
||||||
|
- Make sure each module's header and implementation file match.
|
||||||
|
- Remove all library-like functionality from main.c, so only
|
||||||
|
the actual frontend remains in main.c.
|
||||||
|
- Add C++ brackets to all header files.
|
||||||
|
* avr.c: (Ditto.)
|
||||||
|
* avr.h: (Ditto.)
|
||||||
|
* avr910.c: (Ditto.)
|
||||||
|
* avr910.h: (Ditto.)
|
||||||
|
* avrdude.h: (Ditto.)
|
||||||
|
* avrpart.c: (Ditto.)
|
||||||
|
* avrpart.h: (Ditto.)
|
||||||
|
* bitbang.h: (Ditto.)
|
||||||
|
* butterfly.h: (Ditto.)
|
||||||
|
* config.c: (Ditto.)
|
||||||
|
* config.h: (Ditto.)
|
||||||
|
* confwin.h: (Ditto.)
|
||||||
|
* crc16.c: (Ditto.)
|
||||||
|
* crc16.h: (Ditto.)
|
||||||
|
* fileio.c: (Ditto.)
|
||||||
|
* fileio.h: (Ditto.)
|
||||||
|
* jtagmkI.h: (Ditto.)
|
||||||
|
* jtagmkII.h: (Ditto.)
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* lists.h: (Ditto.)
|
||||||
|
* main.c: (Ditto.)
|
||||||
|
* par.h: (Ditto.)
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
* ppi.c: (Ditto.)
|
||||||
|
* ppi.h: (Ditto.)
|
||||||
|
* safemode.h: (Ditto.)
|
||||||
|
* serbb.h: (Ditto.)
|
||||||
|
* serial.h: (Ditto.)
|
||||||
|
* stk500.h: (Ditto.)
|
||||||
|
* stk500v2.c: (Ditto.)
|
||||||
|
* stk500v2.h: (Ditto.)
|
||||||
|
* term.c: (Ditto.)
|
||||||
|
* term.h: (Ditto.)
|
||||||
|
* usbasp.h: (Ditto.)
|
||||||
|
* update.c: New file.
|
||||||
|
* update.h: New file.
|
||||||
|
* Makefile.am: Include update.c and update.h.
|
||||||
|
|
||||||
|
2007-01-24 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Move all "extern" declarations into a centreal header file.
|
||||||
|
* Makefile.am: Add new avrdude.h.
|
||||||
|
* avrdude.h: New file.
|
||||||
|
* avr.c: Replace private extern decl's by #include "avrdude.h".
|
||||||
|
* avr910.c: (Ditto.)
|
||||||
|
* avrpart.c: (Ditto.)
|
||||||
|
* bitbang.c: (Ditto.)
|
||||||
|
* butterfly.c: (Ditto.)
|
||||||
|
* config.c: (Ditto.)
|
||||||
|
* config_gram.y: (Ditto.)
|
||||||
|
* fileio.c: (Ditto.)
|
||||||
|
* jtagmkI.c: (Ditto.)
|
||||||
|
* jtagmkII.c: (Ditto.)
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* main.c: (Ditto.)
|
||||||
|
* par.c: (Ditto.)
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* ppi.c: (Ditto.)
|
||||||
|
* ppiwin.c: (Ditto.)
|
||||||
|
* ser_avrdoper.c: (Ditto.)
|
||||||
|
* ser_posix.c: (Ditto.)
|
||||||
|
* ser_win32.c: (Ditto.)
|
||||||
|
* serbb_posix.c: (Ditto.)
|
||||||
|
* serbb_win32.c: (Ditto.)
|
||||||
|
* stk500.c: (Ditto.)
|
||||||
|
* stk500generic.c: (Ditto.)
|
||||||
|
* stk500v2.c: (Ditto.)
|
||||||
|
* term.c: (Ditto.)
|
||||||
|
* usb_libusb.c: (Ditto.)
|
||||||
|
* usbasp.c: (Ditto.)
|
||||||
|
|
||||||
|
2007-01-13 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in (ATmega8): Bump the delay values for flash
|
||||||
|
and EEPROM, based on the current Atmel XML file.
|
||||||
|
|
||||||
|
2007-01-12 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Improve the detection of the Win32 HID library,
|
||||||
|
and the presence of the header ddk/hidsdi.h. It now works
|
||||||
|
correctly under Cygwin and several flavours of MinGW.
|
||||||
|
* Makefile.am: Add new LIBHID pattern.
|
||||||
|
|
||||||
|
2007-01-11 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* butterfly.c (butterfly_initialize): when sending the 'T'
|
||||||
|
command (which is ignored by current AVR109 bootloaders),
|
||||||
|
send the first reply from the list of supported device
|
||||||
|
codes back rather than using avrdude.conf's idea about
|
||||||
|
an AVR910 device code. Apparently, this solves disagreements
|
||||||
|
between different versions of at least the ATmega8 AVR910
|
||||||
|
device code.
|
||||||
|
Closes bug #18727: Writing flash failed
|
||||||
|
|
||||||
|
2007-01-07 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Reported by Till Harbaum:
|
||||||
|
* avrdude.conf.in (ATtiny25/45/85): Change HVSP reset from
|
||||||
|
500 microseconds to 1 ms, matching the most recent Atmel XML
|
||||||
|
specs.
|
|
@ -0,0 +1,185 @@
|
||||||
|
2008-11-20 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.h: Change the prototype for usleep() to be more Cygwin-
|
||||||
|
friendly.
|
||||||
|
* ppiwin.c: (Ditto.)
|
||||||
|
|
||||||
|
2008-11-06 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by limor <limor@ladyada.net>
|
||||||
|
* usbtiny.c (usbtiny_cmd): Replace sizeof() by a fixed constant
|
||||||
|
4 for the result array, because otherwise it would take the size
|
||||||
|
of a pointer which miserably fails on 64-bit machines.
|
||||||
|
|
||||||
|
2008-11-05 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
patch #6609: Using PCI parallel port cards on Windows
|
||||||
|
* ppiwin.c (ppi_open): If the port parameter passed from the
|
||||||
|
-p option is neither lpt1/2/3, try interpreting it directly as
|
||||||
|
a base address.
|
||||||
|
* avrdude.1: Document the change.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2008-11-04 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #22882: Erase Cycle Counter does not work for stk500v2
|
||||||
|
* stk500v2.c (stk500v2_chip_erase,stk500hv_chip_erase): Return
|
||||||
|
the expected 0 for success rather than a protocol-dependant
|
||||||
|
number.
|
||||||
|
|
||||||
|
2008-11-04 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #22883: Chip Erase performed even with no-write flag (-n)
|
||||||
|
* main.c: Do not erase the chip if both, -e and -n options have
|
||||||
|
been specified.
|
||||||
|
|
||||||
|
2008-11-04 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #24589: AT90USB64* have wrong signature
|
||||||
|
* avrdude.conf.in: Uncomment the correct, and delete the wrong
|
||||||
|
signature for AT90USB646/647. Alas, the datasheet has never been
|
||||||
|
corrected for years.
|
||||||
|
|
||||||
|
2008-10-31 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkII.c: Fix a serious memory corruption that happened when
|
||||||
|
using the JTAG ICE mkII (or AVR Dragon) in ISP mode. The wrong
|
||||||
|
set of per-programmer private data had been allocated (stk500v2
|
||||||
|
vs. jtagmkII) which was too small to hold the actual data.
|
||||||
|
* jtagmkII.h: (Ditto.)
|
||||||
|
* stk500v2.c: (Ditto.)
|
||||||
|
|
||||||
|
2008-07-29 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkII.c: Implement Xmega JTAG support.
|
||||||
|
* jtagmkII_private.h: Add EMULATOR_MODE_JTAG_XMEGA.
|
||||||
|
|
||||||
|
2008-07-29 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* main.c: Remember whether the device initialization worked, and
|
||||||
|
allow to continue with -F if it failed yet do not attempt to
|
||||||
|
perform anything on the device itself. That way, -tF could be
|
||||||
|
specified for programmers like the STK500/STK600 even without a
|
||||||
|
device connected, just in order to allow changing parameters on
|
||||||
|
the programmer itself.
|
||||||
|
* avrdude.1: Document that possible use of the -F option.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2008-07-29 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c (stk600_xprog_paged_write): Fix a fatal miscalculation
|
||||||
|
of the number of bytes to be written which caused a malloc chunk
|
||||||
|
corruption.
|
||||||
|
|
||||||
|
2008-07-27 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
First implementation of ATxmega support. By now, only the
|
||||||
|
PDI mode of the STK600 is supported. Single-byte EEPROM
|
||||||
|
(and flash) updates do not work yet.
|
||||||
|
* avr.c: "boot" memory is a candidate memory region for paged
|
||||||
|
operations, besides "flash" and "eeprom".
|
||||||
|
* avrdude.conf.in: add ATxmega128A1 and ATxmega128A1revD
|
||||||
|
* avrpart.h: add the AVRPART_HAS_PDI flag (used to distinguish
|
||||||
|
ATxmega parts from classic AVRs), the nvm_base part field, and
|
||||||
|
the offset field for a memory region.
|
||||||
|
* config_gram.y: add "has_pdi", "nvm_base", and "offset"
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* main.c: disable auto_erase for ATxmega parts
|
||||||
|
* stk500v2.c: implement the XPROG functionality, and divert to
|
||||||
|
this for ATxmega parts
|
||||||
|
* avrdude.1: Document the changes.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2008-07-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Fix a bunch of warnings.
|
||||||
|
* avr910.c (avr910_paged_load): possible unitialized use of
|
||||||
|
rd_size
|
||||||
|
* jtagmkI.c (jtagmkI_initialize): pointer signedness mixup
|
||||||
|
* jtagmkII.c (jtagmkII_print_parms1): propagate const'ness
|
||||||
|
of parameter
|
||||||
|
* usbasp.c (usbasp_transmit): pointer signedness mixup
|
||||||
|
* ser_avrdoper.c (usbGetReport): remove useless pointer deref
|
||||||
|
|
||||||
|
2008-07-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by Ville Voipio:
|
||||||
|
patch #6501: New autotools support for avrdude
|
||||||
|
* Makefile.am: add @WINDOWS_DIRS@ to SUBDIR
|
||||||
|
* bootstrap: allow for autconf-2.61 and automake-1.10, too
|
||||||
|
* configure.ac: fix @WINDOWS_DIRS@ recursion, replace
|
||||||
|
AC_PROG_CC by AM_PROG_CC_C_O, for esoteric reasons
|
||||||
|
|
||||||
|
2008-06-13 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by Janos Sallai <janos.sallai@vanderbilt.edu>:
|
||||||
|
patch #6074: added support for crossbow's MIB510 programmer
|
||||||
|
* avrdude.conf.in: Add entry for mib510.
|
||||||
|
* stk500.c: Add special hooks to handle the MIB510 programmer.
|
||||||
|
It mostly talks STK500v1 protocol but has a special hello and
|
||||||
|
goodbye sequence, and uses a fixed block size of 256 bytes.
|
||||||
|
* doc/avrdude.texi: Document support for mib510.
|
||||||
|
|
||||||
|
2008-06-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by Klaus Leidinger <klaus@mikrocontroller-projekte.de>:
|
||||||
|
* main.c: Realign verbose messages.
|
||||||
|
* avrpart.c: (Ditto.)
|
||||||
|
* avr910.c: Print the device code selected in verbose mode.
|
||||||
|
* butterfly.c: (Ditto.)
|
||||||
|
|
||||||
|
2008-06-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by Klaus Leidinger <klaus@mikrocontroller-projekte.de>:
|
||||||
|
Add check for buffermode feature, and use it if present. Can be
|
||||||
|
turned off using -x no_blockmode.
|
||||||
|
* avr910.c: Implement buffermode test and usage.
|
||||||
|
* avrdude.1: Document -x no_blockmode.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2008-03-24 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* usb_libusb.c: #undef interface for Win32
|
||||||
|
|
||||||
|
2008-03-24 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avr910.c: Add support for the -x devcode option.
|
||||||
|
* avrdude.1: Document -x devcode for avr910.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2008-03-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Add initial support for the Atmel STK600, for
|
||||||
|
"classic" AVRs (AT90, ATtiny, ATmega) in both,
|
||||||
|
ISP and high-voltage programming modes.
|
||||||
|
* Makefile.am: Add -lm.
|
||||||
|
* avrdude.conf.in: Add stk600, stk600pp, and stk600hvsp.
|
||||||
|
* config_gram.y: Add support for the stk600* keywords.
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* pgm.h: Add the "chan" parameter to set_varef().
|
||||||
|
* stk500.c: (Ditto.)
|
||||||
|
* serial.h: Add USB endpoint support to struct filedescriptor.
|
||||||
|
* stk500v2.c: Implement the meat of the STK600 support.
|
||||||
|
* stk500v2.h: Add new prototypes for stk600*() programmers.
|
||||||
|
* stk500v2_private.h: Add new constants used in the STK600.
|
||||||
|
* term.c: Add AREF channel support.
|
||||||
|
* usb_libusb.c: Automatically determine the correct write
|
||||||
|
endpoint ID, the STK600 uses 0x83 while all other tools use
|
||||||
|
0x82. Propagate the EP to use through struct filedescriptor.
|
||||||
|
* usbdevs.h: Add the STK600 USB product ID.
|
||||||
|
* tools/get-stk600-cards.xsl: XSL transformation for
|
||||||
|
targetboards.xml to obtain the list of socket and routing
|
||||||
|
card IDs, to be used in stk500v2.c (for displaying the
|
||||||
|
names).
|
||||||
|
* tools/get-stk600-devices.xsl: XSL transformation for
|
||||||
|
targetboards.xml to obtain the table of socket/routing cards
|
||||||
|
and their respective AVR device support for doc/avrdude.texi.
|
||||||
|
* avrdude.1: Document all the STK600 stuff.
|
||||||
|
* doc/avrdude.texi: Ditto. Added a new chapter for
|
||||||
|
Programmer Specific Information.
|
||||||
|
|
||||||
|
2008-01-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c (stk500v2_recv): Make length computation unsigned so
|
||||||
|
it cannot accidentally become negative.
|
||||||
|
|
|
@ -0,0 +1,411 @@
|
||||||
|
2009-11-09 David Hoerl <dhoerl@mac.com>
|
||||||
|
|
||||||
|
* fileio.c: ihex2bin did not properly handle files > 64K bytes
|
||||||
|
* usb_libusb.c: re-enabled usb_reset for Macs (no reset causes lots of failures)
|
||||||
|
* avrdude.1: spacing issue for avr32 fixed.
|
||||||
|
|
||||||
|
2009-11-09 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: Implemented reset= and speed= extended parameters.
|
||||||
|
* avrdude.1: Document the change.
|
||||||
|
|
||||||
|
2009-11-04 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* configure.ac, Makefile.am: Test if GCC accepts -Wno-pointer-sign
|
||||||
|
|
||||||
|
2009-11-04 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: Implemented 'BinMode' support for
|
||||||
|
firmware 2.7 and higher.
|
||||||
|
* avrdude.1: Added info about BusPirate.
|
||||||
|
|
||||||
|
2009-11-03 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* arduino.c: Add on to bug #26703 / patch #6866 - clear DTR/RTS
|
||||||
|
when closing the port.
|
||||||
|
* Makefile.am: Silent warnings about signedness - they're useless
|
||||||
|
and annoying, especially for 'char' vars.
|
||||||
|
|
||||||
|
2009-10-22 David Hoerl <dhoerl@mac.com>
|
||||||
|
|
||||||
|
* usb_libusb.c: disabled usb_reset for Macs (same as FreeBSD)
|
||||||
|
|
||||||
|
2009-10-12 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* main.c: Re-added default to serial port for BusPirate.
|
||||||
|
|
||||||
|
2009-10-12 David Hoerl <dhoerl@mac.com>
|
||||||
|
|
||||||
|
* main.c: removed some avr32 code that was pushed into jtagmkII.c
|
||||||
|
* jtagmkII.c: consolodated the avr32 reset code and avr32_chipreset
|
||||||
|
* avrpart.h: modified AVRPART flags for avr32
|
||||||
|
* lexer.l: added is_avr32 flag - only way to get yacc code to set flag
|
||||||
|
* avrdude.conf.in: updated avr32 section to include "is_avr32" flag
|
||||||
|
|
||||||
|
2009-10-12 David Hoerl <dhoerl@mac.com>
|
||||||
|
|
||||||
|
* config_gram.y: Restored inadvertantly removed buspirate entry
|
||||||
|
* lexer.l: Restored inadvertantly removed buspirate entry
|
||||||
|
|
||||||
|
2009-10-12 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: Replace GNU-only %as with %s in sscanf call.
|
||||||
|
* ser_win32.c(ser_set_dtr_rts): Fixed typo in parameter name.
|
||||||
|
* NEWS: Announce BusPirate.
|
||||||
|
|
||||||
|
2009-10-11 David Hoerl <dhoerl@mac.com>
|
||||||
|
|
||||||
|
Support for AVR32
|
||||||
|
|
||||||
|
* AUTHORS: added myself
|
||||||
|
* NEWS: announced AVR32 support
|
||||||
|
* main.c: AVR32 flag tests to avoid several code blocks
|
||||||
|
* fileio.c: mods to ihex read function to handle address offsets and
|
||||||
|
size of avr32
|
||||||
|
* jtagmkI.c: added cast to printf call to remove warning
|
||||||
|
* arduino.c: added header file to bring in prototype for usleep()
|
||||||
|
* config_gram.y: added defines for avr32, new jtag_mkii variant for avr32
|
||||||
|
* jtagmkII_private.h: new jtag_mkii message types defined (used by
|
||||||
|
avr32program)
|
||||||
|
* jtagmkII.h: extern jtagmkII_avr32_initpgm() addition
|
||||||
|
* jtagmkII.c: huge amount of code in support of avr32
|
||||||
|
* avrpart.h: additional flags to AVRPART for avr32
|
||||||
|
* usb_libusb.c: modified verbose test for USB read per-byte messages by
|
||||||
|
by one, so with verbose=3 you get just full messages, 4 gives you bytes
|
||||||
|
too
|
||||||
|
* lexer.l: additions for avr32
|
||||||
|
|
||||||
|
2009-10-10 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
Support for Arduino auto-reset:
|
||||||
|
* serial.h, ser_avrdoper.c, ser_posix.c, ser_win32.c: Added
|
||||||
|
serial_device.set_dtr_rts implementations.
|
||||||
|
* arduino.c, stk500.c, stk500.h: Call serial_set_dtr_rts()
|
||||||
|
to reset Arduino board before program upload.
|
||||||
|
Inspired by patch #6866, resolves bug #26703
|
||||||
|
|
||||||
|
2009-10-08 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: Optimised buspirate_cmd() - reading 1kB EEPROM now
|
||||||
|
takes only 14 sec instead of almost 2 mins with the original
|
||||||
|
implementation.
|
||||||
|
|
||||||
|
2009-10-08 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c, buspirate.h: Support for the BusPirate programmer
|
||||||
|
* config_gram.y, avrdude.conf.in, main.c, lexer.l, Makefile.am:
|
||||||
|
Glue for BusPirate.
|
||||||
|
|
||||||
|
2009-08-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* usb_libusb.c (usbdev_close): Repair the logic around the
|
||||||
|
conditional compilation of usb_reset() introduced in r798.
|
||||||
|
|
||||||
|
2009-07-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: We are post-5.8 now.
|
||||||
|
|
||||||
|
2009-07-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Prepare for releasing version 5.8
|
||||||
|
|
||||||
|
2009-07-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Roger Wolff:
|
||||||
|
bug #26527: bug in unicode conversion
|
||||||
|
* ser_avrdoper.c (convertUniToAscii): when encountering a UTF-16
|
||||||
|
character that cannot be converted to ASCII, increment the UTF-16
|
||||||
|
pointer anyway when proceeding.
|
||||||
|
|
||||||
|
2009-07-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkI.c (jtagmkI_send): Replace %zd format by %u since not all
|
||||||
|
implementations do understand the C99 formatting options (sigh).
|
||||||
|
* jtagmkII.c (jtagmkII_send): (Ditto.)
|
||||||
|
* stk500v2.c (stk500v2_recv): (Ditto.)
|
||||||
|
|
||||||
|
2009-07-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #26002: HVPP of EEPROM with AVR Dragon and ATmega8 Fails
|
||||||
|
* avrdude.conf.in (ATmega8): add page size for EEPROM.
|
||||||
|
|
||||||
|
2009-07-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c: Fix a serious memory corruption problem resulting
|
||||||
|
out of the chaining of both, the stk500v2 and the jtagmkII
|
||||||
|
programmers for some programming hardware (JTAG ICE mkII and AVR
|
||||||
|
Dragon running in ISP, HVSP or PP mode), where both programmers
|
||||||
|
have to maintain their private programmer data.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Post-release (is pre-release...)
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Prepare for releasing version 5.7
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* main.c: Add my name to the copyright output when being verbose.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by Shaun Jackman <sjackman@gmail.com>
|
||||||
|
bug #21798: Fix both XSLT scripts
|
||||||
|
* tools/get-dw-params.xsl (format-hex): Add the parameter count.
|
||||||
|
* tools/get-hv-params.xsl (format_cstack): Ditto.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #21922: ATmega163 still not working in version 5.5
|
||||||
|
* avrdude.conf.in (atmega163): fill in stk500v2 parameters, correct
|
||||||
|
some flash programming parameters as well.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #22206: avrdude: ser_setspeed(): tcsetattr() failed
|
||||||
|
* ser_posix.c (ser_setspeed): Don't pass TCSAFLUSH to tcsetattr() as
|
||||||
|
it apparently fails to work on Solaris. After reading the
|
||||||
|
documentation again, it seems TCSAFLUSH and TCSANOW are indeed
|
||||||
|
mutually exclusive.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #22234: WINDOWS version: HOWTO: Specify Serial Ports Larger than COM9
|
||||||
|
* ser_win32.c (ser_open): prepend \\.\ to any COM port name, so it is
|
||||||
|
safe to be used for COM ports above 9.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #26408: Crash in stk500v2_open()
|
||||||
|
* stk500generic.c: Implement setup and teardown hooks, calling in turn
|
||||||
|
the respective hooks of the stk500v2 implementation.
|
||||||
|
|
||||||
|
2009-07-02 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #26130: Avrdude doesn't display it's version.
|
||||||
|
* main.c (usage): add a version number display to the default usage
|
||||||
|
message.
|
||||||
|
|
||||||
|
2009-07-01 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #26412: avrdude segfaults when called with a programmer that does not
|
||||||
|
support it
|
||||||
|
* main.c: do not call pgm->perform_osccal() unless it is != 0.
|
||||||
|
|
||||||
|
2009-06-24 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Contributed by Zoltan Laday:
|
||||||
|
patch #6825: xmega problems with JTAGICEmkII
|
||||||
|
* jtagmkII.c: Many fixes for Xmega devices.
|
||||||
|
* jtagmkII_private.h: Add various new constants required for
|
||||||
|
Xmega devices.
|
||||||
|
* avrdude.conf.in: New devices: ATXMEGA64A1, ATXMEGA192A1,
|
||||||
|
ATXMEGA256A1, ATXMEGA64A3, ATXMEGA128A3, ATXMEGA192A3,
|
||||||
|
ATXMEGA256A3, ATXMEGA256A3B, ATXMEGA16A4, ATXMEGA32A4,
|
||||||
|
ATXMEGA64A4, ATXMEGA128A4
|
||||||
|
* avr.c (avr_read, avr_write): Add more names for (Xmega)
|
||||||
|
memory areas that require paged operation.
|
||||||
|
|
||||||
|
2009-06-24 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c (stk600_xprog_write_byte): Handle writing fuse bytes.
|
||||||
|
|
||||||
|
2009-04-28 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Carl Hamilton:
|
||||||
|
* update.c (parse_op): correctly \0-terminate buf after filling
|
||||||
|
it, before it is potentially used as the source of a call to
|
||||||
|
strlen or strcpy.
|
||||||
|
|
||||||
|
2009-04-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* doc/avrdude.texi: Merge the -P 0xXXX option description from
|
||||||
|
avrdude.1.
|
||||||
|
|
||||||
|
2009-04-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: declare AM_PROG_CC_C_O to avoid the warning
|
||||||
|
"compiling `config_gram.c' with per-target flags
|
||||||
|
requires `AM_PROG_CC_C_O' in `configure.ac'"
|
||||||
|
|
||||||
|
2009-03-22 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #25971: "error writing to <stdout>" with multiple -U params.
|
||||||
|
* fileio.c: Do not close the input/output stream when working on an
|
||||||
|
stdio stream.
|
||||||
|
|
||||||
|
2009-02-28 Thomas Fischl <tfischl@gmx.de>
|
||||||
|
|
||||||
|
Based on patch #6484 commited by Jurgis Brigmanis:
|
||||||
|
* usbasp.c: added software control for ISP speed
|
||||||
|
* usbasp.h: (Ditto.)
|
||||||
|
|
||||||
|
2009-02-28 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avr910.c (avr910_read_byte_flash): Eliminate a static variable that
|
||||||
|
hasn't been in use for 5 years.
|
||||||
|
|
||||||
|
2009-02-27 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Post-release 5.6.
|
||||||
|
|
||||||
|
2009-02-27 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Prepare for releasing version 5.6.
|
||||||
|
|
||||||
|
2009-02-27 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Ed Okerson:
|
||||||
|
* jtagmkII.c (jtagmkII_read_byte): Fix signature reading of
|
||||||
|
Xmega.
|
||||||
|
|
||||||
|
2009-02-26 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Mikael Hermansson:
|
||||||
|
* avrdude.conf.in (ATxmega256A3): new device.
|
||||||
|
* stk500v2 (stk500v2_initialize): Enable the AVRISPmkII as a
|
||||||
|
PDI-capable device for ATxmega parts.
|
||||||
|
|
||||||
|
2009-02-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Lars Immisch:
|
||||||
|
patch #6750: Arduino support - new programmer-id
|
||||||
|
* arduino.c: New file, inherits stk500.c.
|
||||||
|
* arduino.h: New file.
|
||||||
|
* Makefile.am: Add arduino.c and arduino.h.
|
||||||
|
* config_gram.y: Add arduino keyword.
|
||||||
|
* lexer.l: (Ditto.)
|
||||||
|
* avrdude.conf.in: (Ditto.)
|
||||||
|
* avrdude.1: Document the new programmer type.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2009-02-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c: Turn all non-const static data into instance data.
|
||||||
|
|
||||||
|
2009-02-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* Makefile.am: Move term.[ch] from the library into the CLI
|
||||||
|
application section, as it is not useful for anything else but
|
||||||
|
the CLI frontend.
|
||||||
|
|
||||||
|
2009-02-25 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in (ATmega1284P): new device.
|
||||||
|
|
||||||
|
2009-02-23 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
More fixes for Solaris, including fixes for the Sunpro compiler:
|
||||||
|
* avr.h: Remove stray semicolon.
|
||||||
|
* configure.ac: Add check for predefined types uint_t and ulong_t.
|
||||||
|
* confwin.c: Include "avrdude.h" on top to avoid empty translation
|
||||||
|
unit warning.
|
||||||
|
* ppwin.c: (Ditto.)
|
||||||
|
* ser_win32.c: (Ditto.)
|
||||||
|
* serbb_win32.c: (Ditto.)
|
||||||
|
* jtagmkII.c (jtagmkII_recv): remove unreachable "return".
|
||||||
|
* stk500.c (stk500_initialize): (Ditto.)
|
||||||
|
* par.c: Test for both, __sun__ and __sun to see whether we are
|
||||||
|
being compiled on Solaris.
|
||||||
|
* ppi.c: (Ditto.)
|
||||||
|
* stk500v2.c: Implement the DEBUG and DEBUGRECV macros in a way
|
||||||
|
that is compatible with the ISO C99 standard.
|
||||||
|
* usbtiny.c: Only typedef uint_t and ulong_t if they have not
|
||||||
|
been found already by the autoconf checks.
|
||||||
|
|
||||||
|
2009-02-23 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #22204: Solaris10/11 Undefiniertes Symbol gethostbyname socket
|
||||||
|
connect
|
||||||
|
* configure.ac: Add checks for gethostent() and socket().
|
||||||
|
While being here, remove some old cruft left from ancient days.
|
||||||
|
|
||||||
|
2009-02-22 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* lexer.l: Bump the %p size so AT&T lex will continue to work.
|
||||||
|
|
||||||
|
2009-02-19 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
(Partially) submitted by John Voltz:
|
||||||
|
bug #20004: AVRDUDE update (-U) operations do not close files
|
||||||
|
* fileio.c (fmt_autodetect, fileio): fclose() files.
|
||||||
|
|
||||||
|
2009-02-18 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* usbtiny.c: Replace all but one (very unlikely to trigger) exit(1)
|
||||||
|
by return -1.
|
||||||
|
|
||||||
|
2009-02-18 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Dick Streefland:
|
||||||
|
patch #6749: make reading from the USBtinyISP programmer more robust
|
||||||
|
* usbtiny.c: Add code to retry failed communication attempts.
|
||||||
|
|
||||||
|
2009-02-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Nick Hibma:
|
||||||
|
bug #22271: usb_reset in usb_libusb.c not necessary in FreeBSD 6.x
|
||||||
|
* usb_libusb.c (usbdev_close): Do not call usb_reset() on FreeBSD.
|
||||||
|
It is not necessary there.
|
||||||
|
|
||||||
|
2009-02-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Andrew O. Shadoura:
|
||||||
|
bug #25156: add direct SPI transfer mode
|
||||||
|
* bitbang.c: Implement direct SPI transfers.
|
||||||
|
* bitbang.h: (Ditto.)
|
||||||
|
* par.c: (Ditto.)
|
||||||
|
* pgm.c: (Ditto.)
|
||||||
|
* pgm.h: (Ditto.)
|
||||||
|
* term.c: Add the "spi" and "pgm" commands.
|
||||||
|
* avrdude.1: Document the changes.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
||||||
|
2009-02-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Limor ("Lady Ada"):
|
||||||
|
bug #24749: add support for '328p
|
||||||
|
* avrdude.conf.in (ATmega328P): new device support.
|
||||||
|
|
||||||
|
2009-02-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by "Womo":
|
||||||
|
bug #25241: AT90USB162, AT90USB82 device support patch for avrdude-5.5
|
||||||
|
(also: bug #21745: AT90USBxx2 support)
|
||||||
|
* avrdude.conf.in (AT90USB162, AT90USB82): new device support.
|
||||||
|
|
||||||
|
2009-02-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Evangelos Arkalis:
|
||||||
|
patch #6069: Atmel AT89ISP Cable
|
||||||
|
* avrdude.conf.in (89isp): new programmer support.
|
||||||
|
|
||||||
|
2009-02-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Bob Paddock:
|
||||||
|
patch #6748: ATTiny88 Config
|
||||||
|
* avrdude.conf.in (ATtiny88): new device support.
|
||||||
|
|
||||||
|
2009-02-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Mark Litwack:
|
||||||
|
patch #6261: avrdude won't use dragon/debugwire to write a file
|
||||||
|
to eeprom
|
||||||
|
* jtagmkII.c (jtagmkII_paged_write): when in debugWire mode,
|
||||||
|
implement a paged write to EEPROM as a series of byte writes.
|
||||||
|
|
||||||
|
2009-02-16 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Janos Sallai:
|
||||||
|
patch #6542: paged_load fails on the MIB510 programming board
|
||||||
|
* stk500.c: Add a workaround for the different signon sequence on
|
||||||
|
MIB510 programmers.
|
||||||
|
|
||||||
|
2009-02-05 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in: Add the ATmega128RFA1.
|
||||||
|
* avrdude.1: document the addition of ATmega128RFA1.
|
||||||
|
* doc/avrdude.texi: (Ditto.)
|
||||||
|
|
|
@ -0,0 +1,354 @@
|
||||||
|
2010-12-17 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in (ATmega128RFA1): Bump two timing values in order to
|
||||||
|
improve ISP programming stability, in particular with the STK600.
|
||||||
|
|
||||||
|
2010-12-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c (stk500v2_command): Detect warning status codes.
|
||||||
|
|
||||||
|
2010-10-22 Nils Springob <nils@nicai-systems.de>
|
||||||
|
|
||||||
|
* serial.h: serial_open() calls will now return -1 on error (no call to exit())
|
||||||
|
* buspirate.c: (Dito.)
|
||||||
|
* jtagmkII.c: (Dito.)
|
||||||
|
* butterfly.c: (Dito.)
|
||||||
|
* jtagmkI.c: (Dito.)
|
||||||
|
* arduino.c: (Dito.)
|
||||||
|
* avr910.c: (Dito.)
|
||||||
|
* stk500.c: (Dito.)
|
||||||
|
* ser_avrdoper.c: (Dito.)
|
||||||
|
* stk500v2.c: (Dito.)
|
||||||
|
* ser_posix.c: (Dito.)
|
||||||
|
* usb_libusb.c: (Dito.)
|
||||||
|
|
||||||
|
2010-07-27 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #30566: MinGW + Ubuntu 9.04
|
||||||
|
* stk500v2.c (stk500v2_open): use same condition to refer to the AVR
|
||||||
|
Doper support as used in the definition in ser_avrdoper.c.
|
||||||
|
(Thanks to Christian Starkjohann for the analysis of the problem.)
|
||||||
|
|
||||||
|
2010-07-19 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: Added compatibility with BusPirate "NewUI" firmware 5.x
|
||||||
|
(contributed by Kari Knuuttila)
|
||||||
|
|
||||||
|
2010-07-12 Nils Springob <nils@nicai-systems.de>
|
||||||
|
|
||||||
|
* avrdude.conf.in (atmega88p): New device.
|
||||||
|
|
||||||
|
2010-06-03 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #29913: 246 Byte Bug - AVRdude crashes
|
||||||
|
doc/avrdude.texi (Troubleshooting): Mention the libusb 0.1 API
|
||||||
|
wrapper issue that is present in some Linux versions.
|
||||||
|
|
||||||
|
2010-03-19 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #29263: Can't build avrdude on windows using latest cygwin 1.7.1
|
||||||
|
* doc/avrdude.texi: Remove the recommendation for building
|
||||||
|
Win32 binaries under Cygwin; mention MinGW as an alternative
|
||||||
|
environment.
|
||||||
|
|
||||||
|
2010-03-08 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* ser_posix.c(ser_set_dtr_rts): Fixed DTR on/off to make
|
||||||
|
Arduino auto-reset work. (bug #29108, patch #7100)
|
||||||
|
|
||||||
|
2010-03-05 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* buspirate.c: Replace printf() by fprintf(stderr)
|
||||||
|
* safemode.c: (Dito.)
|
||||||
|
* usbtiny.c: (Dito.)
|
||||||
|
|
||||||
|
2010-01-22 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Cleanup Cygwin builds.
|
||||||
|
* windows/Makefile.am (loaddrv_LDFLAGS): remove, the -mno-cygwin
|
||||||
|
flag is supposed to be set in CFLAGS by ./configure
|
||||||
|
* configure.ac: add a check for the presence of usleep(), add a
|
||||||
|
check whether the linker accepts -static
|
||||||
|
* avrdude.h: protect prototype for usleep by !defined(HAVE_USLEEP)
|
||||||
|
* ppwin.c (usleep): protect by !defined(HAVE_USLEEP)
|
||||||
|
* main.c: silence "array subscript of type char" compiler warnings
|
||||||
|
by casting all arguments to tolower()/toupper() and isspace()/
|
||||||
|
isdigit()/ispunct() to "int"
|
||||||
|
* butterfly.c: (Dito.)
|
||||||
|
* avr910.c: (Dito.)
|
||||||
|
|
||||||
|
2010-01-19 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Bump for post-5.10.
|
||||||
|
|
||||||
|
2010-01-19 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Released version 5.10.
|
||||||
|
|
||||||
|
2010-01-19 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #28677: Cygwin's GCC no longer supports -mno-cygwin option
|
||||||
|
* configure.ac: For Win32 environments, add a check whether the
|
||||||
|
compiler understands the -mno-cygwin option. If not, don't use
|
||||||
|
it but suggest using a different compiler.
|
||||||
|
|
||||||
|
2010-01-18 David Hoerl <dhoerl@mac.com>
|
||||||
|
|
||||||
|
bug #28660: Problem with loading intel hex rom files that exceed
|
||||||
|
0x10000 bytes
|
||||||
|
* fileio.c: Fix two byte shifts.
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Michael biebl:
|
||||||
|
* configure.ac: Fix FreeBSD default serial port name.
|
||||||
|
* doc/avrdude.texi: (Dito.)
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkII.c: If entering JTAG mode fails with a bad JTAG ID
|
||||||
|
message, retry with external reset applied (in case the target
|
||||||
|
is in sleep mode or has asserted the JTD bit).
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Aurelien Jarno:
|
||||||
|
* configure.ac: Fix build for GNU/kFreeBSD.
|
||||||
|
* ppi.c: (Dito.)
|
||||||
|
* par.c: (Dito.)
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Bump version for post-5.8.
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* configure.ac: Bump version for release 5.8.
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Soren Jorvang:
|
||||||
|
bug #28611: -i delay not being applied to all serial port
|
||||||
|
bit banging state transitions
|
||||||
|
* serbb_win32.c: Apply ispdelay everywhere.
|
||||||
|
* serbb_posix.c: (Dito.)
|
||||||
|
|
||||||
|
2010-01-15 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2_private.h: Implement TPI mode for AVRISPmkII/STK600
|
||||||
|
* config_gram.y: (Dito.)
|
||||||
|
* avrpart.h: (Dito.)
|
||||||
|
* stk500v2.c: (Dito.)
|
||||||
|
* main.c: (Dito.)
|
||||||
|
* lexer.l: (Dito.)
|
||||||
|
* avrdude.conf.in: Add ATtiny4/5/9/10
|
||||||
|
* avrdude.1: Document TPI and new device support.
|
||||||
|
* doc/avrdude.texi: (Dito.)
|
||||||
|
|
||||||
|
2010-01-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by clint fisher:
|
||||||
|
patch #7038: Adding Atmega32U4 Device to avrdude.conf.in
|
||||||
|
* avrdude.conf.in (atmega32u4): New device.
|
||||||
|
* avrdude.1: Document the new device support.
|
||||||
|
* doc/avrdude.texi: (Dito.)
|
||||||
|
|
||||||
|
2010-01-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Thomas Pircher:
|
||||||
|
patch #6927: Documentation patches
|
||||||
|
* doc/avrdude.texi: Fix various typos, and remove the last
|
||||||
|
remnants of obsoleted options -i/-o/-m/-f.
|
||||||
|
* avrdude.1: Merge typo fixes from avrdude.texi where
|
||||||
|
applicable.
|
||||||
|
|
||||||
|
2010-01-14 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* avrdude.1: Update documentation to match the reality (device
|
||||||
|
support, memory areas).
|
||||||
|
* doc/avrdude.texi: Update documentation to match the
|
||||||
|
reality (device support, programmer support, memory areas).
|
||||||
|
Merge buspirate-specific comments from avrdude.1.
|
||||||
|
* jtagmkII.c: Add some firmware feature checks.
|
||||||
|
|
||||||
|
2010-01-13 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkII.c: Implement PDI mode support for the JTAG ICE mkII
|
||||||
|
and the AVR Dragon.
|
||||||
|
* jtagmkII.h: (Dito.)
|
||||||
|
* config_gram.y: (Dito.)
|
||||||
|
* jtagmkII_private.h: (Dito.)
|
||||||
|
* avrdude.conf.in: (Dito.)
|
||||||
|
* lexer.l: (Dito.)
|
||||||
|
|
||||||
|
2010-01-13 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c: Update STK600 routing and socket card data from XML
|
||||||
|
file.
|
||||||
|
|
||||||
|
2010-01-13 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* stk500v2.c: Cleanup the open/close handling to avoid accessing
|
||||||
|
unallocated memory (in the atexit handler) in case of bailing out.
|
||||||
|
* main.c: (Ditto.)
|
||||||
|
|
||||||
|
2010-01-13 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkII.c: Stylistic changes: move #defines out into
|
||||||
|
jtagmkII_private.h, drop all #if 0 blocks, fold overly long lines,
|
||||||
|
move the *_initpgm() functions to the end of the file; while being
|
||||||
|
here, remove all trailing whitespace.
|
||||||
|
* jtagmkII_private.h: move AVR32 #defines here.
|
||||||
|
|
||||||
|
2010-01-12 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* bootstrap: autoconf 2.62 works well.
|
||||||
|
|
||||||
|
2010-01-12 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Various fixes for Xmega devices.
|
||||||
|
* avrdude.conf.in: Correctly declare EEPROM page sizes for
|
||||||
|
all Xmega devices (0x20 instead of 0x100).
|
||||||
|
* avr.c: If a memory region has a page size declared, try
|
||||||
|
using the paged IO routines regardless of the target memory
|
||||||
|
name. Xmega EEPROM requires to be written in paged mode.
|
||||||
|
Correctly use a long (rather than unsigned long) variable to
|
||||||
|
evaluate the success status of the paged mode write attempt.
|
||||||
|
* stk500v2.c: Don't apply TIF space offsets twice (bug #27995:
|
||||||
|
AVRDUDE 5.8svn fails to program and read XMEGA); use
|
||||||
|
stk500v2_loadaddr() prior to paged mode (EEPROM and flash) writes,
|
||||||
|
otherwise programming of flash areas will fail; while being there,
|
||||||
|
check the return value of stk500v2_loadaddr() everywhere; use the
|
||||||
|
correct write/erase mode bits (same as AVR Studio does).
|
||||||
|
|
||||||
|
2010-01-12 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: Initialise firmware version to v0.0
|
||||||
|
prior to parsing the buspirate banner.
|
||||||
|
|
||||||
|
2010-01-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Clean-up the Xmega erase functions.
|
||||||
|
* jtagmkII_private.h: Add CMND_XMEGA_ERASE as well as
|
||||||
|
the various XMEGA_ERASE_* definitions (from updated
|
||||||
|
appnote AVR067)
|
||||||
|
* jtagmkII.c (jtagmkII_chip_erase): Correctly implement Xmega chip
|
||||||
|
erase based on CMND_XMEGA_ERASE. After erasing an Xmega part, do
|
||||||
|
*not* reinitialize the world, as a subsequent programming
|
||||||
|
operation will fail (for unknown reasons). Actually, this was
|
||||||
|
really only required for ancient AVRs, but doesn't hurt on mega
|
||||||
|
and tiny devices.
|
||||||
|
* jtagmkII.c (jtagmkII_pre_write): Remove, this turned out
|
||||||
|
to be just a chip erase.
|
||||||
|
* jtagmkII.c (jtagmkII_program_disable): Don't try reading
|
||||||
|
"hfuse" for Xmega parts; they don't have it.
|
||||||
|
* main.c (main): Re-enable auto-erase. It's been done
|
||||||
|
before (as "jtagmkII_pre_write") in jtagmkII_paged_write()
|
||||||
|
anyway. Xmega boot and application flash areas should be
|
||||||
|
handled separately in the future, so auto_erase can only
|
||||||
|
affect the area just being programmed.
|
||||||
|
|
||||||
|
2010-01-11 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* main.c (main): disable safemode for Xmega parts.
|
||||||
|
|
||||||
|
2010-01-12 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
|
||||||
|
* buspirate.c: If the BusPirate doesn't respond
|
||||||
|
to a standard a reset command assume it was in binmode
|
||||||
|
and attempt to exit to text mode first.
|
||||||
|
|
||||||
|
2010-01-08 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* bitbang.c: Fix Win32 build error: move freq up to the file
|
||||||
|
level.
|
||||||
|
* buspirate.c: Fix Win32 build warning: include <malloc.h> to
|
||||||
|
to get a declaration for alloca().
|
||||||
|
|
||||||
|
2010-01-08 Thomas Fischl <tfischl@gmx.de>
|
||||||
|
|
||||||
|
bug #28520: Programming with USBasp with low clock speed fails
|
||||||
|
* usbasp.c: Change blocksize depending on sck frequency to
|
||||||
|
avoid usb transmition timeouts.
|
||||||
|
|
||||||
|
2010-01-08 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #27505: serbb_posix does not cope with inverted pins
|
||||||
|
* serbb_posix (serbb_highpulsepin): apply PIN_MASK when
|
||||||
|
checking pin numbers.
|
||||||
|
* serbb_win32 (serbb_highpulsepin): (Dito.)
|
||||||
|
|
||||||
|
2010-01-08 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
bug #28516: Linux/Dragon: Error message on exit
|
||||||
|
* stk500v2.c: Fix the "bad response to GO command:
|
||||||
|
RSP_ILLEGAL_EMULATOR_MODE" message. jtagmkII_close()
|
||||||
|
has been called with the wrong pgm->cookie. Wrap it
|
||||||
|
inside stk500v2_jtagmkII_close(), adjusting the cookie
|
||||||
|
data appropriately.
|
||||||
|
|
||||||
|
2010-01-08 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Doug:
|
||||||
|
patch #7010: Win32 enhanced bitbang_delay
|
||||||
|
* bitbang.c (bitbang_calibrate_delay, bitbang_delay): On Win32,
|
||||||
|
use the high-resolution performance counter rather than the
|
||||||
|
uneducated delay loop guess if it is available on the target
|
||||||
|
hardware.
|
||||||
|
|
||||||
|
2010-01-08 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Gerard:
|
||||||
|
patch #6828: Using arbitrary BAUD rates
|
||||||
|
* ser_posix.c (serial_baud_lookup): Allow non-standard baud
|
||||||
|
rates.
|
||||||
|
* ser_win32.c (serial_baud_lookup): (Dito.)
|
||||||
|
|
||||||
|
2010-01-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Eric Trein:
|
||||||
|
bug #27596: AT90s2333 is not correctly supported in avrdude.conf
|
||||||
|
* avrdude.conf.in (at90s2333): add various STK500v2 parameters.
|
||||||
|
|
||||||
|
2010-01-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Gyorgy Szekely:
|
||||||
|
bug #28458: Buffer line is incorrectly released for PP programmers
|
||||||
|
* par.c (par_close): use par_setmany() rather than par_setpin()
|
||||||
|
for PPI_AVR_BUFF.
|
||||||
|
|
||||||
|
2010-01-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Submitted by Lukasz Goralczyk:
|
||||||
|
bug #27507: SIGSEGV when using avrdragon (avrdude 5.8)
|
||||||
|
* stk500v2.c (stk500v2_dragon_isp_initpgm): Use
|
||||||
|
stk500v2_jtagmkII_setup/stk500v2_jtagmkII_rather than their
|
||||||
|
jtagII counterparts, to get the private data properly
|
||||||
|
initialized.
|
||||||
|
|
||||||
|
2010-01-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* buspirate.c: Cosmetics: remove UTF-8 dashes, adjust for 8-column
|
||||||
|
hard tabs.
|
||||||
|
|
||||||
|
2010-01-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* buspirate.c: add $ Id $ line.
|
||||||
|
* buspirate.h: add $ Id $ line.
|
||||||
|
|
||||||
|
2010-01-07 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
Fix a few warnings that came up recently (some of them only triggered
|
||||||
|
by recent GCC versions).
|
||||||
|
* config_gram.y (parse_cmdbits): "brkt possibly used uninitialized"
|
||||||
|
(GCC errs here)
|
||||||
|
* jtagmkII.c (jtagmkII_reset32): "status possibly used uninitialized"
|
||||||
|
(I think GCC errs, too)
|
||||||
|
* buspirate.c: "pointers differ in signedness" (mismatch between
|
||||||
|
string processing and the use of "unsigned char" throughought the
|
||||||
|
AVRDUDE API)
|
||||||
|
|
||||||
|
2010-01-01 Joerg Wunsch <j.gnu@uriah.heep.sax.de>
|
||||||
|
|
||||||
|
* jtagmkII.c (jtagmkII_smc_init32): replace sleep() by usleep() for
|
||||||
|
win32 compatibility.
|
|
@ -0,0 +1,179 @@
|
||||||
|
#
|
||||||
|
# avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
# Copyright (C) 2003, 2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
#
|
||||||
|
# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# $Id$
|
||||||
|
#
|
||||||
|
|
||||||
|
EXTRA_DIST = \
|
||||||
|
ChangeLog \
|
||||||
|
ChangeLog-2001 \
|
||||||
|
ChangeLog-2002 \
|
||||||
|
ChangeLog-2003 \
|
||||||
|
ChangeLog-2004-2006 \
|
||||||
|
ChangeLog-2007 \
|
||||||
|
ChangeLog-2008 \
|
||||||
|
ChangeLog-2009 \
|
||||||
|
avrdude.1 \
|
||||||
|
avrdude.spec \
|
||||||
|
bootstrap
|
||||||
|
|
||||||
|
CLEANFILES = \
|
||||||
|
config_gram.c \
|
||||||
|
config_gram.h \
|
||||||
|
lexer.c
|
||||||
|
|
||||||
|
BUILT_SOURCES = $(CLEANFILES)
|
||||||
|
|
||||||
|
#SUBDIRS = doc @WINDOWS_DIRS@
|
||||||
|
#DIST_SUBDIRS = doc windows
|
||||||
|
|
||||||
|
SUBDIRS = @SUBDIRS_AC@
|
||||||
|
SUBDIRS += @WINDOWS_DIRS@
|
||||||
|
DIST_SUBDIRS = @DIST_SUBDIRS_AC@
|
||||||
|
|
||||||
|
AM_YFLAGS = -d
|
||||||
|
|
||||||
|
avrdude_CPPFLAGS = -DCONFIG_DIR=\"$(sysconfdir)\"
|
||||||
|
|
||||||
|
libavrdude_a_CPPFLAGS = -DCONFIG_DIR=\"$(sysconfdir)\"
|
||||||
|
|
||||||
|
avrdude_CFLAGS = @ENABLE_WARNINGS@
|
||||||
|
|
||||||
|
libavrdude_a_CFLAGS = @ENABLE_WARNINGS@
|
||||||
|
|
||||||
|
avrdude_LDADD = $(top_builddir)/$(noinst_LIBRARIES) @LIBUSB@ @LIBUSB_1_0@ @LIBFTDI@ @LIBHID@ -lm
|
||||||
|
|
||||||
|
bin_PROGRAMS = avrdude
|
||||||
|
|
||||||
|
noinst_LIBRARIES = libavrdude.a
|
||||||
|
|
||||||
|
# automake thinks these generated files should be in the distribution,
|
||||||
|
# but this might cause trouble for some users, so we rather don't want
|
||||||
|
# to have them there.
|
||||||
|
#
|
||||||
|
# See
|
||||||
|
#
|
||||||
|
# https://savannah.nongnu.org/bugs/index.php?func=detailitem&item_id=15536
|
||||||
|
#
|
||||||
|
# for why we don't want to have them.
|
||||||
|
dist-hook:
|
||||||
|
rm -f \
|
||||||
|
$(distdir)/lexer.c \
|
||||||
|
$(distdir)/config_gram.c \
|
||||||
|
$(distdir)/config_gram.h
|
||||||
|
|
||||||
|
libavrdude_a_SOURCES = \
|
||||||
|
config_gram.y \
|
||||||
|
lexer.l \
|
||||||
|
arduino.h \
|
||||||
|
arduino.c \
|
||||||
|
avr.c \
|
||||||
|
avr.h \
|
||||||
|
avr910.c \
|
||||||
|
avr910.h \
|
||||||
|
avrdude.h \
|
||||||
|
avrftdi.c \
|
||||||
|
avrftdi.h \
|
||||||
|
avrpart.c \
|
||||||
|
avrpart.h \
|
||||||
|
bitbang.c \
|
||||||
|
bitbang.h \
|
||||||
|
buspirate.c \
|
||||||
|
buspirate.h \
|
||||||
|
butterfly.c \
|
||||||
|
butterfly.h \
|
||||||
|
config.c \
|
||||||
|
config.h \
|
||||||
|
confwin.c \
|
||||||
|
confwin.h \
|
||||||
|
crc16.c \
|
||||||
|
crc16.h \
|
||||||
|
fileio.c \
|
||||||
|
fileio.h \
|
||||||
|
freebsd_ppi.h \
|
||||||
|
jtagmkI.c \
|
||||||
|
jtagmkI.h \
|
||||||
|
jtagmkI_private.h \
|
||||||
|
jtagmkII.c \
|
||||||
|
jtagmkII.h \
|
||||||
|
jtagmkII_private.h \
|
||||||
|
linux_ppdev.h \
|
||||||
|
lists.c \
|
||||||
|
lists.h \
|
||||||
|
my_ddk_hidsdi.h \
|
||||||
|
par.c \
|
||||||
|
par.h \
|
||||||
|
pgm.c \
|
||||||
|
pgm.h \
|
||||||
|
pindefs.h \
|
||||||
|
ppi.c \
|
||||||
|
ppi.h \
|
||||||
|
ppiwin.c \
|
||||||
|
safemode.c \
|
||||||
|
safemode.h \
|
||||||
|
serial.h \
|
||||||
|
serbb.h \
|
||||||
|
serbb_posix.c \
|
||||||
|
serbb_win32.c \
|
||||||
|
ser_avrdoper.c \
|
||||||
|
ser_posix.c \
|
||||||
|
ser_win32.c \
|
||||||
|
solaris_ecpp.h \
|
||||||
|
stk500.c \
|
||||||
|
stk500.h \
|
||||||
|
stk500_private.h \
|
||||||
|
stk500v2.c \
|
||||||
|
stk500v2.h \
|
||||||
|
stk500v2_private.h \
|
||||||
|
stk500generic.c \
|
||||||
|
stk500generic.h \
|
||||||
|
tpi.h \
|
||||||
|
usbasp.c \
|
||||||
|
usbasp.h \
|
||||||
|
usbdevs.h \
|
||||||
|
usb_libusb.c \
|
||||||
|
usbtiny.h \
|
||||||
|
usbtiny.c \
|
||||||
|
update.h \
|
||||||
|
update.c \
|
||||||
|
wiring.h \
|
||||||
|
wiring.c
|
||||||
|
|
||||||
|
avrdude_SOURCES = \
|
||||||
|
main.c \
|
||||||
|
term.c \
|
||||||
|
term.h
|
||||||
|
|
||||||
|
man_MANS = avrdude.1
|
||||||
|
|
||||||
|
sysconf_DATA = avrdude.conf
|
||||||
|
|
||||||
|
install-exec-local: backup-avrdude-conf
|
||||||
|
|
||||||
|
distclean-local:
|
||||||
|
rm -f avrdude.conf
|
||||||
|
|
||||||
|
# This will get run before the config file is installed.
|
||||||
|
backup-avrdude-conf:
|
||||||
|
@echo "Backing up avrdude.conf in ${DESTDIR}${sysconfdir}"
|
||||||
|
@if test -e ${DESTDIR}${sysconfdir}/avrdude.conf; then \
|
||||||
|
cp -pR ${DESTDIR}${sysconfdir}/avrdude.conf \
|
||||||
|
${DESTDIR}${sysconfdir}/avrdude.conf.bak; \
|
||||||
|
fi
|
|
@ -0,0 +1,572 @@
|
||||||
|
$Id$
|
||||||
|
|
||||||
|
Approximate change log for AVRDUDE by version.
|
||||||
|
|
||||||
|
(For more detailed changes, see the ChangeLog file.)
|
||||||
|
|
||||||
|
----------------------------------------------------------------------
|
||||||
|
Current:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
- ATmega88P/168P
|
||||||
|
- ATmega8U2/16U2/32U2
|
||||||
|
- ATtiny4313
|
||||||
|
|
||||||
|
* New programmers supported:
|
||||||
|
- TPI programming through bitbang programmers (both, serial
|
||||||
|
and parallel ones)
|
||||||
|
- FT2232 (and relatives) based programmers (MPSSE bitbang mode)
|
||||||
|
- Wiring environment (http://wiring.org.co/)
|
||||||
|
- butterfly-style bootloader of the Mikrokopter.de device
|
||||||
|
|
||||||
|
* Bugfixes
|
||||||
|
|
||||||
|
|
||||||
|
Version 5.10:
|
||||||
|
|
||||||
|
* Bugfixes
|
||||||
|
- bug #28660: Problem with loading intel hex rom files that exceed
|
||||||
|
0x10000 bytes
|
||||||
|
- see ChangeLog for further details
|
||||||
|
|
||||||
|
* New Features
|
||||||
|
- (JTAG ICE / AVR Dragon) apply external reset if JTAG ID could
|
||||||
|
not be read
|
||||||
|
|
||||||
|
Version 5.9:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- AVR32A0512 (JTAGMKII only)
|
||||||
|
- ATmega32U4
|
||||||
|
- ATtiny4
|
||||||
|
- ATtiny5
|
||||||
|
- ATtiny9
|
||||||
|
- ATtiny10
|
||||||
|
|
||||||
|
* New programmers supported:
|
||||||
|
|
||||||
|
- BusPirate
|
||||||
|
- Arduino
|
||||||
|
- JTAGICEmkII and AVR Dragon in PDI mode (ATxmega devices)
|
||||||
|
- STK600 and AVRISP mkII in TPI mode (ATtiny4/5/9/10)
|
||||||
|
|
||||||
|
* Bugfixes
|
||||||
|
|
||||||
|
- see ChangeLog and ChangeLog-2009 for details
|
||||||
|
|
||||||
|
Version 5.8:
|
||||||
|
|
||||||
|
* Bugfixes; most importantly, fix a serious memory corruption for
|
||||||
|
that JTAG ICE mkII and AVR Dragon in ISP/HVSP/PP mode.
|
||||||
|
|
||||||
|
Version 5.7:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- ATXMEGA64A1
|
||||||
|
- ATXMEGA192A1
|
||||||
|
- ATXMEGA256A1
|
||||||
|
- ATXMEGA64A3
|
||||||
|
- ATXMEGA128A3
|
||||||
|
- ATXMEGA192A3
|
||||||
|
- ATXMEGA256A3
|
||||||
|
- ATXMEGA256A3B
|
||||||
|
- ATXMEGA16A4
|
||||||
|
- ATXMEGA32A4
|
||||||
|
- ATXMEGA64A4
|
||||||
|
- ATXMEGA128A4
|
||||||
|
|
||||||
|
* Major Xmega fixes for the JTAG ICE mkII (patch #6825)
|
||||||
|
|
||||||
|
* Bugfixes.
|
||||||
|
|
||||||
|
Version 5.6:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- AT90USB82
|
||||||
|
- AT90USB162
|
||||||
|
- ATtiny88
|
||||||
|
- ATmega328P
|
||||||
|
- ATmega1284P
|
||||||
|
- ATmega128RFA1
|
||||||
|
- ATxmega128A1 rev D
|
||||||
|
- ATxmega128A1
|
||||||
|
- ATxmega256A3
|
||||||
|
|
||||||
|
* New programmers supported:
|
||||||
|
|
||||||
|
- AT89ISP cable (patch #6069)
|
||||||
|
- Arduino
|
||||||
|
|
||||||
|
* Add support for the -x option to pass extended parameters to the
|
||||||
|
programmer backend.
|
||||||
|
|
||||||
|
* Add support for JTAG daisy-chains, using the -x daisychain=
|
||||||
|
option.
|
||||||
|
|
||||||
|
* Add support for the Atmel STK600 for "classic" AVRs (AT90, ATtiny,
|
||||||
|
ATmega), using either ISP or high-voltage programming modes.
|
||||||
|
|
||||||
|
* Add support for the -x devcode extended parameter to the avr910
|
||||||
|
programmer, to allow overriding the device code sent to the
|
||||||
|
programmer.
|
||||||
|
|
||||||
|
* Add support for the Crossbow MIB510 programmer (patch #6074, #6542).
|
||||||
|
|
||||||
|
* Add support to bootstrap with GNU autoconf 2.61, and automake 1.10,
|
||||||
|
respectively.
|
||||||
|
|
||||||
|
* Add support for ATxmega128A1 (including the revision D engineering
|
||||||
|
samples) for STK600 and AVRISPmkII tools using PDI
|
||||||
|
|
||||||
|
* The option combination -tF now enters terminal mode even if the
|
||||||
|
device initialization failed, so the user can modify programmer
|
||||||
|
parameters (like Vtarget).
|
||||||
|
|
||||||
|
* Add preliminary support for ATxmega128A1 for the JTAG ICE mkII using
|
||||||
|
JTAG.
|
||||||
|
|
||||||
|
* Add support for direct SPI transfers (bug #25156).
|
||||||
|
|
||||||
|
* Bugfixes.
|
||||||
|
|
||||||
|
Version 5.5:
|
||||||
|
|
||||||
|
* Add support for the USBtinyISP programmer (patch #6233)
|
||||||
|
|
||||||
|
* Add support for the C2N232I serial bitbang programmer (patch #6121)
|
||||||
|
|
||||||
|
* Bugfixes.
|
||||||
|
|
||||||
|
Version 5.4:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- AT90PWM2B/AT90PWM3B
|
||||||
|
|
||||||
|
* Bugfixes.
|
||||||
|
|
||||||
|
* Source code rearranged so that the functionality is now built
|
||||||
|
into a libavrdude.a library where main.c is currently the only
|
||||||
|
existing frontend.
|
||||||
|
|
||||||
|
* Implement ATmega256x support for butterfly/avr109.
|
||||||
|
|
||||||
|
Version 5.3.1:
|
||||||
|
|
||||||
|
* Add support for the AVR Dragon (all modes: ISP, JTAG, HVSP, PP,
|
||||||
|
debugWire).
|
||||||
|
|
||||||
|
* Add support for debugWire (both, JTAG ICE mkII, and AVR Dragon).
|
||||||
|
|
||||||
|
* Add support for the AVR Doper USB HID-class programmer.
|
||||||
|
|
||||||
|
* Bugfixes.
|
||||||
|
|
||||||
|
Version 5.2:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- AT90USB646/647/1286/1287
|
||||||
|
- ATmega2560/2561
|
||||||
|
- ATmega325/3250/645/6450
|
||||||
|
- ATtiny11 (HVSP only device)
|
||||||
|
- ATtiny261/461/861
|
||||||
|
|
||||||
|
* Fixed paged flash write operations for AT90PWMx devices
|
||||||
|
(error in datasheet).
|
||||||
|
|
||||||
|
* Add signature verification.
|
||||||
|
|
||||||
|
* Add high-voltage mode programming for the STK500 (both,
|
||||||
|
parallel, and high-voltage serial programming).
|
||||||
|
|
||||||
|
* Add support for using the JTAG ICE mkII as a generic ISP
|
||||||
|
programmer.
|
||||||
|
|
||||||
|
* Allow for specifying the ISP clock delay as an option for
|
||||||
|
bit-bang programming adapters.
|
||||||
|
|
||||||
|
* Add support for Thomas Fischl's USBasp low-cost USB-attached
|
||||||
|
programmer.
|
||||||
|
|
||||||
|
* The "stk500" programmer type is now implemented as a stub
|
||||||
|
that tries to probe for either "stk500v1" or "stk500v2".
|
||||||
|
|
||||||
|
* Many bugfixes.
|
||||||
|
|
||||||
|
Version 5.1:
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- ATmega640/1280/1281
|
||||||
|
- ATtiny24/44/84
|
||||||
|
|
||||||
|
* JTAG mkII support now works with libusb-win32, too
|
||||||
|
|
||||||
|
* JTAG ICE mkI support has been added
|
||||||
|
|
||||||
|
* Solaris support has been added (including ecpp(7D) parallel-port
|
||||||
|
bit-bang mode)
|
||||||
|
|
||||||
|
|
||||||
|
Version 5.0:
|
||||||
|
|
||||||
|
* Support for JTAGICE MkII device
|
||||||
|
|
||||||
|
* Support for STK500 Version 2 Protocol
|
||||||
|
|
||||||
|
* New devices supported:
|
||||||
|
|
||||||
|
- AT90CAN128
|
||||||
|
- ATmega329x/649x
|
||||||
|
- ATmega164/324/644
|
||||||
|
- AT90PWM2/3,
|
||||||
|
- ATmega164/324/644
|
||||||
|
- ATmega329x/649x
|
||||||
|
- ATtiny25/45/85
|
||||||
|
|
||||||
|
* Support for serial bit-bang adapters: Ponyprog serial, UISP DASA,
|
||||||
|
UISP DASA3.
|
||||||
|
|
||||||
|
* DAPA programmer pinout supported
|
||||||
|
|
||||||
|
* New "safemode" feature where fuse bits are verified before exit
|
||||||
|
and if possible recovered if they have changed. This is intended
|
||||||
|
to protect against changed fuses which were not requested which is
|
||||||
|
reported to sometimes happen due to improper power supply or other
|
||||||
|
reasons.
|
||||||
|
|
||||||
|
* Various fixes for avr910 and butterfly programmers
|
||||||
|
|
||||||
|
* Full support for AVR109 boot loaders (butterfly)
|
||||||
|
|
||||||
|
* Adding -q -q suppresses most terminal output
|
||||||
|
|
||||||
|
|
||||||
|
Version 4.4.0:
|
||||||
|
|
||||||
|
* Native Win32 support: The windows build doesn't need Cygwin
|
||||||
|
anymore. Additionally, the delay timing on windows should be
|
||||||
|
more accurate now.
|
||||||
|
Contributed by Martin Thomas
|
||||||
|
|
||||||
|
* Add support for
|
||||||
|
- ATmega48, ATmega88 (contributed by Galen Seitz)
|
||||||
|
- ATtiny2313 (contributed by Bob Paddock)
|
||||||
|
- ATtiny13 (contributed by Pawel Moll)
|
||||||
|
|
||||||
|
* Added command to change the SCK of STK500-programmers. Now it
|
||||||
|
is possible to program uC with slow oscillator.
|
||||||
|
Contributed by Galen Seitz
|
||||||
|
|
||||||
|
* Baudrate for serial programmers (STK500 and AVR910) is
|
||||||
|
configurable in the config or at the command-line.
|
||||||
|
This way some more tweaked bootloaders and programmers can be used.
|
||||||
|
|
||||||
|
* Deprecated options have been removed.
|
||||||
|
Now the "-U" option must be used.
|
||||||
|
|
||||||
|
* MacOS X now supported by default.
|
||||||
|
|
||||||
|
Version 4.3.0:
|
||||||
|
|
||||||
|
* Added support for "Butterfly" evaluation board.
|
||||||
|
|
||||||
|
* Make cycle-count work with AVR910-programmers.
|
||||||
|
|
||||||
|
* Added "Troubleshooting"-Appendix to the manual.
|
||||||
|
|
||||||
|
* Add ATmega8515 support.
|
||||||
|
Contributed by: Matthias Weißer <matthias@matwei.de>
|
||||||
|
|
||||||
|
* Add ATmega64 support.
|
||||||
|
Contributed by: Erik Christiansen <erik@dd.nec.com.au>
|
||||||
|
|
||||||
|
* Improved polling algorithm to speed up
|
||||||
|
programming of byte oriented parallel programmers.
|
||||||
|
Contributed by: Jan-Hinnerk Reichert <jan-hinnerk_reichert@hamburg.de>
|
||||||
|
|
||||||
|
* Add "fuse" and "lock" definitions for the AT90S8535.
|
||||||
|
|
||||||
|
* STK500 skips empty pages in paged write resulting in faster downloads
|
||||||
|
when there are empty blocks in between code (such as files that contain
|
||||||
|
application code and bootloader code).
|
||||||
|
|
||||||
|
Version 4.2.0:
|
||||||
|
|
||||||
|
* Add basic support for reading and writing fuses via SPI with avr910
|
||||||
|
programmers. Submitted by
|
||||||
|
Jan-Hinnerk Reichert <jan-hinnerk_reichert@hamburg.de>.
|
||||||
|
|
||||||
|
* Perform an auto erase before programming if the flash memory is
|
||||||
|
anywhere specified to be written by any of the -U requests. Old
|
||||||
|
style memory specification options (-f, -i, -I, -m, and -o) are
|
||||||
|
deprecated in favor of the new -U options. Auto erase is disabled
|
||||||
|
if any of the old-style options (specifically -i and -o) are
|
||||||
|
specified.
|
||||||
|
|
||||||
|
* Add new -U option for specifying programming operations - allows
|
||||||
|
multiple memory operations on a single command line.
|
||||||
|
|
||||||
|
* New progress reporting, looks nicer and is nicer to wrapper
|
||||||
|
environments such as emacs.
|
||||||
|
|
||||||
|
* Fix long-standing timing (verify) problems on Windows platform.
|
||||||
|
Submitted by Alex Shepherd <ashepherd@wave.co.nz>.
|
||||||
|
|
||||||
|
* Add new file format option - 'm' for "immediate mode." In this
|
||||||
|
case, the filename argument of the -o, -i, or -U options is
|
||||||
|
treated as the data for uploading - useful for specifying fuse
|
||||||
|
bits without having to create a single-byte file for uploading.
|
||||||
|
|
||||||
|
* Add support for displaying and setting the various STK500 operational
|
||||||
|
parameters (Vtarget, Varef, Master clock).
|
||||||
|
|
||||||
|
* Add 'picoweb' programming cable programmer.
|
||||||
|
Contributed by Rune Christensen <rune.christensen@adslhome.dk>.
|
||||||
|
|
||||||
|
* Add support for the sp12 programmer. Submitted by
|
||||||
|
Larry Barello <larryba@barrello.net>.
|
||||||
|
|
||||||
|
|
||||||
|
Version 4.1.0
|
||||||
|
|
||||||
|
* Add support for the Bascom SAMPLE programmer. Submitted by
|
||||||
|
Larry Barello <larryba@barrello.net>.
|
||||||
|
|
||||||
|
* Add support for avr910 type programmers (mcu00100, pavr avr910, etc).
|
||||||
|
|
||||||
|
* Support new devices: ATmega8535, ATtiny26
|
||||||
|
|
||||||
|
|
||||||
|
Version 4.0.0
|
||||||
|
|
||||||
|
* Now support Linux - added by "Theodore A. Roth" <troth@openavr.org>.
|
||||||
|
|
||||||
|
* Now support Windows - added by "Eric B. Weddington" <eric@ecentral.com>.
|
||||||
|
|
||||||
|
* Use 'configure' scripts to tailor the code to the system avrdude
|
||||||
|
is getting ready to be compiled on - added by "Theodore A. Roth"
|
||||||
|
<troth@openavr.org>.
|
||||||
|
|
||||||
|
* Motorola S-Record support - submitted by "Alexey V.Levdikov "
|
||||||
|
<tsar@kemford.com>.
|
||||||
|
|
||||||
|
* Support parallel programming on the STK500. Introduce 'pagel' and
|
||||||
|
'bs2' keywords to the config file for this purpose.
|
||||||
|
|
||||||
|
* Add support for the AT90S2343
|
||||||
|
|
||||||
|
* Add support for the ATmega169
|
||||||
|
|
||||||
|
* Add ability to specify system defaults within the config file
|
||||||
|
(default parallel port, default serial port).
|
||||||
|
|
||||||
|
* Specify the default programmer seperately from the programmer
|
||||||
|
definition. This is now done in the config file using the
|
||||||
|
'default_programmer' keyword.
|
||||||
|
|
||||||
|
* Support a per-user config file (~/.avrduderc) so that one can
|
||||||
|
override system wide defaults if desired.
|
||||||
|
|
||||||
|
* Follow the datasheet more closely for several parts in the "retry"
|
||||||
|
code when entering programming mode fails initially. Introduce
|
||||||
|
'retry_pulse' to the config file for this purpose.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Version 3.1.0
|
||||||
|
|
||||||
|
* This change represents a name change only. There is currently an
|
||||||
|
effort to port AVRPROG to other platforms including Linux and
|
||||||
|
Windows. Since Atmel's programmer binary that's included within
|
||||||
|
their AVR Studio software is named AVRPROG.EXE on the Windows OS,
|
||||||
|
there is the chance for confusion if we keep calling this program
|
||||||
|
AVRPROG as well. Up until now the name hasn't really been a
|
||||||
|
problem since there was no chance to confuse 'avrprog' on Unix
|
||||||
|
with Atmel's AVRPROG because Atmel's tools only run on Windows.
|
||||||
|
But with the Unix 'avrprog' possibly being ported to Windows, I
|
||||||
|
felt a name change was the best way to avoid problems.
|
||||||
|
|
||||||
|
So - from this point forward, my FreeBSD Unix program formerly
|
||||||
|
known as AVRPROG will subsequently be known as AVRDUDE (AVR
|
||||||
|
Downloader/UploaDEr).
|
||||||
|
|
||||||
|
This change also represents a time when the AVRDUDE sources move
|
||||||
|
from my own private repository to a public repository. This will
|
||||||
|
give other developers a chance to port AVRDUDE to other platforms
|
||||||
|
and extend its functionality to support additional programming
|
||||||
|
hardware, etc.
|
||||||
|
|
||||||
|
So goodbye AVRPROG, welcome AVRDUDE!
|
||||||
|
|
||||||
|
Version 3.0.0
|
||||||
|
|
||||||
|
* Rewrite parts of the code to make it easy to support other types
|
||||||
|
of programmers besides the directly connected parallel port
|
||||||
|
programmer (PPI).
|
||||||
|
|
||||||
|
* Add support for Atmel's STK500 programmer/development board. The
|
||||||
|
STK500's "paged mode" read/write is supported which makes this
|
||||||
|
programmer very fast. This is sorely needed on parts with large
|
||||||
|
memories such as the ATmega128. My 12K test program burns in
|
||||||
|
about 5 seconds, add another 5 to read it back out for
|
||||||
|
verification.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.1.5:
|
||||||
|
|
||||||
|
* When getting ready to initiate communications with the AVR device,
|
||||||
|
first pull /RESET low for a short period of time before enabling
|
||||||
|
the buffer chip. This sequence allows the AVR to be reset before
|
||||||
|
the buffer is enabled to avoid a short period of time where the
|
||||||
|
AVR may be driving the programming lines at the same time the
|
||||||
|
programmer tries to. Of course, if a buffer is being used, then
|
||||||
|
the /RESET line from the programmer needs to be directly connected
|
||||||
|
to the AVR /RESET line and not via the buffer chip.
|
||||||
|
|
||||||
|
Feature contributed by Rick C. Petty <rick@KIWI-Computer.com>.
|
||||||
|
|
||||||
|
* When in interactive terminal mode and dumping memory using the
|
||||||
|
'dump <memtype>' command without any address information, and the
|
||||||
|
end of memory is reached, wrap back around to zero on the next
|
||||||
|
invocation.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.1.4:
|
||||||
|
|
||||||
|
* Fix -Y option.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.1.3:
|
||||||
|
|
||||||
|
* Be backward compatible when reading 2-byte rewrite cycle counters
|
||||||
|
as written by avrprog version 2.1.0. Version 2.1.1 changed over
|
||||||
|
to a 4-byte counter, which caused avrprog versions 2.1.1 and 2.1.2
|
||||||
|
to report a negative count for parts that were initialized using
|
||||||
|
version 2.1.0. Thanks to Joerg Wunsch for noticing this.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.1.2:
|
||||||
|
|
||||||
|
* Add '-V' option to disable automatic verify check with uploading
|
||||||
|
data.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.1.1:
|
||||||
|
|
||||||
|
* Fix ATmega128 instruction sequences for reading fuse bits -
|
||||||
|
contributed by Joerg Wunsch.
|
||||||
|
|
||||||
|
* Modify erase-rewrite cycle counter code to use a 4 byte counter
|
||||||
|
instead of a two byte counter.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.1.0:
|
||||||
|
|
||||||
|
* Implement a per-part erase-rewrite cycle counter; requires the use
|
||||||
|
of two bytes of EEPROM memory.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.0.5:
|
||||||
|
|
||||||
|
* Support for ATtiny15 - contributed by Asher Hoskins
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.0.4:
|
||||||
|
|
||||||
|
* Config file fixes for various parts.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.0.3:
|
||||||
|
|
||||||
|
* Work around problem programming fuse bits on parts like the
|
||||||
|
at90s4433 as described in the following errata:
|
||||||
|
|
||||||
|
http://www.atmel.com/atmel/acrobat/doc1280.pdf
|
||||||
|
|
||||||
|
* Add part definition for at90s4414, at90s4433.
|
||||||
|
|
||||||
|
* Add fuse/lock bit memory instructions for the at90s1200,
|
||||||
|
at90s2333, at90s4433 and at90s8515.
|
||||||
|
|
||||||
|
* Fix setting of programmer status LEDs under certain write-fail
|
||||||
|
conditions.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.0.2 :
|
||||||
|
|
||||||
|
* Fix writing to read-only memories such as the lock bits of the
|
||||||
|
AT90S2313.
|
||||||
|
|
||||||
|
* Copyright updates.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.0.1 :
|
||||||
|
|
||||||
|
* Use correct parallel port pins for VCC.
|
||||||
|
|
||||||
|
* Add programmer definition for Atmel's STK200.
|
||||||
|
|
||||||
|
* Add programmer definition for the AVR3 board.
|
||||||
|
|
||||||
|
* Fix address bit encoding for many parts.
|
||||||
|
|
||||||
|
* Allow the ``BUFF'' signal to be asserted by multiple pins of the
|
||||||
|
parallel port (like VCC) instead of just one. The STK200 appears
|
||||||
|
to need this feature.
|
||||||
|
|
||||||
|
|
||||||
|
Version 2.0.0 :
|
||||||
|
|
||||||
|
* Add support for programming fuse and lock bits if supported by the
|
||||||
|
part.
|
||||||
|
|
||||||
|
* Move instruction encoding into the config file. Now any part can
|
||||||
|
be supported as long as it uses the same basic serial programming
|
||||||
|
instruction format.
|
||||||
|
|
||||||
|
* Add part definitions for the ATMega163 and ATMega8 devices.
|
||||||
|
|
||||||
|
|
||||||
|
Version 1.4.3 :
|
||||||
|
|
||||||
|
* Mostly internal code cleanup.
|
||||||
|
|
||||||
|
|
||||||
|
Version 1.4.2 :
|
||||||
|
|
||||||
|
* Fixes for ATMega paged memory support.
|
||||||
|
|
||||||
|
* Support for ATMega16 device.
|
||||||
|
|
||||||
|
|
||||||
|
Version 1.4.1 :
|
||||||
|
|
||||||
|
* No functional changes, update to Copyrights only.
|
||||||
|
|
||||||
|
|
||||||
|
Version 1.4.0 :
|
||||||
|
|
||||||
|
* Add part definitions to the config file.
|
||||||
|
|
||||||
|
* Add initial support for Atmel's ATMega paged memory parts.
|
||||||
|
|
||||||
|
* Config file documentation added.
|
||||||
|
|
||||||
|
* Add a definition for the Dontronics DT006 programmer.
|
||||||
|
|
||||||
|
* Fix Intel Hex support for addresses larger than 64k.
|
||||||
|
|
||||||
|
|
||||||
|
Version 1.3.0 :
|
||||||
|
|
||||||
|
* Make programmer pin assignments configurable.
|
||||||
|
|
||||||
|
|
||||||
|
Version 1.2.2 :
|
||||||
|
|
||||||
|
* Initial public release.
|
|
@ -0,0 +1,6 @@
|
||||||
|
See the documentation file for the details.
|
||||||
|
|
||||||
|
The latest version of AVRDUDE is always available here:
|
||||||
|
|
||||||
|
http://savannah.nongnu.org/projects/avrdude
|
||||||
|
|
|
@ -0,0 +1,131 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2009 Lars Immisch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avrdude interface for Arduino programmer
|
||||||
|
*
|
||||||
|
* The Arduino programmer is mostly a STK500v1, just the signature bytes
|
||||||
|
* are read differently.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "stk500_private.h"
|
||||||
|
#include "stk500.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
/* read signature bytes - arduino version */
|
||||||
|
static int arduino_read_sig_bytes(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m)
|
||||||
|
{
|
||||||
|
unsigned char buf[32];
|
||||||
|
|
||||||
|
/* Signature byte reads are always 3 bytes. */
|
||||||
|
|
||||||
|
if (m->size < 3) {
|
||||||
|
fprintf(stderr, "%s: memsize too small for sig byte read", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf[0] = Cmnd_STK_READ_SIGN;
|
||||||
|
buf[1] = Sync_CRC_EOP;
|
||||||
|
|
||||||
|
serial_send(&pgm->fd, buf, 2);
|
||||||
|
|
||||||
|
if (serial_recv(&pgm->fd, buf, 5) < 0)
|
||||||
|
return -1;
|
||||||
|
if (buf[0] == Resp_STK_NOSYNC) {
|
||||||
|
fprintf(stderr, "%s: stk500_cmd(): programmer is out of sync\n",
|
||||||
|
progname);
|
||||||
|
return -1;
|
||||||
|
} else if (buf[0] != Resp_STK_INSYNC) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"\n%s: arduino_read_sig_bytes(): (a) protocol error, "
|
||||||
|
"expect=0x%02x, resp=0x%02x\n",
|
||||||
|
progname, Resp_STK_INSYNC, buf[0]);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
if (buf[4] != Resp_STK_OK) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"\n%s: arduino_read_sig_bytes(): (a) protocol error, "
|
||||||
|
"expect=0x%02x, resp=0x%02x\n",
|
||||||
|
progname, Resp_STK_OK, buf[4]);
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
|
||||||
|
m->buf[0] = buf[1];
|
||||||
|
m->buf[1] = buf[2];
|
||||||
|
m->buf[2] = buf[3];
|
||||||
|
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int arduino_open(PROGRAMMER * pgm, char * port)
|
||||||
|
{
|
||||||
|
strcpy(pgm->port, port);
|
||||||
|
if (serial_open(port, pgm->baudrate? pgm->baudrate: 115200, &pgm->fd)==-1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Clear DTR and RTS to unload the RESET capacitor
|
||||||
|
* (for example in Arduino) */
|
||||||
|
serial_set_dtr_rts(&pgm->fd, 0);
|
||||||
|
usleep(50*1000);
|
||||||
|
/* Set DTR and RTS back to high */
|
||||||
|
serial_set_dtr_rts(&pgm->fd, 1);
|
||||||
|
usleep(50*1000);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* drain any extraneous input
|
||||||
|
*/
|
||||||
|
stk500_drain(pgm, 0);
|
||||||
|
|
||||||
|
if (stk500_getsync(pgm) < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void arduino_close(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
serial_set_dtr_rts(&pgm->fd, 0);
|
||||||
|
serial_close(&pgm->fd);
|
||||||
|
pgm->fd.ifd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void arduino_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/* This is mostly a STK500; just the signature is read
|
||||||
|
differently than on real STK500v1
|
||||||
|
and the DTR signal is set when opening the serial port
|
||||||
|
for the Auto-Reset feature */
|
||||||
|
stk500_initpgm(pgm);
|
||||||
|
|
||||||
|
strcpy(pgm->type, "Arduino");
|
||||||
|
pgm->read_sig_bytes = arduino_read_sig_bytes;
|
||||||
|
pgm->open = arduino_open;
|
||||||
|
pgm->close = arduino_close;
|
||||||
|
}
|
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2009 Lars Immisch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef arduino_h__
|
||||||
|
#define arduino_h__
|
||||||
|
|
||||||
|
void arduino_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,77 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef __avr_h__
|
||||||
|
#define __avr_h__
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "avrpart.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
|
||||||
|
typedef void (*FP_UpdateProgress)(int percent, double etime, char *hdr);
|
||||||
|
|
||||||
|
extern struct avrpart parts[];
|
||||||
|
|
||||||
|
extern FP_UpdateProgress update_progress;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int avr_tpi_poll_nvmbsy(PROGRAMMER *pgm);
|
||||||
|
int avr_read_byte_default(PROGRAMMER * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char * value);
|
||||||
|
|
||||||
|
int avr_read(PROGRAMMER * pgm, AVRPART * p, char * memtype, int size,
|
||||||
|
int verbose);
|
||||||
|
|
||||||
|
int avr_write_page(PROGRAMMER * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr);
|
||||||
|
|
||||||
|
int avr_write_byte(PROGRAMMER * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char data);
|
||||||
|
|
||||||
|
int avr_write_byte_default(PROGRAMMER * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char data);
|
||||||
|
|
||||||
|
int avr_write(PROGRAMMER * pgm, AVRPART * p, char * memtype, int size,
|
||||||
|
int verbose);
|
||||||
|
|
||||||
|
int avr_signature(PROGRAMMER * pgm, AVRPART * p);
|
||||||
|
|
||||||
|
int avr_verify(AVRPART * p, AVRPART * v, char * memtype, int size);
|
||||||
|
|
||||||
|
int avr_get_cycle_count(PROGRAMMER * pgm, AVRPART * p, int * cycles);
|
||||||
|
|
||||||
|
int avr_put_cycle_count(PROGRAMMER * pgm, AVRPART * p, int cycles);
|
||||||
|
|
||||||
|
int avr_mem_hiaddr(AVRMEM * mem);
|
||||||
|
|
||||||
|
int avr_chip_erase(PROGRAMMER * pgm, AVRPART * p);
|
||||||
|
|
||||||
|
void report_progress (int completed, int total, char *hdr);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,798 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
* Copyright 2007 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
* Copyright 2008 Klaus Leidinger <klaus@mikrocontroller-projekte.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avrdude interface for Atmel Low Cost Serial programmers which adher to the
|
||||||
|
* protocol described in application note avr910.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "config.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "avr910.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Private data for this programmer.
|
||||||
|
*/
|
||||||
|
struct pdata
|
||||||
|
{
|
||||||
|
char has_auto_incr_addr;
|
||||||
|
unsigned char devcode;
|
||||||
|
unsigned int buffersize;
|
||||||
|
unsigned char test_blockmode;
|
||||||
|
unsigned char use_blockmode;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define PDATA(pgm) ((struct pdata *)(pgm->cookie))
|
||||||
|
|
||||||
|
static void avr910_setup(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
if ((pgm->cookie = malloc(sizeof(struct pdata))) == 0) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_setup(): Out of memory allocating private data\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
memset(pgm->cookie, 0, sizeof(struct pdata));
|
||||||
|
PDATA(pgm)->test_blockmode = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void avr910_teardown(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
free(pgm->cookie);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_send(PROGRAMMER * pgm, char * buf, size_t len)
|
||||||
|
{
|
||||||
|
return serial_send(&pgm->fd, (unsigned char *)buf, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_recv(PROGRAMMER * pgm, char * buf, size_t len)
|
||||||
|
{
|
||||||
|
int rv;
|
||||||
|
|
||||||
|
rv = serial_recv(&pgm->fd, (unsigned char *)buf, len);
|
||||||
|
if (rv < 0) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_recv(): programmer is not responding\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_drain(PROGRAMMER * pgm, int display)
|
||||||
|
{
|
||||||
|
return serial_drain(&pgm->fd, display);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_vfy_cmd_sent(PROGRAMMER * pgm, char * errmsg)
|
||||||
|
{
|
||||||
|
char c;
|
||||||
|
|
||||||
|
avr910_recv(pgm, &c, 1);
|
||||||
|
if (c != '\r') {
|
||||||
|
fprintf(stderr, "%s: error: programmer did not respond to command: %s\n",
|
||||||
|
progname, errmsg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* issue the 'chip erase' command to the AVR device
|
||||||
|
*/
|
||||||
|
static int avr910_chip_erase(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
avr910_send(pgm, "e", 1);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "chip erase");
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avr910 firmware may not delay long enough
|
||||||
|
*/
|
||||||
|
usleep (p->chip_erase_delay);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_enter_prog_mode(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
avr910_send(pgm, "P", 1);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "enter prog mode");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_leave_prog_mode(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
avr910_send(pgm, "L", 1);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "leave prog mode");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* issue the 'program enable' command to the AVR device
|
||||||
|
*/
|
||||||
|
static int avr910_program_enable(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* initialize the AVR device and prepare it to accept commands
|
||||||
|
*/
|
||||||
|
static int avr910_initialize(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
char id[8];
|
||||||
|
char sw[2];
|
||||||
|
char hw[2];
|
||||||
|
char buf[10];
|
||||||
|
char type;
|
||||||
|
char c, devtype_1st;
|
||||||
|
int dev_supported = 0;
|
||||||
|
AVRPART * part;
|
||||||
|
|
||||||
|
/* Get the programmer identifier. Programmer returns exactly 7 chars
|
||||||
|
_without_ the null.*/
|
||||||
|
|
||||||
|
avr910_send(pgm, "S", 1);
|
||||||
|
memset (id, 0, sizeof(id));
|
||||||
|
avr910_recv(pgm, id, sizeof(id)-1);
|
||||||
|
|
||||||
|
/* Get the HW and SW versions to see if the programmer is present. */
|
||||||
|
|
||||||
|
avr910_send(pgm, "V", 1);
|
||||||
|
avr910_recv(pgm, sw, sizeof(sw));
|
||||||
|
|
||||||
|
avr910_send(pgm, "v", 1);
|
||||||
|
avr910_recv(pgm, hw, sizeof(hw));
|
||||||
|
|
||||||
|
/* Get the programmer type (serial or parallel). Expect serial. */
|
||||||
|
|
||||||
|
avr910_send(pgm, "p", 1);
|
||||||
|
avr910_recv(pgm, &type, 1);
|
||||||
|
|
||||||
|
fprintf(stderr, "Found programmer: Id = \"%s\"; type = %c\n", id, type);
|
||||||
|
fprintf(stderr, " Software Version = %c.%c; ", sw[0], sw[1]);
|
||||||
|
fprintf(stderr, "Hardware Version = %c.%c\n", hw[0], hw[1]);
|
||||||
|
|
||||||
|
/* See if programmer supports autoincrement of address. */
|
||||||
|
|
||||||
|
avr910_send(pgm, "a", 1);
|
||||||
|
avr910_recv(pgm, &PDATA(pgm)->has_auto_incr_addr, 1);
|
||||||
|
if (PDATA(pgm)->has_auto_incr_addr == 'Y')
|
||||||
|
fprintf(stderr, "Programmer supports auto addr increment.\n");
|
||||||
|
|
||||||
|
/* Check support for buffered memory access, ignore if not available */
|
||||||
|
|
||||||
|
if (PDATA(pgm)->test_blockmode == 1) {
|
||||||
|
avr910_send(pgm, "b", 1);
|
||||||
|
avr910_recv(pgm, &c, 1);
|
||||||
|
if (c == 'Y') {
|
||||||
|
avr910_recv(pgm, &c, 1);
|
||||||
|
PDATA(pgm)->buffersize = (unsigned int)(unsigned char)c<<8;
|
||||||
|
avr910_recv(pgm, &c, 1);
|
||||||
|
PDATA(pgm)->buffersize += (unsigned int)(unsigned char)c;
|
||||||
|
fprintf(stderr,
|
||||||
|
"Programmer supports buffered memory access with "
|
||||||
|
"buffersize = %u bytes.\n",
|
||||||
|
PDATA(pgm)->buffersize);
|
||||||
|
PDATA(pgm)->use_blockmode = 1;
|
||||||
|
} else {
|
||||||
|
PDATA(pgm)->use_blockmode = 0;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
PDATA(pgm)->use_blockmode = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (PDATA(pgm)->devcode == 0) {
|
||||||
|
|
||||||
|
/* Get list of devices that the programmer supports. */
|
||||||
|
|
||||||
|
avr910_send(pgm, "t", 1);
|
||||||
|
fprintf(stderr, "\nProgrammer supports the following devices:\n");
|
||||||
|
devtype_1st = 0;
|
||||||
|
while (1) {
|
||||||
|
avr910_recv(pgm, &c, 1);
|
||||||
|
if (devtype_1st == 0)
|
||||||
|
devtype_1st = c;
|
||||||
|
if (c == 0)
|
||||||
|
break;
|
||||||
|
part = locate_part_by_avr910_devcode(part_list, c);
|
||||||
|
|
||||||
|
fprintf(stderr, " Device code: 0x%02x = %s\n", c, part ? part->desc : "(unknown)");
|
||||||
|
|
||||||
|
/* FIXME: Need to lookup devcode and report the device. */
|
||||||
|
|
||||||
|
if (p->avr910_devcode == c)
|
||||||
|
dev_supported = 1;
|
||||||
|
};
|
||||||
|
fprintf(stderr,"\n");
|
||||||
|
|
||||||
|
if (!dev_supported) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: %s: selected device is not supported by programmer: %s\n",
|
||||||
|
progname, ovsigck? "warning": "error", p->id);
|
||||||
|
if (!ovsigck)
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
/* If the user forced the selection, use the first device
|
||||||
|
type that is supported by the programmer. */
|
||||||
|
buf[1] = ovsigck? devtype_1st: p->avr910_devcode;
|
||||||
|
} else {
|
||||||
|
/* devcode overridden by -x devcode= option */
|
||||||
|
buf[1] = (char)(PDATA(pgm)->devcode);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tell the programmer which part we selected. */
|
||||||
|
buf[0] = 'T';
|
||||||
|
/* buf[1] has been set up above */
|
||||||
|
|
||||||
|
avr910_send(pgm, buf, 2);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "select device");
|
||||||
|
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_devcode selected: 0x%02x\n",
|
||||||
|
progname, (unsigned)buf[1]);
|
||||||
|
|
||||||
|
avr910_enter_prog_mode(pgm);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_disable(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_enable(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* transmit an AVR device command and return the results; 'cmd' and
|
||||||
|
* 'res' must point to at least a 4 byte data buffer
|
||||||
|
*/
|
||||||
|
static int avr910_cmd(PROGRAMMER * pgm, unsigned char cmd[4],
|
||||||
|
unsigned char res[4])
|
||||||
|
{
|
||||||
|
char buf[5];
|
||||||
|
|
||||||
|
/* FIXME: Insert version check here */
|
||||||
|
|
||||||
|
buf[0] = '.'; /* New Universal Command */
|
||||||
|
buf[1] = cmd[0];
|
||||||
|
buf[2] = cmd[1];
|
||||||
|
buf[3] = cmd[2];
|
||||||
|
buf[4] = cmd[3];
|
||||||
|
|
||||||
|
avr910_send (pgm, buf, 5);
|
||||||
|
avr910_recv (pgm, buf, 2);
|
||||||
|
|
||||||
|
res[0] = 0x00; /* Dummy value */
|
||||||
|
res[1] = cmd[0];
|
||||||
|
res[2] = cmd[1];
|
||||||
|
res[3] = buf[0];
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_parseextparms(PROGRAMMER * pgm, LISTID extparms)
|
||||||
|
{
|
||||||
|
LNODEID ln;
|
||||||
|
const char *extended_param;
|
||||||
|
int rv = 0;
|
||||||
|
|
||||||
|
for (ln = lfirst(extparms); ln; ln = lnext(ln)) {
|
||||||
|
extended_param = ldata(ln);
|
||||||
|
|
||||||
|
if (strncmp(extended_param, "devcode=", strlen("devcode=")) == 0) {
|
||||||
|
int devcode;
|
||||||
|
if (sscanf(extended_param, "devcode=%i", &devcode) != 1 ||
|
||||||
|
devcode <= 0 || devcode > 255) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_parseextparms(): invalid devcode '%s'\n",
|
||||||
|
progname, extended_param);
|
||||||
|
rv = -1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (verbose >= 2) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_parseextparms(): devcode overwritten as 0x%02x\n",
|
||||||
|
progname, devcode);
|
||||||
|
}
|
||||||
|
PDATA(pgm)->devcode = devcode;
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (strncmp(extended_param, "no_blockmode", strlen("no_blockmode")) == 0) {
|
||||||
|
if (verbose >= 2) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_parseextparms(-x): no testing for Blockmode\n",
|
||||||
|
progname);
|
||||||
|
}
|
||||||
|
PDATA(pgm)->test_blockmode = 0;
|
||||||
|
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avr910_parseextparms(): invalid extended parameter '%s'\n",
|
||||||
|
progname, extended_param);
|
||||||
|
rv = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_open(PROGRAMMER * pgm, char * port)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* If baudrate was not specified use 19.200 Baud
|
||||||
|
*/
|
||||||
|
if(pgm->baudrate == 0) {
|
||||||
|
pgm->baudrate = 19200;
|
||||||
|
}
|
||||||
|
|
||||||
|
strcpy(pgm->port, port);
|
||||||
|
if (serial_open(port, pgm->baudrate, &pgm->fd)==-1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* drain any extraneous input
|
||||||
|
*/
|
||||||
|
avr910_drain (pgm, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void avr910_close(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
avr910_leave_prog_mode(pgm);
|
||||||
|
|
||||||
|
serial_close(&pgm->fd);
|
||||||
|
pgm->fd.ifd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_display(PROGRAMMER * pgm, const char * p)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void avr910_set_addr(PROGRAMMER * pgm, unsigned long addr)
|
||||||
|
{
|
||||||
|
char cmd[3];
|
||||||
|
|
||||||
|
cmd[0] = 'A';
|
||||||
|
cmd[1] = (addr >> 8) & 0xff;
|
||||||
|
cmd[2] = addr & 0xff;
|
||||||
|
|
||||||
|
avr910_send(pgm, cmd, sizeof(cmd));
|
||||||
|
avr910_vfy_cmd_sent(pgm, "set addr");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_write_byte(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char value)
|
||||||
|
{
|
||||||
|
char cmd[2];
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "flash") == 0) {
|
||||||
|
if (addr & 0x01) {
|
||||||
|
cmd[0] = 'C'; /* Write Program Mem high byte */
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
cmd[0] = 'c';
|
||||||
|
}
|
||||||
|
|
||||||
|
addr >>= 1;
|
||||||
|
}
|
||||||
|
else if (strcmp(m->desc, "eeprom") == 0) {
|
||||||
|
cmd[0] = 'D';
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return avr_write_byte_default(pgm, p, m, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd[1] = value;
|
||||||
|
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
|
||||||
|
avr910_send(pgm, cmd, sizeof(cmd));
|
||||||
|
avr910_vfy_cmd_sent(pgm, "write byte");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_read_byte_flash(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
char buf[2];
|
||||||
|
|
||||||
|
avr910_set_addr(pgm, addr >> 1);
|
||||||
|
|
||||||
|
avr910_send(pgm, "R", 1);
|
||||||
|
|
||||||
|
/* Read back the program mem word (MSB first) */
|
||||||
|
avr910_recv(pgm, buf, sizeof(buf));
|
||||||
|
|
||||||
|
if ((addr & 0x01) == 0) {
|
||||||
|
*value = buf[1];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
*value = buf[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_read_byte_eeprom(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
avr910_send(pgm, "d", 1);
|
||||||
|
avr910_recv(pgm, (char *)value, 1);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_read_byte(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
if (strcmp(m->desc, "flash") == 0) {
|
||||||
|
return avr910_read_byte_flash(pgm, p, m, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "eeprom") == 0) {
|
||||||
|
return avr910_read_byte_eeprom(pgm, p, m, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return avr_read_byte_default(pgm, p, m, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_paged_write_flash(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes)
|
||||||
|
{
|
||||||
|
unsigned char cmd[] = {'c', 'C'};
|
||||||
|
char buf[2];
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr = n_bytes;
|
||||||
|
unsigned int page_addr;
|
||||||
|
int page_bytes = page_size;
|
||||||
|
int page_wr_cmd_pending = 0;
|
||||||
|
|
||||||
|
page_addr = addr;
|
||||||
|
avr910_set_addr(pgm, addr>>1);
|
||||||
|
|
||||||
|
while (addr < max_addr) {
|
||||||
|
page_wr_cmd_pending = 1;
|
||||||
|
buf[0] = cmd[addr & 0x01];
|
||||||
|
buf[1] = m->buf[addr];
|
||||||
|
avr910_send(pgm, buf, sizeof(buf));
|
||||||
|
avr910_vfy_cmd_sent(pgm, "write byte");
|
||||||
|
|
||||||
|
addr++;
|
||||||
|
page_bytes--;
|
||||||
|
|
||||||
|
if (m->paged && (page_bytes == 0)) {
|
||||||
|
/* Send the "Issue Page Write" if we have sent a whole page. */
|
||||||
|
|
||||||
|
avr910_set_addr(pgm, page_addr>>1);
|
||||||
|
avr910_send(pgm, "m", 1);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "flush page");
|
||||||
|
|
||||||
|
page_wr_cmd_pending = 0;
|
||||||
|
usleep(m->max_write_delay);
|
||||||
|
avr910_set_addr(pgm, addr>>1);
|
||||||
|
|
||||||
|
/* Set page address for next page. */
|
||||||
|
|
||||||
|
page_addr = addr;
|
||||||
|
page_bytes = page_size;
|
||||||
|
}
|
||||||
|
else if ((PDATA(pgm)->has_auto_incr_addr != 'Y') && ((addr & 0x01) == 0)) {
|
||||||
|
avr910_set_addr(pgm, addr>>1);
|
||||||
|
}
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If we didn't send the page wr cmd after the last byte written in the
|
||||||
|
loop, send it now. */
|
||||||
|
|
||||||
|
if (page_wr_cmd_pending) {
|
||||||
|
avr910_set_addr(pgm, page_addr>>1);
|
||||||
|
avr910_send(pgm, "m", 1);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "flush final page");
|
||||||
|
usleep(m->max_write_delay);
|
||||||
|
}
|
||||||
|
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_paged_write_eeprom(PROGRAMMER * pgm, AVRPART * p,
|
||||||
|
AVRMEM * m, int page_size, int n_bytes)
|
||||||
|
{
|
||||||
|
char cmd[2];
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr = n_bytes;
|
||||||
|
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
|
||||||
|
cmd[0] = 'D';
|
||||||
|
|
||||||
|
while (addr < max_addr) {
|
||||||
|
cmd[1] = m->buf[addr];
|
||||||
|
avr910_send(pgm, cmd, sizeof(cmd));
|
||||||
|
avr910_vfy_cmd_sent(pgm, "write byte");
|
||||||
|
usleep(m->max_write_delay);
|
||||||
|
|
||||||
|
addr++;
|
||||||
|
|
||||||
|
if (PDATA(pgm)->has_auto_incr_addr != 'Y') {
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_paged_write(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes)
|
||||||
|
{
|
||||||
|
int rval = 0;
|
||||||
|
if (PDATA(pgm)->use_blockmode == 0) {
|
||||||
|
if (strcmp(m->desc, "flash") == 0) {
|
||||||
|
rval = avr910_paged_write_flash(pgm, p, m, page_size, n_bytes);
|
||||||
|
} else if (strcmp(m->desc, "eeprom") == 0) {
|
||||||
|
rval = avr910_paged_write_eeprom(pgm, p, m, page_size, n_bytes);
|
||||||
|
} else {
|
||||||
|
rval = -2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (PDATA(pgm)->use_blockmode == 1) {
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr = n_bytes;
|
||||||
|
char *cmd;
|
||||||
|
unsigned int blocksize = PDATA(pgm)->buffersize;
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "flash") && strcmp(m->desc, "eeprom"))
|
||||||
|
rval = -2;
|
||||||
|
|
||||||
|
if (m->desc[0] == 'e')
|
||||||
|
blocksize = 1; /* Write to eeprom single bytes only */
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
|
||||||
|
cmd = malloc(4 + blocksize);
|
||||||
|
if (!cmd) rval = -1;
|
||||||
|
cmd[0] = 'B';
|
||||||
|
cmd[3] = toupper((int)(m->desc[0]));
|
||||||
|
|
||||||
|
while (addr < max_addr) {
|
||||||
|
if ((max_addr - addr) < blocksize) {
|
||||||
|
blocksize = max_addr - addr;
|
||||||
|
};
|
||||||
|
memcpy(&cmd[4], &m->buf[addr], blocksize);
|
||||||
|
cmd[1] = (blocksize >> 8) & 0xff;
|
||||||
|
cmd[2] = blocksize & 0xff;
|
||||||
|
|
||||||
|
avr910_send(pgm, cmd, 4 + blocksize);
|
||||||
|
avr910_vfy_cmd_sent(pgm, "write block");
|
||||||
|
|
||||||
|
addr += blocksize;
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
} /* while */
|
||||||
|
free(cmd);
|
||||||
|
|
||||||
|
rval = addr;
|
||||||
|
}
|
||||||
|
return rval;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int avr910_paged_load(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes)
|
||||||
|
{
|
||||||
|
char cmd;
|
||||||
|
int rd_size = 1;
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr;
|
||||||
|
char buf[2];
|
||||||
|
int rval=0;
|
||||||
|
|
||||||
|
if (PDATA(pgm)->use_blockmode == 0) {
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "flash") == 0) {
|
||||||
|
cmd = 'R';
|
||||||
|
rd_size = 2; /* read two bytes per addr */
|
||||||
|
} else if (strcmp(m->desc, "eeprom") == 0) {
|
||||||
|
cmd = 'd';
|
||||||
|
rd_size = 1;
|
||||||
|
} else {
|
||||||
|
rval = -2;
|
||||||
|
}
|
||||||
|
|
||||||
|
max_addr = n_bytes/rd_size;
|
||||||
|
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
|
||||||
|
while (addr < max_addr) {
|
||||||
|
avr910_send(pgm, &cmd, 1);
|
||||||
|
if (cmd == 'R') {
|
||||||
|
/* The 'R' command returns two bytes, MSB first, we need to put the data
|
||||||
|
into the memory buffer LSB first. */
|
||||||
|
avr910_recv(pgm, buf, 2);
|
||||||
|
m->buf[addr*2] = buf[1]; /* LSB */
|
||||||
|
m->buf[addr*2+1] = buf[0]; /* MSB */
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
avr910_recv(pgm, (char *)&m->buf[addr], 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
addr++;
|
||||||
|
|
||||||
|
if (PDATA(pgm)->has_auto_incr_addr != 'Y') {
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
rval = addr * rd_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (PDATA(pgm)->use_blockmode == 1) {
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr = n_bytes;
|
||||||
|
int rd_size = 1;
|
||||||
|
|
||||||
|
/* check parameter syntax: only "flash" or "eeprom" is allowed */
|
||||||
|
if (strcmp(m->desc, "flash") && strcmp(m->desc, "eeprom"))
|
||||||
|
rval = -2;
|
||||||
|
|
||||||
|
/* use buffered mode */
|
||||||
|
char cmd[4];
|
||||||
|
int blocksize = PDATA(pgm)->buffersize;
|
||||||
|
|
||||||
|
cmd[0] = 'g';
|
||||||
|
cmd[3] = toupper((int)(m->desc[0]));
|
||||||
|
|
||||||
|
avr910_set_addr(pgm, addr);
|
||||||
|
|
||||||
|
while (addr < max_addr) {
|
||||||
|
if ((max_addr - addr) < blocksize) {
|
||||||
|
blocksize = max_addr - addr;
|
||||||
|
};
|
||||||
|
cmd[1] = (blocksize >> 8) & 0xff;
|
||||||
|
cmd[2] = blocksize & 0xff;
|
||||||
|
|
||||||
|
avr910_send(pgm, cmd, 4);
|
||||||
|
avr910_recv(pgm, (char *)&m->buf[addr], blocksize);
|
||||||
|
|
||||||
|
addr += blocksize;
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
rval = addr * rd_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Signature byte reads are always 3 bytes. */
|
||||||
|
|
||||||
|
static int avr910_read_sig_bytes(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m)
|
||||||
|
{
|
||||||
|
unsigned char tmp;
|
||||||
|
|
||||||
|
if (m->size < 3) {
|
||||||
|
fprintf(stderr, "%s: memsize too small for sig byte read", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
avr910_send(pgm, "s", 1);
|
||||||
|
avr910_recv(pgm, (char *)m->buf, 3);
|
||||||
|
/* Returned signature has wrong order. */
|
||||||
|
tmp = m->buf[2];
|
||||||
|
m->buf[2] = m->buf[0];
|
||||||
|
m->buf[0] = tmp;
|
||||||
|
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void avr910_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "avr910");
|
||||||
|
|
||||||
|
/*
|
||||||
|
* mandatory functions
|
||||||
|
*/
|
||||||
|
pgm->initialize = avr910_initialize;
|
||||||
|
pgm->display = avr910_display;
|
||||||
|
pgm->enable = avr910_enable;
|
||||||
|
pgm->disable = avr910_disable;
|
||||||
|
pgm->program_enable = avr910_program_enable;
|
||||||
|
pgm->chip_erase = avr910_chip_erase;
|
||||||
|
pgm->cmd = avr910_cmd;
|
||||||
|
pgm->open = avr910_open;
|
||||||
|
pgm->close = avr910_close;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* optional functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
pgm->write_byte = avr910_write_byte;
|
||||||
|
pgm->read_byte = avr910_read_byte;
|
||||||
|
|
||||||
|
pgm->paged_write = avr910_paged_write;
|
||||||
|
pgm->paged_load = avr910_paged_load;
|
||||||
|
|
||||||
|
pgm->read_sig_bytes = avr910_read_sig_bytes;
|
||||||
|
|
||||||
|
pgm->parseextparams = avr910_parseextparms;
|
||||||
|
pgm->setup = avr910_setup;
|
||||||
|
pgm->teardown = avr910_teardown;
|
||||||
|
}
|
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef avr910_h
|
||||||
|
#define avr910_h
|
||||||
|
|
||||||
|
#include "avrpart.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void avr910_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* avr910_h */
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,56 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2007 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef avrdude_h
|
||||||
|
#define avrdude_h
|
||||||
|
|
||||||
|
extern char * progname; /* name of program, for messages */
|
||||||
|
extern char progbuf[]; /* spaces same length as progname */
|
||||||
|
|
||||||
|
extern int do_cycles; /* track erase-rewrite cycles (-y) */
|
||||||
|
extern int ovsigck; /* override signature check (-F) */
|
||||||
|
extern int verbose; /* verbosity level (-v, -vv, ...) */
|
||||||
|
extern int quell_progress; /* quiteness level (-q, -qq) */
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(HAVE_USLEEP)
|
||||||
|
int usleep(unsigned int us);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(HAVE_GETTIMEOFDAY)
|
||||||
|
struct timezone;
|
||||||
|
int gettimeofday(struct timeval *tv, struct timezone *tz);
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#endif /* HAVE_GETTIMEOFDAY */
|
||||||
|
|
||||||
|
#endif /* defined(WIN32NATIVE) */
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,113 @@
|
||||||
|
## -*- mode: rpm-spec; -*-
|
||||||
|
##
|
||||||
|
## $Id$
|
||||||
|
##
|
||||||
|
## @configure_input@
|
||||||
|
##
|
||||||
|
|
||||||
|
%define debug_package %{nil}
|
||||||
|
|
||||||
|
%define _with_docs 1
|
||||||
|
%{?_without_docs: %define _with_docs 0}
|
||||||
|
|
||||||
|
Summary: AVRDUDE is software for programming Atmel AVR Microcontrollers.
|
||||||
|
Name: avrdude
|
||||||
|
Version: @VERSION@
|
||||||
|
Release: 1
|
||||||
|
URL: http://savannah.nongnu.org/projects/avrdude
|
||||||
|
Source0: %{name}-%{version}.tar.gz
|
||||||
|
License: GPL
|
||||||
|
Group: Development/Tools
|
||||||
|
BuildRoot: %{_tmppath}/%{name}-%{version}-root
|
||||||
|
|
||||||
|
%description
|
||||||
|
AVRDUDE is software for programming Atmel AVR Microcontrollers.
|
||||||
|
|
||||||
|
%if %{_with_docs}
|
||||||
|
## The avrdude-docs subpackage
|
||||||
|
%package docs
|
||||||
|
Summary: Documentation for AVRDUDE.
|
||||||
|
Group: Documentation
|
||||||
|
%description docs
|
||||||
|
Documentation for avrdude in info, html, postscript and pdf formats.
|
||||||
|
%endif
|
||||||
|
|
||||||
|
%prep
|
||||||
|
%setup -q
|
||||||
|
|
||||||
|
%build
|
||||||
|
|
||||||
|
./configure --prefix=%{_prefix} --sysconfdir=/etc --mandir=%{_mandir} \
|
||||||
|
--infodir=%{_infodir} \
|
||||||
|
%if %{_with_docs}
|
||||||
|
--enable-doc=yes
|
||||||
|
%else
|
||||||
|
--enable-doc=no
|
||||||
|
%endif
|
||||||
|
|
||||||
|
make
|
||||||
|
|
||||||
|
%install
|
||||||
|
rm -rf $RPM_BUILD_ROOT
|
||||||
|
make prefix=$RPM_BUILD_ROOT%{_prefix} \
|
||||||
|
sysconfdir=$RPM_BUILD_ROOT/etc \
|
||||||
|
mandir=$RPM_BUILD_ROOT%{_mandir} \
|
||||||
|
infodir=$RPM_BUILD_ROOT%{_infodir} \
|
||||||
|
install
|
||||||
|
|
||||||
|
rm -rf $RPM_BUILD_ROOT%{_datadir}/doc/%{name}-%{version}
|
||||||
|
rm -f $RPM_BUILD_ROOT%{_infodir}/dir
|
||||||
|
|
||||||
|
%clean
|
||||||
|
rm -rf $RPM_BUILD_ROOT
|
||||||
|
|
||||||
|
%if %{_with_docs}
|
||||||
|
%post docs
|
||||||
|
[ -f %{_infodir}/avrdude.info ] && \
|
||||||
|
/sbin/install-info %{_infodir}/avrdude.info %{_infodir}/dir || :
|
||||||
|
[ -f %{_infodir}/avrdude.info.gz ] && \
|
||||||
|
/sbin/install-info %{_infodir}/avrdude.info.gz %{_infodir}/dir || :
|
||||||
|
|
||||||
|
%preun docs
|
||||||
|
if [ $1 = 0 ]; then
|
||||||
|
[ -f %{_infodir}/avrdude.info ] && \
|
||||||
|
/sbin/install-info --delete %{_infodir}/avrdude.info %{_infodir}/dir || :
|
||||||
|
[ -f %{_infodir}/avrdude.info.gz ] && \
|
||||||
|
/sbin/install-info --delete %{_infodir}/avrdude.info.gz %{_infodir}/dir || :
|
||||||
|
fi
|
||||||
|
%endif
|
||||||
|
|
||||||
|
%files
|
||||||
|
%defattr(-,root,root)
|
||||||
|
%{_prefix}/bin/avrdude
|
||||||
|
%{_mandir}/man1/avrdude.1.gz
|
||||||
|
%attr(0644,root,root) %config /etc/avrdude.conf
|
||||||
|
|
||||||
|
%if %{_with_docs}
|
||||||
|
%files docs
|
||||||
|
%doc %{_infodir}/*info*
|
||||||
|
%doc doc/avrdude-html/*.html
|
||||||
|
%doc doc/TODO
|
||||||
|
%doc doc/avrdude.ps
|
||||||
|
%doc doc/avrdude.pdf
|
||||||
|
%endif
|
||||||
|
|
||||||
|
%changelog
|
||||||
|
* Fri Sep 23 2005 Galen Seitz <galens@seitzassoc.com>
|
||||||
|
- Default to enable-doc=yes during configure.
|
||||||
|
- Move info file to docs package.
|
||||||
|
- Make building of docs package conditional. Basic idea copied from avr-gcc.
|
||||||
|
|
||||||
|
* Wed Aug 27 2003 Theodore A. Roth <troth@openavr.org>
|
||||||
|
[Thanks to Artur Lipowski <LAL@pro.onet.pl>]
|
||||||
|
- Do not build debug package.
|
||||||
|
- Remove files not packaged to quell RH9 rpmbuild complaints.
|
||||||
|
|
||||||
|
* Wed Mar 05 2003 Theodore A. Roth <troth@openavr.org>
|
||||||
|
- Add docs sub-package.
|
||||||
|
- Add %post and %preun scriptlets for handling info files.
|
||||||
|
|
||||||
|
* Wed Feb 26 2003 Theodore A. Roth <troth@openavr.org>
|
||||||
|
- Initial build.
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,56 @@
|
||||||
|
/*
|
||||||
|
* avrftdi - extension for avrdude, Wolfgang Moser, Ville Voipio
|
||||||
|
* Copyright (C) 2011 Hannes Weisbach, Doug Springer
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef avrftdi_h
|
||||||
|
#define avrfdti_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define SCK 0x01
|
||||||
|
#define SDO 0x02
|
||||||
|
#define SDI 0x04
|
||||||
|
|
||||||
|
#define RX 0x20
|
||||||
|
#define TX 0x11
|
||||||
|
|
||||||
|
#define TRX (RX | TX)
|
||||||
|
|
||||||
|
#define TYPE_C_D 0x500
|
||||||
|
#define TYPE_H 0x700
|
||||||
|
#define TYPE_4H 0x800
|
||||||
|
|
||||||
|
#define E(x) if ((x)) { fprintf(stdout, "%s:%d %s() %s: %s (%d)\n\t%s\n", __FILE__, __LINE__, __FUNCTION__, \
|
||||||
|
#x, strerror(errno), errno, ftdi_get_error_string(&ftdic)); return -1; }
|
||||||
|
|
||||||
|
#define E_VOID(x) if ((x)) { fprintf(stdout, "%s:%d %s() %s: %s (%d)\n\t%s\n", __FILE__, __LINE__, __FUNCTION__, \
|
||||||
|
#x, strerror(errno), errno, ftdi_get_error_string(&ftdic)); }
|
||||||
|
|
||||||
|
void avrftdi_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,547 @@
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avrpart.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
|
||||||
|
/***
|
||||||
|
*** Elementary functions dealing with OPCODE structures
|
||||||
|
***/
|
||||||
|
|
||||||
|
OPCODE * avr_new_opcode(void)
|
||||||
|
{
|
||||||
|
OPCODE * m;
|
||||||
|
|
||||||
|
m = (OPCODE *)malloc(sizeof(*m));
|
||||||
|
if (m == NULL) {
|
||||||
|
fprintf(stderr, "avr_new_opcode(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(m, 0, sizeof(*m));
|
||||||
|
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avr_set_bits()
|
||||||
|
*
|
||||||
|
* Set instruction bits in the specified command based on the opcode.
|
||||||
|
*/
|
||||||
|
int avr_set_bits(OPCODE * op, unsigned char * cmd)
|
||||||
|
{
|
||||||
|
int i, j, bit;
|
||||||
|
unsigned char mask;
|
||||||
|
|
||||||
|
for (i=0; i<32; i++) {
|
||||||
|
if (op->bit[i].type == AVR_CMDBIT_VALUE) {
|
||||||
|
j = 3 - i / 8;
|
||||||
|
bit = i % 8;
|
||||||
|
mask = 1 << bit;
|
||||||
|
if (op->bit[i].value)
|
||||||
|
cmd[j] = cmd[j] | mask;
|
||||||
|
else
|
||||||
|
cmd[j] = cmd[j] & ~mask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avr_set_addr()
|
||||||
|
*
|
||||||
|
* Set address bits in the specified command based on the opcode, and
|
||||||
|
* the address.
|
||||||
|
*/
|
||||||
|
int avr_set_addr(OPCODE * op, unsigned char * cmd, unsigned long addr)
|
||||||
|
{
|
||||||
|
int i, j, bit;
|
||||||
|
unsigned long value;
|
||||||
|
unsigned char mask;
|
||||||
|
|
||||||
|
for (i=0; i<32; i++) {
|
||||||
|
if (op->bit[i].type == AVR_CMDBIT_ADDRESS) {
|
||||||
|
j = 3 - i / 8;
|
||||||
|
bit = i % 8;
|
||||||
|
mask = 1 << bit;
|
||||||
|
value = addr >> op->bit[i].bitno & 0x01;
|
||||||
|
if (value)
|
||||||
|
cmd[j] = cmd[j] | mask;
|
||||||
|
else
|
||||||
|
cmd[j] = cmd[j] & ~mask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avr_set_input()
|
||||||
|
*
|
||||||
|
* Set input data bits in the specified command based on the opcode,
|
||||||
|
* and the data byte.
|
||||||
|
*/
|
||||||
|
int avr_set_input(OPCODE * op, unsigned char * cmd, unsigned char data)
|
||||||
|
{
|
||||||
|
int i, j, bit;
|
||||||
|
unsigned char value;
|
||||||
|
unsigned char mask;
|
||||||
|
|
||||||
|
for (i=0; i<32; i++) {
|
||||||
|
if (op->bit[i].type == AVR_CMDBIT_INPUT) {
|
||||||
|
j = 3 - i / 8;
|
||||||
|
bit = i % 8;
|
||||||
|
mask = 1 << bit;
|
||||||
|
value = data >> op->bit[i].bitno & 0x01;
|
||||||
|
if (value)
|
||||||
|
cmd[j] = cmd[j] | mask;
|
||||||
|
else
|
||||||
|
cmd[j] = cmd[j] & ~mask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avr_get_output()
|
||||||
|
*
|
||||||
|
* Retreive output data bits from the command results based on the
|
||||||
|
* opcode data.
|
||||||
|
*/
|
||||||
|
int avr_get_output(OPCODE * op, unsigned char * res, unsigned char * data)
|
||||||
|
{
|
||||||
|
int i, j, bit;
|
||||||
|
unsigned char value;
|
||||||
|
unsigned char mask;
|
||||||
|
|
||||||
|
for (i=0; i<32; i++) {
|
||||||
|
if (op->bit[i].type == AVR_CMDBIT_OUTPUT) {
|
||||||
|
j = 3 - i / 8;
|
||||||
|
bit = i % 8;
|
||||||
|
mask = 1 << bit;
|
||||||
|
value = ((res[j] & mask) >> bit) & 0x01;
|
||||||
|
value = value << op->bit[i].bitno;
|
||||||
|
if (value)
|
||||||
|
*data = *data | value;
|
||||||
|
else
|
||||||
|
*data = *data & ~value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static char * avr_op_str(int op)
|
||||||
|
{
|
||||||
|
switch (op) {
|
||||||
|
case AVR_OP_READ : return "READ"; break;
|
||||||
|
case AVR_OP_WRITE : return "WRITE"; break;
|
||||||
|
case AVR_OP_READ_LO : return "READ_LO"; break;
|
||||||
|
case AVR_OP_READ_HI : return "READ_HI"; break;
|
||||||
|
case AVR_OP_WRITE_LO : return "WRITE_LO"; break;
|
||||||
|
case AVR_OP_WRITE_HI : return "WRITE_HI"; break;
|
||||||
|
case AVR_OP_LOADPAGE_LO : return "LOADPAGE_LO"; break;
|
||||||
|
case AVR_OP_LOADPAGE_HI : return "LOADPAGE_HI"; break;
|
||||||
|
case AVR_OP_LOAD_EXT_ADDR : return "LOAD_EXT_ADDR"; break;
|
||||||
|
case AVR_OP_WRITEPAGE : return "WRITEPAGE"; break;
|
||||||
|
case AVR_OP_CHIP_ERASE : return "CHIP_ERASE"; break;
|
||||||
|
case AVR_OP_PGM_ENABLE : return "PGM_ENABLE"; break;
|
||||||
|
default : return "<unknown opcode>"; break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static char * bittype(int type)
|
||||||
|
{
|
||||||
|
switch (type) {
|
||||||
|
case AVR_CMDBIT_IGNORE : return "IGNORE"; break;
|
||||||
|
case AVR_CMDBIT_VALUE : return "VALUE"; break;
|
||||||
|
case AVR_CMDBIT_ADDRESS : return "ADDRESS"; break;
|
||||||
|
case AVR_CMDBIT_INPUT : return "INPUT"; break;
|
||||||
|
case AVR_CMDBIT_OUTPUT : return "OUTPUT"; break;
|
||||||
|
default : return "<unknown bit type>"; break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/***
|
||||||
|
*** Elementary functions dealing with AVRMEM structures
|
||||||
|
***/
|
||||||
|
|
||||||
|
AVRMEM * avr_new_memtype(void)
|
||||||
|
{
|
||||||
|
AVRMEM * m;
|
||||||
|
|
||||||
|
m = (AVRMEM *)malloc(sizeof(*m));
|
||||||
|
if (m == NULL) {
|
||||||
|
fprintf(stderr, "avr_new_memtype(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(m, 0, sizeof(*m));
|
||||||
|
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Allocate and initialize memory buffers for each of the device's
|
||||||
|
* defined memory regions.
|
||||||
|
*/
|
||||||
|
int avr_initmem(AVRPART * p)
|
||||||
|
{
|
||||||
|
LNODEID ln;
|
||||||
|
AVRMEM * m;
|
||||||
|
|
||||||
|
for (ln=lfirst(p->mem); ln; ln=lnext(ln)) {
|
||||||
|
m = ldata(ln);
|
||||||
|
m->buf = (unsigned char *) malloc(m->size);
|
||||||
|
if (m->buf == NULL) {
|
||||||
|
fprintf(stderr, "%s: can't alloc buffer for %s size of %d bytes\n",
|
||||||
|
progname, m->desc, m->size);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
AVRMEM * avr_dup_mem(AVRMEM * m)
|
||||||
|
{
|
||||||
|
AVRMEM * n;
|
||||||
|
|
||||||
|
n = avr_new_memtype();
|
||||||
|
|
||||||
|
*n = *m;
|
||||||
|
|
||||||
|
n->buf = (unsigned char *)malloc(n->size);
|
||||||
|
if (n->buf == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"avr_dup_mem(): out of memory (memsize=%d)\n",
|
||||||
|
n->size);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
memset(n->buf, 0, n->size);
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
AVRMEM * avr_locate_mem(AVRPART * p, char * desc)
|
||||||
|
{
|
||||||
|
AVRMEM * m, * match;
|
||||||
|
LNODEID ln;
|
||||||
|
int matches;
|
||||||
|
int l;
|
||||||
|
|
||||||
|
l = strlen(desc);
|
||||||
|
matches = 0;
|
||||||
|
match = NULL;
|
||||||
|
for (ln=lfirst(p->mem); ln; ln=lnext(ln)) {
|
||||||
|
m = ldata(ln);
|
||||||
|
if (strncmp(desc, m->desc, l) == 0) {
|
||||||
|
match = m;
|
||||||
|
matches++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (matches == 1)
|
||||||
|
return match;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void avr_mem_display(const char * prefix, FILE * f, AVRMEM * m, int type,
|
||||||
|
int verbose)
|
||||||
|
{
|
||||||
|
int i, j;
|
||||||
|
char * optr;
|
||||||
|
|
||||||
|
if (m == NULL) {
|
||||||
|
fprintf(f,
|
||||||
|
"%s Block Poll Page Polled\n"
|
||||||
|
"%sMemory Type Mode Delay Size Indx Paged Size Size #Pages MinW MaxW ReadBack\n"
|
||||||
|
"%s----------- ---- ----- ----- ---- ------ ------ ---- ------ ----- ----- ---------\n",
|
||||||
|
prefix, prefix, prefix);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (verbose > 2) {
|
||||||
|
fprintf(f,
|
||||||
|
"%s Block Poll Page Polled\n"
|
||||||
|
"%sMemory Type Mode Delay Size Indx Paged Size Size #Pages MinW MaxW ReadBack\n"
|
||||||
|
"%s----------- ---- ----- ----- ---- ------ ------ ---- ------ ----- ----- ---------\n",
|
||||||
|
prefix, prefix, prefix);
|
||||||
|
}
|
||||||
|
fprintf(f,
|
||||||
|
"%s%-11s %4d %5d %5d %4d %-6s %6d %4d %6d %5d %5d 0x%02x 0x%02x\n",
|
||||||
|
prefix, m->desc, m->mode, m->delay, m->blocksize, m->pollindex,
|
||||||
|
m->paged ? "yes" : "no",
|
||||||
|
m->size,
|
||||||
|
m->page_size,
|
||||||
|
m->num_pages,
|
||||||
|
m->min_write_delay,
|
||||||
|
m->max_write_delay,
|
||||||
|
m->readback[0],
|
||||||
|
m->readback[1]);
|
||||||
|
if (verbose > 4) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s Memory Ops:\n"
|
||||||
|
"%s Oeration Inst Bit Bit Type Bitno Value\n"
|
||||||
|
"%s ----------- -------- -------- ----- -----\n",
|
||||||
|
prefix, prefix, prefix);
|
||||||
|
for (i=0; i<AVR_OP_MAX; i++) {
|
||||||
|
if (m->op[i]) {
|
||||||
|
for (j=31; j>=0; j--) {
|
||||||
|
if (j==31)
|
||||||
|
optr = avr_op_str(i);
|
||||||
|
else
|
||||||
|
optr = " ";
|
||||||
|
fprintf(f,
|
||||||
|
"%s %-11s %8d %8s %5d %5d\n",
|
||||||
|
prefix, optr, j,
|
||||||
|
bittype(m->op[i]->bit[j].type),
|
||||||
|
m->op[i]->bit[j].bitno,
|
||||||
|
m->op[i]->bit[j].value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Elementary functions dealing with AVRPART structures
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
AVRPART * avr_new_part(void)
|
||||||
|
{
|
||||||
|
AVRPART * p;
|
||||||
|
|
||||||
|
p = (AVRPART *)malloc(sizeof(AVRPART));
|
||||||
|
if (p == NULL) {
|
||||||
|
fprintf(stderr, "new_part(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(p, 0, sizeof(*p));
|
||||||
|
|
||||||
|
p->id[0] = 0;
|
||||||
|
p->desc[0] = 0;
|
||||||
|
p->reset_disposition = RESET_DEDICATED;
|
||||||
|
p->retry_pulse = PIN_AVR_SCK;
|
||||||
|
p->flags = AVRPART_SERIALOK | AVRPART_PARALLELOK | AVRPART_ENABLEPAGEPROGRAMMING;
|
||||||
|
p->config_file[0] = 0;
|
||||||
|
p->lineno = 0;
|
||||||
|
memset(p->signature, 0xFF, 3);
|
||||||
|
p->ctl_stack_type = CTL_STACK_NONE;
|
||||||
|
|
||||||
|
p->mem = lcreat(NULL, 0);
|
||||||
|
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
AVRPART * avr_dup_part(AVRPART * d)
|
||||||
|
{
|
||||||
|
AVRPART * p;
|
||||||
|
LISTID save;
|
||||||
|
LNODEID ln;
|
||||||
|
|
||||||
|
p = avr_new_part();
|
||||||
|
save = p->mem;
|
||||||
|
|
||||||
|
*p = *d;
|
||||||
|
|
||||||
|
p->mem = save;
|
||||||
|
|
||||||
|
for (ln=lfirst(d->mem); ln; ln=lnext(ln)) {
|
||||||
|
ladd(p->mem, avr_dup_mem(ldata(ln)));
|
||||||
|
}
|
||||||
|
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
AVRPART * locate_part(LISTID parts, char * partdesc)
|
||||||
|
{
|
||||||
|
LNODEID ln1;
|
||||||
|
AVRPART * p = NULL;
|
||||||
|
int found;
|
||||||
|
|
||||||
|
found = 0;
|
||||||
|
|
||||||
|
for (ln1=lfirst(parts); ln1 && !found; ln1=lnext(ln1)) {
|
||||||
|
p = ldata(ln1);
|
||||||
|
if ((strcasecmp(partdesc, p->id) == 0) ||
|
||||||
|
(strcasecmp(partdesc, p->desc) == 0))
|
||||||
|
found = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found)
|
||||||
|
return p;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
AVRPART * locate_part_by_avr910_devcode(LISTID parts, int devcode)
|
||||||
|
{
|
||||||
|
LNODEID ln1;
|
||||||
|
AVRPART * p = NULL;
|
||||||
|
|
||||||
|
for (ln1=lfirst(parts); ln1; ln1=lnext(ln1)) {
|
||||||
|
p = ldata(ln1);
|
||||||
|
if (p->avr910_devcode == devcode)
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Iterate over the list of avrparts given as "avrparts", and
|
||||||
|
* call the callback function cb for each entry found. cb is being
|
||||||
|
* passed the following arguments:
|
||||||
|
* . the name of the avrpart (for -p)
|
||||||
|
* . the descriptive text given in the config file
|
||||||
|
* . the name of the config file this avrpart has been defined in
|
||||||
|
* . the line number of the config file this avrpart has been defined at
|
||||||
|
* . the "cookie" passed into walk_avrparts() (opaque client data)
|
||||||
|
*/
|
||||||
|
void walk_avrparts(LISTID avrparts, walk_avrparts_cb cb, void *cookie)
|
||||||
|
{
|
||||||
|
LNODEID ln1;
|
||||||
|
AVRPART * p;
|
||||||
|
|
||||||
|
for (ln1 = lfirst(avrparts); ln1; ln1 = lnext(ln1)) {
|
||||||
|
p = ldata(ln1);
|
||||||
|
cb(p->id, p->desc, p->config_file, p->lineno, cookie);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static char * reset_disp_str(int r)
|
||||||
|
{
|
||||||
|
switch (r) {
|
||||||
|
case RESET_DEDICATED : return "dedicated";
|
||||||
|
case RESET_IO : return "possible i/o";
|
||||||
|
default : return "<invalid>";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static char * pin_name(int pinno)
|
||||||
|
{
|
||||||
|
switch (pinno) {
|
||||||
|
case PIN_AVR_RESET : return "RESET";
|
||||||
|
case PIN_AVR_MISO : return "MISO";
|
||||||
|
case PIN_AVR_MOSI : return "MOSI";
|
||||||
|
case PIN_AVR_SCK : return "SCK";
|
||||||
|
default : return "<unknown>";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void avr_display(FILE * f, AVRPART * p, const char * prefix, int verbose)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char * buf;
|
||||||
|
const char * px;
|
||||||
|
LNODEID ln;
|
||||||
|
AVRMEM * m;
|
||||||
|
|
||||||
|
fprintf(f,
|
||||||
|
"%sAVR Part : %s\n"
|
||||||
|
"%sChip Erase delay : %d us\n"
|
||||||
|
"%sPAGEL : P%02X\n"
|
||||||
|
"%sBS2 : P%02X\n"
|
||||||
|
"%sRESET disposition : %s\n"
|
||||||
|
"%sRETRY pulse : %s\n"
|
||||||
|
"%sserial program mode : %s\n"
|
||||||
|
"%sparallel program mode : %s\n"
|
||||||
|
"%sTimeout : %d\n"
|
||||||
|
"%sStabDelay : %d\n"
|
||||||
|
"%sCmdexeDelay : %d\n"
|
||||||
|
"%sSyncLoops : %d\n"
|
||||||
|
"%sByteDelay : %d\n"
|
||||||
|
"%sPollIndex : %d\n"
|
||||||
|
"%sPollValue : 0x%02x\n"
|
||||||
|
"%sMemory Detail :\n\n",
|
||||||
|
prefix, p->desc,
|
||||||
|
prefix, p->chip_erase_delay,
|
||||||
|
prefix, p->pagel,
|
||||||
|
prefix, p->bs2,
|
||||||
|
prefix, reset_disp_str(p->reset_disposition),
|
||||||
|
prefix, pin_name(p->retry_pulse),
|
||||||
|
prefix, (p->flags & AVRPART_SERIALOK) ? "yes" : "no",
|
||||||
|
prefix, (p->flags & AVRPART_PARALLELOK) ?
|
||||||
|
((p->flags & AVRPART_PSEUDOPARALLEL) ? "psuedo" : "yes") : "no",
|
||||||
|
prefix, p->timeout,
|
||||||
|
prefix, p->stabdelay,
|
||||||
|
prefix, p->cmdexedelay,
|
||||||
|
prefix, p->synchloops,
|
||||||
|
prefix, p->bytedelay,
|
||||||
|
prefix, p->pollindex,
|
||||||
|
prefix, p->pollvalue,
|
||||||
|
prefix);
|
||||||
|
|
||||||
|
px = prefix;
|
||||||
|
i = strlen(prefix) + 5;
|
||||||
|
buf = (char *)malloc(i);
|
||||||
|
if (buf == NULL) {
|
||||||
|
/* ugh, this is not important enough to bail, just ignore it */
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
strcpy(buf, prefix);
|
||||||
|
strcat(buf, " ");
|
||||||
|
px = buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (verbose <= 2) {
|
||||||
|
avr_mem_display(px, f, NULL, 0, verbose);
|
||||||
|
}
|
||||||
|
for (ln=lfirst(p->mem); ln; ln=lnext(ln)) {
|
||||||
|
m = ldata(ln);
|
||||||
|
avr_mem_display(px, f, m, i, verbose);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buf)
|
||||||
|
free(buf);
|
||||||
|
}
|
|
@ -0,0 +1,225 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef avrpart_h
|
||||||
|
#define avrpart_h
|
||||||
|
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
|
#include "lists.h"
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* AVR serial programming instructions
|
||||||
|
*/
|
||||||
|
enum {
|
||||||
|
AVR_OP_READ,
|
||||||
|
AVR_OP_WRITE,
|
||||||
|
AVR_OP_READ_LO,
|
||||||
|
AVR_OP_READ_HI,
|
||||||
|
AVR_OP_WRITE_LO,
|
||||||
|
AVR_OP_WRITE_HI,
|
||||||
|
AVR_OP_LOADPAGE_LO,
|
||||||
|
AVR_OP_LOADPAGE_HI,
|
||||||
|
AVR_OP_LOAD_EXT_ADDR,
|
||||||
|
AVR_OP_WRITEPAGE,
|
||||||
|
AVR_OP_CHIP_ERASE,
|
||||||
|
AVR_OP_PGM_ENABLE,
|
||||||
|
AVR_OP_MAX
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
enum {
|
||||||
|
AVR_CMDBIT_IGNORE, /* bit is ignored on input and output */
|
||||||
|
AVR_CMDBIT_VALUE, /* bit is set to 0 or 1 for input or output */
|
||||||
|
AVR_CMDBIT_ADDRESS, /* this bit represents an input address bit */
|
||||||
|
AVR_CMDBIT_INPUT, /* this bit is an input bit */
|
||||||
|
AVR_CMDBIT_OUTPUT /* this bit is an output bit */
|
||||||
|
};
|
||||||
|
|
||||||
|
enum { /* these are assigned to reset_disposition of AVRPART */
|
||||||
|
RESET_DEDICATED, /* reset pin is dedicated */
|
||||||
|
RESET_IO /* reset pin might be configured as an I/O pin */
|
||||||
|
};
|
||||||
|
|
||||||
|
enum ctl_stack_t {
|
||||||
|
CTL_STACK_NONE, /* no control stack defined */
|
||||||
|
CTL_STACK_PP, /* parallel programming control stack */
|
||||||
|
CTL_STACK_HVSP /* high voltage serial programming control stack */
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* serial programming instruction bit specifications
|
||||||
|
*/
|
||||||
|
typedef struct cmdbit {
|
||||||
|
int type; /* AVR_CMDBIT_* */
|
||||||
|
int bitno; /* which input bit to use for this command bit */
|
||||||
|
int value; /* bit value if type == AVR_CMDBIT_VALUD */
|
||||||
|
} CMDBIT;
|
||||||
|
|
||||||
|
typedef struct opcode {
|
||||||
|
CMDBIT bit[32]; /* opcode bit specs */
|
||||||
|
} OPCODE;
|
||||||
|
|
||||||
|
|
||||||
|
#define AVRPART_SERIALOK 0x0001 /* part supports serial programming */
|
||||||
|
#define AVRPART_PARALLELOK 0x0002 /* part supports parallel programming */
|
||||||
|
#define AVRPART_PSEUDOPARALLEL 0x0004 /* part has pseudo parallel support */
|
||||||
|
#define AVRPART_HAS_JTAG 0x0008 /* part has a JTAG i/f */
|
||||||
|
#define AVRPART_ALLOWFULLPAGEBITSTREAM 0x0010 /* JTAG ICE mkII param. */
|
||||||
|
#define AVRPART_ENABLEPAGEPROGRAMMING 0x0020 /* JTAG ICE mkII param. */
|
||||||
|
#define AVRPART_HAS_DW 0x0040 /* part has a debugWire i/f */
|
||||||
|
#define AVRPART_HAS_PDI 0x0080 /* part has PDI i/f rather than ISP (ATxmega) */
|
||||||
|
#define AVRPART_AVR32 0x0100 /* part is in AVR32 family */
|
||||||
|
#define AVRPART_INIT_SMC 0x0200 /* part will undergo chip erase */
|
||||||
|
#define AVRPART_WRITE 0x0400 /* at least one write operation specified */
|
||||||
|
#define AVRPART_HAS_TPI 0x0800 /* part has TPI i/f rather than ISP (ATtiny4/5/9/10) */
|
||||||
|
|
||||||
|
#define AVR_DESCLEN 64
|
||||||
|
#define AVR_IDLEN 32
|
||||||
|
#define CTL_STACK_SIZE 32
|
||||||
|
#define FLASH_INSTR_SIZE 3
|
||||||
|
#define EEPROM_INSTR_SIZE 20
|
||||||
|
typedef struct avrpart {
|
||||||
|
char desc[AVR_DESCLEN]; /* long part name */
|
||||||
|
char id[AVR_IDLEN]; /* short part name */
|
||||||
|
int stk500_devcode; /* stk500 device code */
|
||||||
|
int avr910_devcode; /* avr910 device code */
|
||||||
|
int chip_erase_delay; /* microseconds */
|
||||||
|
unsigned char pagel; /* for parallel programming */
|
||||||
|
unsigned char bs2; /* for parallel programming */
|
||||||
|
unsigned char signature[3]; /* expected value of signature bytes */
|
||||||
|
int reset_disposition; /* see RESET_ enums */
|
||||||
|
int retry_pulse; /* retry program enable by pulsing
|
||||||
|
this pin (PIN_AVR_*) */
|
||||||
|
unsigned flags; /* see AVRPART_ masks */
|
||||||
|
|
||||||
|
int timeout; /* stk500 v2 xml file parameter */
|
||||||
|
int stabdelay; /* stk500 v2 xml file parameter */
|
||||||
|
int cmdexedelay; /* stk500 v2 xml file parameter */
|
||||||
|
int synchloops; /* stk500 v2 xml file parameter */
|
||||||
|
int bytedelay; /* stk500 v2 xml file parameter */
|
||||||
|
int pollindex; /* stk500 v2 xml file parameter */
|
||||||
|
unsigned char pollvalue; /* stk500 v2 xml file parameter */
|
||||||
|
int predelay; /* stk500 v2 xml file parameter */
|
||||||
|
int postdelay; /* stk500 v2 xml file parameter */
|
||||||
|
int pollmethod; /* stk500 v2 xml file parameter */
|
||||||
|
|
||||||
|
enum ctl_stack_t ctl_stack_type; /* what to use the ctl stack for */
|
||||||
|
unsigned char controlstack[CTL_STACK_SIZE]; /* stk500v2 PP/HVSP ctl stack */
|
||||||
|
unsigned char flash_instr[FLASH_INSTR_SIZE]; /* flash instructions (debugWire, JTAG) */
|
||||||
|
unsigned char eeprom_instr[EEPROM_INSTR_SIZE]; /* EEPROM instructions (debugWire, JTAG) */
|
||||||
|
|
||||||
|
int hventerstabdelay; /* stk500 v2 hv mode parameter */
|
||||||
|
int progmodedelay; /* stk500 v2 hv mode parameter */
|
||||||
|
int latchcycles; /* stk500 v2 hv mode parameter */
|
||||||
|
int togglevtg; /* stk500 v2 hv mode parameter */
|
||||||
|
int poweroffdelay; /* stk500 v2 hv mode parameter */
|
||||||
|
int resetdelayms; /* stk500 v2 hv mode parameter */
|
||||||
|
int resetdelayus; /* stk500 v2 hv mode parameter */
|
||||||
|
int hvleavestabdelay; /* stk500 v2 hv mode parameter */
|
||||||
|
int resetdelay; /* stk500 v2 hv mode parameter */
|
||||||
|
int chiperasepulsewidth; /* stk500 v2 hv mode parameter */
|
||||||
|
int chiperasepolltimeout; /* stk500 v2 hv mode parameter */
|
||||||
|
int chiperasetime; /* stk500 v2 hv mode parameter */
|
||||||
|
int programfusepulsewidth; /* stk500 v2 hv mode parameter */
|
||||||
|
int programfusepolltimeout; /* stk500 v2 hv mode parameter */
|
||||||
|
int programlockpulsewidth; /* stk500 v2 hv mode parameter */
|
||||||
|
int programlockpolltimeout; /* stk500 v2 hv mode parameter */
|
||||||
|
int synchcycles; /* stk500 v2 hv mode parameter */
|
||||||
|
int hvspcmdexedelay; /* stk500 v2 xml file parameter */
|
||||||
|
|
||||||
|
unsigned char idr; /* JTAG ICE mkII XML file parameter */
|
||||||
|
unsigned char rampz; /* JTAG ICE mkII XML file parameter */
|
||||||
|
unsigned char spmcr; /* JTAG ICE mkII XML file parameter */
|
||||||
|
unsigned short eecr; /* JTAC ICE mkII XML file parameter */
|
||||||
|
unsigned int nvm_base; /* Base address of NVM controller in ATxmega devices */
|
||||||
|
|
||||||
|
OPCODE * op[AVR_OP_MAX]; /* opcodes */
|
||||||
|
|
||||||
|
LISTID mem; /* avr memory definitions */
|
||||||
|
char config_file[PATH_MAX]; /* config file where defined */
|
||||||
|
int lineno; /* config file line number */
|
||||||
|
} AVRPART;
|
||||||
|
|
||||||
|
#define AVR_MEMDESCLEN 64
|
||||||
|
typedef struct avrmem {
|
||||||
|
char desc[AVR_MEMDESCLEN]; /* memory description ("flash", "eeprom", etc) */
|
||||||
|
int paged; /* page addressed (e.g. ATmega flash) */
|
||||||
|
int size; /* total memory size in bytes */
|
||||||
|
int page_size; /* size of memory page (if page addressed) */
|
||||||
|
int num_pages; /* number of pages (if page addressed) */
|
||||||
|
unsigned int offset; /* offset in IO memory (ATxmega) */
|
||||||
|
int min_write_delay; /* microseconds */
|
||||||
|
int max_write_delay; /* microseconds */
|
||||||
|
int pwroff_after_write; /* after this memory type is written to,
|
||||||
|
the device must be powered off and
|
||||||
|
back on, see errata
|
||||||
|
http://www.atmel.com/atmel/acrobat/doc1280.pdf */
|
||||||
|
unsigned char readback[2]; /* polled read-back values */
|
||||||
|
|
||||||
|
int mode; /* stk500 v2 xml file parameter */
|
||||||
|
int delay; /* stk500 v2 xml file parameter */
|
||||||
|
int blocksize; /* stk500 v2 xml file parameter */
|
||||||
|
int readsize; /* stk500 v2 xml file parameter */
|
||||||
|
int pollindex; /* stk500 v2 xml file parameter */
|
||||||
|
|
||||||
|
unsigned char * buf; /* pointer to memory buffer */
|
||||||
|
OPCODE * op[AVR_OP_MAX]; /* opcodes */
|
||||||
|
} AVRMEM;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Functions for OPCODE structures */
|
||||||
|
OPCODE * avr_new_opcode(void);
|
||||||
|
int avr_set_bits(OPCODE * op, unsigned char * cmd);
|
||||||
|
int avr_set_addr(OPCODE * op, unsigned char * cmd, unsigned long addr);
|
||||||
|
int avr_set_input(OPCODE * op, unsigned char * cmd, unsigned char data);
|
||||||
|
int avr_get_output(OPCODE * op, unsigned char * res, unsigned char * data);
|
||||||
|
|
||||||
|
/* Functions for AVRMEM structures */
|
||||||
|
AVRMEM * avr_new_memtype(void);
|
||||||
|
int avr_initmem(AVRPART * p);
|
||||||
|
AVRMEM * avr_dup_mem(AVRMEM * m);
|
||||||
|
AVRMEM * avr_locate_mem(AVRPART * p, char * desc);
|
||||||
|
void avr_mem_display(const char * prefix, FILE * f, AVRMEM * m, int type,
|
||||||
|
int verbose);
|
||||||
|
|
||||||
|
/* Functions for AVRPART structures */
|
||||||
|
AVRPART * avr_new_part(void);
|
||||||
|
AVRPART * avr_dup_part(AVRPART * d);
|
||||||
|
AVRPART * locate_part(LISTID parts, char * partdesc);
|
||||||
|
AVRPART * locate_part_by_avr910_devcode(LISTID parts, int devcode);
|
||||||
|
void avr_display(FILE * f, AVRPART * p, const char * prefix, int verbose);
|
||||||
|
|
||||||
|
typedef void (*walk_avrparts_cb)(const char *name, const char *desc,
|
||||||
|
const char *cfgname, int cfglineno,
|
||||||
|
void *cookie);
|
||||||
|
void walk_avrparts(LISTID programmers, walk_avrparts_cb cb, void *cookie);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* avrpart_h */
|
|
@ -0,0 +1,660 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000, 2001, 2002, 2003 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2005 Michael Holzt <kju-avr@fqdn.org>
|
||||||
|
* Copyright (C) 2011 Darell Tan <darell.tan@gmail.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
#if !defined(WIN32NATIVE)
|
||||||
|
# include <signal.h>
|
||||||
|
# include <sys/time.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "par.h"
|
||||||
|
#include "serbb.h"
|
||||||
|
#include "tpi.h"
|
||||||
|
|
||||||
|
static int delay_decrement;
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
static int has_perfcount;
|
||||||
|
static LARGE_INTEGER freq;
|
||||||
|
#else
|
||||||
|
static volatile int done;
|
||||||
|
|
||||||
|
typedef void (*mysighandler_t)(int);
|
||||||
|
static mysighandler_t saved_alarmhandler;
|
||||||
|
|
||||||
|
static void alarmhandler(int signo)
|
||||||
|
{
|
||||||
|
done = 1;
|
||||||
|
signal(SIGALRM, saved_alarmhandler);
|
||||||
|
}
|
||||||
|
#endif /* WIN32NATIVE */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calibrate the microsecond delay loop below.
|
||||||
|
*/
|
||||||
|
static void bitbang_calibrate_delay(void)
|
||||||
|
{
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
/*
|
||||||
|
* If the hardware supports a high-resolution performance counter,
|
||||||
|
* we ultimately prefer that one, as it gives quite accurate delays
|
||||||
|
* on modern high-speed CPUs.
|
||||||
|
*/
|
||||||
|
if (QueryPerformanceFrequency(&freq))
|
||||||
|
{
|
||||||
|
has_perfcount = 1;
|
||||||
|
if (verbose >= 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: Using performance counter for bitbang delays\n",
|
||||||
|
progname);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* If a high-resolution performance counter is not available, we
|
||||||
|
* don't have any Win32 implementation for setting up the
|
||||||
|
* per-microsecond delay count, so we can only run on a
|
||||||
|
* preconfigured delay stepping there. The figure below should at
|
||||||
|
* least be correct within an order of magnitude, judging from the
|
||||||
|
* auto-calibration figures seen on various Unix systems on
|
||||||
|
* comparable hardware.
|
||||||
|
*/
|
||||||
|
if (verbose >= 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: Using guessed per-microsecond delay count for bitbang delays\n",
|
||||||
|
progname);
|
||||||
|
delay_decrement = 100;
|
||||||
|
}
|
||||||
|
#else /* !WIN32NATIVE */
|
||||||
|
struct itimerval itv;
|
||||||
|
volatile int i;
|
||||||
|
|
||||||
|
if (verbose >= 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: Calibrating delay loop...",
|
||||||
|
progname);
|
||||||
|
i = 0;
|
||||||
|
done = 0;
|
||||||
|
saved_alarmhandler = signal(SIGALRM, alarmhandler);
|
||||||
|
/*
|
||||||
|
* Set ITIMER_REAL to 100 ms. All known systems have a timer
|
||||||
|
* granularity of 10 ms or better, so counting the delay cycles
|
||||||
|
* accumulating over 100 ms should give us a rather realistic
|
||||||
|
* picture, without annoying the user by a lengthy startup time (as
|
||||||
|
* an alarm(1) would do). Of course, if heavy system activity
|
||||||
|
* happens just during calibration but stops before the remaining
|
||||||
|
* part of AVRDUDE runs, this will yield wrong values. There's not
|
||||||
|
* much we can do about this.
|
||||||
|
*/
|
||||||
|
itv.it_value.tv_sec = 0;
|
||||||
|
itv.it_value.tv_usec = 100000;
|
||||||
|
itv.it_interval.tv_sec = itv.it_interval.tv_usec = 0;
|
||||||
|
setitimer(ITIMER_REAL, &itv, 0);
|
||||||
|
while (!done)
|
||||||
|
i--;
|
||||||
|
itv.it_value.tv_sec = itv.it_value.tv_usec = 0;
|
||||||
|
setitimer(ITIMER_REAL, &itv, 0);
|
||||||
|
/*
|
||||||
|
* Calculate back from 100 ms to 1 us.
|
||||||
|
*/
|
||||||
|
delay_decrement = -i / 100000;
|
||||||
|
if (verbose >= 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
" calibrated to %d cycles per us\n",
|
||||||
|
delay_decrement);
|
||||||
|
#endif /* WIN32NATIVE */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Delay for approximately the number of microseconds specified.
|
||||||
|
* usleep()'s granularity is usually like 1 ms or 10 ms, so it's not
|
||||||
|
* really suitable for short delays in bit-bang algorithms.
|
||||||
|
*/
|
||||||
|
void bitbang_delay(int us)
|
||||||
|
{
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
LARGE_INTEGER countNow, countEnd;
|
||||||
|
|
||||||
|
if (has_perfcount)
|
||||||
|
{
|
||||||
|
QueryPerformanceCounter(&countNow);
|
||||||
|
countEnd.QuadPart = countNow.QuadPart + freq.QuadPart * us / 1000000ll;
|
||||||
|
|
||||||
|
while (countNow.QuadPart < countEnd.QuadPart)
|
||||||
|
QueryPerformanceCounter(&countNow);
|
||||||
|
}
|
||||||
|
else /* no performance counters -- run normal uncalibrated delay */
|
||||||
|
{
|
||||||
|
#endif /* WIN32NATIVE */
|
||||||
|
volatile int del = us * delay_decrement;
|
||||||
|
|
||||||
|
while (del > 0)
|
||||||
|
del--;
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
}
|
||||||
|
#endif /* WIN32NATIVE */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* transmit and receive a byte of data to/from the AVR device
|
||||||
|
*/
|
||||||
|
static unsigned char bitbang_txrx(PROGRAMMER * pgm, unsigned char byte)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
unsigned char r, b, rbyte;
|
||||||
|
|
||||||
|
rbyte = 0;
|
||||||
|
for (i=7; i>=0; i--) {
|
||||||
|
/*
|
||||||
|
* Write and read one bit on SPI.
|
||||||
|
* Some notes on timing: Let T be the time it takes to do
|
||||||
|
* one pgm->setpin()-call resp. par clrpin()-call, then
|
||||||
|
* - SCK is high for 2T
|
||||||
|
* - SCK is low for 2T
|
||||||
|
* - MOSI setuptime is 1T
|
||||||
|
* - MOSI holdtime is 3T
|
||||||
|
* - SCK low to MISO read is 2T to 3T
|
||||||
|
* So we are within programming specs (expect for AT90S1200),
|
||||||
|
* if and only if T>t_CLCL (t_CLCL=clock period of target system).
|
||||||
|
*
|
||||||
|
* Due to the delay introduced by "IN" and "OUT"-commands,
|
||||||
|
* T is greater than 1us (more like 2us) on x86-architectures.
|
||||||
|
* So programming works safely down to 1MHz target clock.
|
||||||
|
*/
|
||||||
|
|
||||||
|
b = (byte >> i) & 0x01;
|
||||||
|
|
||||||
|
/* set the data input line as desired */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], b);
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_SCK], 1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* read the result bit (it is either valid from a previous falling
|
||||||
|
* edge or it is ignored in the current context)
|
||||||
|
*/
|
||||||
|
r = pgm->getpin(pgm, pgm->pinno[PIN_AVR_MISO]);
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_SCK], 0);
|
||||||
|
|
||||||
|
rbyte |= r << i;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rbyte;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int bitbang_tpi_clk(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
unsigned char r = 0;
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_SCK], 1);
|
||||||
|
|
||||||
|
r = pgm->getpin(pgm, pgm->pinno[PIN_AVR_MISO]);
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_SCK], 0);
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
void bitbang_tpi_tx(PROGRAMMER * pgm, unsigned char byte)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
unsigned char b, parity;
|
||||||
|
|
||||||
|
/* start bit */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], 0);
|
||||||
|
bitbang_tpi_clk(pgm);
|
||||||
|
|
||||||
|
parity = 0;
|
||||||
|
for (i = 0; i <= 7; i++) {
|
||||||
|
b = (byte >> i) & 0x01;
|
||||||
|
parity ^= b;
|
||||||
|
|
||||||
|
/* set the data input line as desired */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], b);
|
||||||
|
bitbang_tpi_clk(pgm);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* parity bit */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], parity);
|
||||||
|
bitbang_tpi_clk(pgm);
|
||||||
|
|
||||||
|
/* 2 stop bits */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], 1);
|
||||||
|
bitbang_tpi_clk(pgm);
|
||||||
|
bitbang_tpi_clk(pgm);
|
||||||
|
}
|
||||||
|
|
||||||
|
int bitbang_tpi_rx(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
unsigned char b, rbyte, parity;
|
||||||
|
|
||||||
|
/* make sure pin is on for "pullup" */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], 1);
|
||||||
|
|
||||||
|
/* wait for start bit (up to 10 bits) */
|
||||||
|
b = 1;
|
||||||
|
for (i = 0; i < 10; i++) {
|
||||||
|
b = bitbang_tpi_clk(pgm);
|
||||||
|
if (b == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (b != 0) {
|
||||||
|
fprintf(stderr, "bitbang_tpi_rx: start bit not received correctly\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
rbyte = 0;
|
||||||
|
parity = 0;
|
||||||
|
for (i=0; i<=7; i++) {
|
||||||
|
b = bitbang_tpi_clk(pgm);
|
||||||
|
parity ^= b;
|
||||||
|
|
||||||
|
rbyte |= b << i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* parity bit */
|
||||||
|
if (bitbang_tpi_clk(pgm) != parity) {
|
||||||
|
fprintf(stderr, "bitbang_tpi_rx: parity bit is wrong\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 2 stop bits */
|
||||||
|
b = 1;
|
||||||
|
b &= bitbang_tpi_clk(pgm);
|
||||||
|
b &= bitbang_tpi_clk(pgm);
|
||||||
|
if (b != 1) {
|
||||||
|
fprintf(stderr, "bitbang_tpi_rx: stop bits not received correctly\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rbyte;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bitbang_rdy_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_LED_RDY], !value);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bitbang_err_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_LED_ERR], !value);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bitbang_pgm_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_LED_PGM], !value);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bitbang_vfy_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_LED_VFY], !value);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* transmit an AVR device command and return the results; 'cmd' and
|
||||||
|
* 'res' must point to at least a 4 byte data buffer
|
||||||
|
*/
|
||||||
|
int bitbang_cmd(PROGRAMMER * pgm, unsigned char cmd[4],
|
||||||
|
unsigned char res[4])
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i=0; i<4; i++) {
|
||||||
|
res[i] = bitbang_txrx(pgm, cmd[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(verbose >= 2)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "bitbang_cmd(): [ ");
|
||||||
|
for(i = 0; i < 4; i++)
|
||||||
|
fprintf(stderr, "%02X ", cmd[i]);
|
||||||
|
fprintf(stderr, "] [ ");
|
||||||
|
for(i = 0; i < 4; i++)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%02X ", res[i]);
|
||||||
|
}
|
||||||
|
fprintf(stderr, "]\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int bitbang_cmd_tpi(PROGRAMMER * pgm, unsigned char cmd[],
|
||||||
|
int cmd_len, unsigned char res[], int res_len)
|
||||||
|
{
|
||||||
|
int i, r;
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, ON);
|
||||||
|
|
||||||
|
for (i=0; i<cmd_len; i++) {
|
||||||
|
bitbang_tpi_tx(pgm, cmd[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
r = 0;
|
||||||
|
for (i=0; i<res_len; i++) {
|
||||||
|
r = bitbang_tpi_rx(pgm);
|
||||||
|
if (r == -1)
|
||||||
|
break;
|
||||||
|
res[i] = r;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(verbose >= 2)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "bitbang_cmd_tpi(): [ ");
|
||||||
|
for(i = 0; i < cmd_len; i++)
|
||||||
|
fprintf(stderr, "%02X ", cmd[i]);
|
||||||
|
fprintf(stderr, "] [ ");
|
||||||
|
for(i = 0; i < res_len; i++)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%02X ", res[i]);
|
||||||
|
}
|
||||||
|
fprintf(stderr, "]\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, OFF);
|
||||||
|
if (r == -1)
|
||||||
|
return -1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* transmit bytes via SPI and return the results; 'cmd' and
|
||||||
|
* 'res' must point to data buffers
|
||||||
|
*/
|
||||||
|
int bitbang_spi(PROGRAMMER * pgm, unsigned char cmd[],
|
||||||
|
unsigned char res[], int count)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_LED_PGM], 0);
|
||||||
|
|
||||||
|
for (i=0; i<count; i++) {
|
||||||
|
res[i] = bitbang_txrx(pgm, cmd[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_LED_PGM], 1);
|
||||||
|
|
||||||
|
if(verbose >= 2)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "bitbang_cmd(): [ ");
|
||||||
|
for(i = 0; i < count; i++)
|
||||||
|
fprintf(stderr, "%02X ", cmd[i]);
|
||||||
|
fprintf(stderr, "] [ ");
|
||||||
|
for(i = 0; i < count; i++)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%02X ", res[i]);
|
||||||
|
}
|
||||||
|
fprintf(stderr, "]\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* issue the 'chip erase' command to the AVR device
|
||||||
|
*/
|
||||||
|
int bitbang_chip_erase(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
unsigned char cmd[4];
|
||||||
|
unsigned char res[4];
|
||||||
|
AVRMEM *mem;
|
||||||
|
|
||||||
|
if (p->flags & AVRPART_HAS_TPI) {
|
||||||
|
pgm->pgm_led(pgm, ON);
|
||||||
|
|
||||||
|
while (avr_tpi_poll_nvmbsy(pgm));
|
||||||
|
|
||||||
|
/* NVMCMD <- CHIP_ERASE */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SOUT | TPI_SIO_ADDR(TPI_IOREG_NVMCMD));
|
||||||
|
bitbang_tpi_tx(pgm, TPI_NVMCMD_CHIP_ERASE); /* CHIP_ERASE */
|
||||||
|
|
||||||
|
/* Set Pointer Register */
|
||||||
|
mem = avr_locate_mem(p, "flash");
|
||||||
|
if (mem == NULL) {
|
||||||
|
fprintf(stderr, "No flash memory to erase for part %s\n",
|
||||||
|
p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SSTPR | 0);
|
||||||
|
bitbang_tpi_tx(pgm, (mem->offset & 0xFF) | 1); /* high byte */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SSTPR | 1);
|
||||||
|
bitbang_tpi_tx(pgm, (mem->offset >> 8) & 0xFF);
|
||||||
|
|
||||||
|
/* write dummy value to start erase */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SST);
|
||||||
|
bitbang_tpi_tx(pgm, 0xFF);
|
||||||
|
|
||||||
|
while (avr_tpi_poll_nvmbsy(pgm));
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, OFF);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->op[AVR_OP_CHIP_ERASE] == NULL) {
|
||||||
|
fprintf(stderr, "chip erase instruction not defined for part \"%s\"\n",
|
||||||
|
p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, ON);
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(cmd));
|
||||||
|
|
||||||
|
avr_set_bits(p->op[AVR_OP_CHIP_ERASE], cmd);
|
||||||
|
pgm->cmd(pgm, cmd, res);
|
||||||
|
usleep(p->chip_erase_delay);
|
||||||
|
pgm->initialize(pgm, p);
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, OFF);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* issue the 'program enable' command to the AVR device
|
||||||
|
*/
|
||||||
|
int bitbang_program_enable(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
unsigned char cmd[4];
|
||||||
|
unsigned char res[4];
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (p->flags & AVRPART_HAS_TPI) {
|
||||||
|
/* enable NVM programming */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SKEY);
|
||||||
|
for (i = sizeof(tpi_skey) - 1; i >= 0; i--)
|
||||||
|
bitbang_tpi_tx(pgm, tpi_skey[i]);
|
||||||
|
|
||||||
|
/* check NVMEN bit */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SLDCS | TPI_REG_TPISR);
|
||||||
|
i = bitbang_tpi_rx(pgm);
|
||||||
|
return (i != -1 && (i & TPI_REG_TPISR_NVMEN)) ? 0 : -2;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->op[AVR_OP_PGM_ENABLE] == NULL) {
|
||||||
|
fprintf(stderr, "program enable instruction not defined for part \"%s\"\n",
|
||||||
|
p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(cmd));
|
||||||
|
avr_set_bits(p->op[AVR_OP_PGM_ENABLE], cmd);
|
||||||
|
pgm->cmd(pgm, cmd, res);
|
||||||
|
|
||||||
|
if (res[2] != cmd[1])
|
||||||
|
return -2;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* initialize the AVR device and prepare it to accept commands
|
||||||
|
*/
|
||||||
|
int bitbang_initialize(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
int tries;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
bitbang_calibrate_delay();
|
||||||
|
|
||||||
|
pgm->powerup(pgm);
|
||||||
|
usleep(20000);
|
||||||
|
|
||||||
|
/* TPIDATA is a single line, so MISO & MOSI should be connected */
|
||||||
|
if (p->flags & AVRPART_HAS_TPI) {
|
||||||
|
/* make sure cmd_tpi() is defined */
|
||||||
|
if (pgm->cmd_tpi == NULL) {
|
||||||
|
fprintf(stderr, "%s: Error: %s programmer does not support TPI\n",
|
||||||
|
progname, pgm->type);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* bring RESET high first */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_RESET], 1);
|
||||||
|
usleep(1000);
|
||||||
|
|
||||||
|
if (verbose >= 2)
|
||||||
|
fprintf(stderr, "doing MOSI-MISO link check\n");
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], 0);
|
||||||
|
if (pgm->getpin(pgm, pgm->pinno[PIN_AVR_MISO]) != 0) {
|
||||||
|
fprintf(stderr, "MOSI->MISO 0 failed\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], 1);
|
||||||
|
if (pgm->getpin(pgm, pgm->pinno[PIN_AVR_MISO]) != 1) {
|
||||||
|
fprintf(stderr, "MOSI->MISO 1 failed\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (verbose >= 2)
|
||||||
|
fprintf(stderr, "MOSI-MISO link present\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_SCK], 0);
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_RESET], 0);
|
||||||
|
usleep(20000);
|
||||||
|
|
||||||
|
if (p->flags & AVRPART_HAS_TPI) {
|
||||||
|
/* keep TPIDATA high for 16 clock cycles */
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_MOSI], 1);
|
||||||
|
for (i = 0; i < 16; i++)
|
||||||
|
pgm->highpulsepin(pgm, pgm->pinno[PIN_AVR_SCK]);
|
||||||
|
|
||||||
|
/* remove extra guard timing bits */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SSTCS | TPI_REG_TPIPCR);
|
||||||
|
bitbang_tpi_tx(pgm, 0x7);
|
||||||
|
|
||||||
|
/* read TPI ident reg */
|
||||||
|
bitbang_tpi_tx(pgm, TPI_CMD_SLDCS | TPI_REG_TPIIR);
|
||||||
|
rc = bitbang_tpi_rx(pgm);
|
||||||
|
if (rc != 0x80) {
|
||||||
|
fprintf(stderr, "TPIIR not correct\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
pgm->highpulsepin(pgm, pgm->pinno[PIN_AVR_RESET]);
|
||||||
|
}
|
||||||
|
|
||||||
|
usleep(20000); /* 20 ms XXX should be a per-chip parameter */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Enable programming mode. If we are programming an AT90S1200, we
|
||||||
|
* can only issue the command and hope it worked. If we are using
|
||||||
|
* one of the other chips, the chip will echo 0x53 when issuing the
|
||||||
|
* third byte of the command. In this case, try up to 32 times in
|
||||||
|
* order to possibly get back into sync with the chip if we are out
|
||||||
|
* of sync.
|
||||||
|
*/
|
||||||
|
if (strcmp(p->desc, "AT90S1200")==0) {
|
||||||
|
pgm->program_enable(pgm, p);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
tries = 0;
|
||||||
|
do {
|
||||||
|
rc = pgm->program_enable(pgm, p);
|
||||||
|
if ((rc == 0)||(rc == -1))
|
||||||
|
break;
|
||||||
|
pgm->highpulsepin(pgm, pgm->pinno[p->retry_pulse/*PIN_AVR_SCK*/]);
|
||||||
|
tries++;
|
||||||
|
} while (tries < 65);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* can't sync with the device, maybe it's not attached?
|
||||||
|
*/
|
||||||
|
if (rc) {
|
||||||
|
fprintf(stderr, "%s: AVR device not responding\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void verify_pin_assigned(PROGRAMMER * pgm, int pin, char * desc)
|
||||||
|
{
|
||||||
|
if (pgm->pinno[pin] == 0) {
|
||||||
|
fprintf(stderr, "%s: error: no pin has been assigned for %s\n",
|
||||||
|
progname, desc);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Verify all prerequisites for a bit-bang programmer are present.
|
||||||
|
*/
|
||||||
|
void bitbang_check_prerequisites(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
|
||||||
|
verify_pin_assigned(pgm, PIN_AVR_RESET, "AVR RESET");
|
||||||
|
verify_pin_assigned(pgm, PIN_AVR_SCK, "AVR SCK");
|
||||||
|
verify_pin_assigned(pgm, PIN_AVR_MISO, "AVR MISO");
|
||||||
|
verify_pin_assigned(pgm, PIN_AVR_MOSI, "AVR MOSI");
|
||||||
|
|
||||||
|
if (pgm->cmd == NULL) {
|
||||||
|
fprintf(stderr, "%s: error: no cmd() method defined for bitbang programmer\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000, 2001, 2002, 2003 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2005 Michael Holzt <kju-avr@fqdn.org>
|
||||||
|
* Copyright (C) 2011 Darell Tan <darell.tan@gmail.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef bitbang_h
|
||||||
|
#define bitbang_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int bitbang_setpin(int fd, int pin, int value);
|
||||||
|
int bitbang_getpin(int fd, int pin);
|
||||||
|
int bitbang_highpulsepin(int fd, int pin);
|
||||||
|
void bitbang_delay(unsigned int us);
|
||||||
|
|
||||||
|
void bitbang_check_prerequisites(PROGRAMMER *pgm);
|
||||||
|
|
||||||
|
int bitbang_rdy_led (PROGRAMMER * pgm, int value);
|
||||||
|
int bitbang_err_led (PROGRAMMER * pgm, int value);
|
||||||
|
int bitbang_pgm_led (PROGRAMMER * pgm, int value);
|
||||||
|
int bitbang_vfy_led (PROGRAMMER * pgm, int value);
|
||||||
|
int bitbang_cmd (PROGRAMMER * pgm, unsigned char cmd[4],
|
||||||
|
unsigned char res[4]);
|
||||||
|
int bitbang_cmd_tpi (PROGRAMMER * pgm, unsigned char cmd[],
|
||||||
|
int cmd_len, unsigned char res[], int res_len);
|
||||||
|
int bitbang_spi (PROGRAMMER * pgm, unsigned char cmd[],
|
||||||
|
unsigned char res[], int count);
|
||||||
|
int bitbang_chip_erase (PROGRAMMER * pgm, AVRPART * p);
|
||||||
|
int bitbang_program_enable (PROGRAMMER * pgm, AVRPART * p);
|
||||||
|
void bitbang_powerup (PROGRAMMER * pgm);
|
||||||
|
void bitbang_powerdown (PROGRAMMER * pgm);
|
||||||
|
int bitbang_initialize (PROGRAMMER * pgm, AVRPART * p);
|
||||||
|
void bitbang_disable (PROGRAMMER * pgm);
|
||||||
|
void bitbang_enable (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,19 @@
|
||||||
|
#! /bin/sh
|
||||||
|
|
||||||
|
: ${AUTOHEADER="autoheader${AC_VER}"}
|
||||||
|
: ${AUTOCONF="autoconf${AC_VER}"}
|
||||||
|
: ${ACLOCAL="aclocal${AM_VER}"}
|
||||||
|
: ${AUTOMAKE="automake${AM_VER}"}
|
||||||
|
|
||||||
|
export ACLOCAL AUTOHEADER AUTOCONF AUTOMAKE
|
||||||
|
|
||||||
|
# Bootstrap the build system.
|
||||||
|
|
||||||
|
set -x
|
||||||
|
|
||||||
|
rm -rf autom4te.cache
|
||||||
|
|
||||||
|
${ACLOCAL}
|
||||||
|
${AUTOHEADER}
|
||||||
|
${AUTOCONF}
|
||||||
|
${AUTOMAKE} -a -c
|
|
@ -0,0 +1,795 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
*
|
||||||
|
* avrdude support for The Bus Pirate - universal serial interface
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* BusPirate AVR Chip
|
||||||
|
* --------- --------
|
||||||
|
* GND <-> GND
|
||||||
|
* +5V <-> Vcc
|
||||||
|
* CS <-> RESET
|
||||||
|
* MOSI <-> MOSI
|
||||||
|
* MISO <-> MISO
|
||||||
|
* SCL/CLK <-> SCK
|
||||||
|
*
|
||||||
|
* Tested with BusPirate PTH, firmware version 2.1 programming ATmega328P
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
# include <malloc.h> /* for alloca() */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
/* ====== Private data structure ====== */
|
||||||
|
/* CS and AUX pin bitmasks in
|
||||||
|
* 0100wxyz - Configure peripherals command */
|
||||||
|
#define BP_RESET_CS 0x01
|
||||||
|
#define BP_RESET_AUX 0x02
|
||||||
|
#define BP_RESET_AUX2 0x04
|
||||||
|
|
||||||
|
#define BP_FLAG_IN_BINMODE (1<<0)
|
||||||
|
#define BP_FLAG_XPARM_FORCE_ASCII (1<<1)
|
||||||
|
#define BP_FLAG_XPARM_RESET (1<<2)
|
||||||
|
#define BP_FLAG_XPARM_SPIFREQ (1<<3)
|
||||||
|
|
||||||
|
struct pdata
|
||||||
|
{
|
||||||
|
char hw_version[10];
|
||||||
|
int fw_version; /* = 100*fw_major + fw_minor */
|
||||||
|
int binmode_version;
|
||||||
|
int bin_spi_version;
|
||||||
|
int current_peripherals_config;
|
||||||
|
int spifreq; /* 0..7 - see buspirate manual for what freq each value means */
|
||||||
|
int reset; /* See BP_RESET_* above */
|
||||||
|
};
|
||||||
|
#define PDATA(pgm) ((struct pdata *)(pgm->cookie))
|
||||||
|
|
||||||
|
/* Binary mode is available from firmware v2.7 on */
|
||||||
|
#define FW_BINMODE_VER 207
|
||||||
|
|
||||||
|
/* ====== Feature checks ====== */
|
||||||
|
static inline int
|
||||||
|
buspirate_has_aux2(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
return ((PDATA(pgm)->fw_version >= 300) &&
|
||||||
|
strcmp(PDATA(pgm)->hw_version, "v1a") == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
buspirate_uses_ascii(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
return (pgm->flag & BP_FLAG_XPARM_FORCE_ASCII) ||
|
||||||
|
(PDATA(pgm)->fw_version < FW_BINMODE_VER);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ====== Serial talker functions - binmode ====== */
|
||||||
|
|
||||||
|
static void dump_mem(char *buf, size_t len)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
for (i = 0; i<len; i++) {
|
||||||
|
if (i % 8 == 0)
|
||||||
|
fprintf(stderr, "\t");
|
||||||
|
fprintf(stderr, "0x%02x ", (unsigned)buf[i] & 0xFF);
|
||||||
|
if (i % 8 == 3)
|
||||||
|
fprintf(stderr, " ");
|
||||||
|
else if (i % 8 == 7)
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
if (i % 8 != 7)
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_send_bin(struct programmer_t *pgm, char *data, size_t len)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
if (verbose > 1) {
|
||||||
|
fprintf(stderr, "%s: buspirate_send_bin():\n", progname);
|
||||||
|
dump_mem(data, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = serial_send(&pgm->fd, (unsigned char *)data, len);
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_recv_bin(struct programmer_t *pgm, char *buf, size_t len)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = serial_recv(&pgm->fd, (unsigned char *)buf, len);
|
||||||
|
if (rc < 0)
|
||||||
|
return EOF;
|
||||||
|
if (verbose > 1) {
|
||||||
|
fprintf(stderr, "%s: buspirate_recv_bin():\n", progname);
|
||||||
|
dump_mem(buf, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_expect_bin(struct programmer_t *pgm,
|
||||||
|
char *send_data, size_t send_len,
|
||||||
|
char *expect_data, size_t expect_len)
|
||||||
|
{
|
||||||
|
char *recv_buf = alloca(expect_len);
|
||||||
|
if (!pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
fprintf(stderr, "BusPirate: Internal error: buspirate_send_bin() called from ascii mode");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
buspirate_send_bin(pgm, send_data, send_len);
|
||||||
|
buspirate_recv_bin(pgm, recv_buf, expect_len);
|
||||||
|
if (memcmp(expect_data, recv_buf, expect_len) != 0)
|
||||||
|
return 0;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_expect_bin_byte(struct programmer_t *pgm,
|
||||||
|
char send_byte, char expect_byte)
|
||||||
|
{
|
||||||
|
return buspirate_expect_bin(pgm, &send_byte, 1, &expect_byte, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ====== Serial talker functions - ascii mode ====== */
|
||||||
|
|
||||||
|
static int buspirate_getc(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
unsigned char ch = 0;
|
||||||
|
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
fprintf(stderr, "BusPirate: Internal error: buspirate_getc() called from binmode");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = serial_recv(&pgm->fd, &ch, 1);
|
||||||
|
if (rc < 0)
|
||||||
|
return EOF;
|
||||||
|
return ch;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *buspirate_readline_noexit(struct programmer_t *pgm, char *buf, size_t len)
|
||||||
|
{
|
||||||
|
char *buf_p;
|
||||||
|
long orig_serial_recv_timeout = serial_recv_timeout;
|
||||||
|
|
||||||
|
/* Static local buffer - this may come handy at times */
|
||||||
|
static char buf_local[100];
|
||||||
|
|
||||||
|
if (buf == NULL) {
|
||||||
|
buf = buf_local;
|
||||||
|
len = sizeof(buf_local);
|
||||||
|
}
|
||||||
|
buf_p = buf;
|
||||||
|
memset(buf, 0, len);
|
||||||
|
while (buf_p < (buf + len - 1)) { /* keep the very last byte == 0 */
|
||||||
|
*buf_p = buspirate_getc(pgm);
|
||||||
|
if (*buf_p == '\r')
|
||||||
|
continue;
|
||||||
|
if (*buf_p == '\n')
|
||||||
|
break;
|
||||||
|
if (*buf_p == EOF) {
|
||||||
|
*buf_p = '\0';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
buf_p++;
|
||||||
|
serial_recv_timeout = 100;
|
||||||
|
}
|
||||||
|
serial_recv_timeout = orig_serial_recv_timeout;
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr, "%s: buspirate_readline(): %s%s",
|
||||||
|
progname, buf,
|
||||||
|
buf[strlen(buf) - 1] == '\n' ? "" : "\n");
|
||||||
|
if (! buf[0])
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *buspirate_readline(struct programmer_t *pgm, char *buf, size_t len)
|
||||||
|
{
|
||||||
|
char *ret;
|
||||||
|
|
||||||
|
ret = buspirate_readline_noexit(pgm, buf, len);
|
||||||
|
if (! ret) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: buspirate_readline(): programmer is not responding\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
static int buspirate_send(struct programmer_t *pgm, char *str)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr, "%s: buspirate_send(): %s", progname, str);
|
||||||
|
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
fprintf(stderr, "BusPirate: Internal error: buspirate_send() called from binmode");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = serial_send(&pgm->fd, (unsigned char *)str, strlen(str));
|
||||||
|
if (rc)
|
||||||
|
return rc;
|
||||||
|
while (strcmp(buspirate_readline(pgm, NULL, 0), str) != 0)
|
||||||
|
/* keep reading until we get what we sent there */
|
||||||
|
;
|
||||||
|
/* by now we should be in sync */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_is_prompt(char *str)
|
||||||
|
{
|
||||||
|
/* Prompt ends with '>' or '> '
|
||||||
|
* all other input probably ends with '\n' */
|
||||||
|
return (str[strlen(str) - 1] == '>' || str[strlen(str) - 2] == '>');
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_expect(struct programmer_t *pgm, char *send,
|
||||||
|
char *expect, int wait_for_prompt)
|
||||||
|
{
|
||||||
|
int got_it = 0;
|
||||||
|
size_t expect_len = strlen(expect);
|
||||||
|
char *rcvd;
|
||||||
|
|
||||||
|
buspirate_send(pgm, send);
|
||||||
|
while (1) {
|
||||||
|
rcvd = buspirate_readline(pgm, NULL, 0);
|
||||||
|
|
||||||
|
if (strncmp(rcvd, expect, expect_len) == 0)
|
||||||
|
got_it = 1;
|
||||||
|
|
||||||
|
if (buspirate_is_prompt(rcvd))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return got_it;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ====== Do-nothing functions ====== */
|
||||||
|
static void buspirate_dummy_6(struct programmer_t *pgm,
|
||||||
|
const char *p)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ====== Config / parameters handling functions ====== */
|
||||||
|
static int
|
||||||
|
buspirate_parseextparms(struct programmer_t *pgm, LISTID extparms)
|
||||||
|
{
|
||||||
|
LNODEID ln;
|
||||||
|
const char *extended_param;
|
||||||
|
char reset[10];
|
||||||
|
char *preset = reset; /* for strtok() */
|
||||||
|
int spifreq;
|
||||||
|
|
||||||
|
for (ln = lfirst(extparms); ln; ln = lnext(ln)) {
|
||||||
|
extended_param = ldata(ln);
|
||||||
|
if (strcmp(extended_param, "ascii") == 0) {
|
||||||
|
pgm->flag |= BP_FLAG_XPARM_FORCE_ASCII;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (sscanf(extended_param, "spifreq=%d", &spifreq) == 1) {
|
||||||
|
if (spifreq & (~0x07)) {
|
||||||
|
fprintf(stderr, "BusPirate: spifreq must be between 0 and 7.\n");
|
||||||
|
fprintf(stderr, "BusPirate: see BusPirate manual for details.\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
PDATA(pgm)->spifreq = spifreq;
|
||||||
|
pgm->flag |= BP_FLAG_XPARM_SPIFREQ;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sscanf(extended_param, "reset=%s", reset) == 1) {
|
||||||
|
char *resetpin;
|
||||||
|
while ((resetpin = strtok(preset, ","))) {
|
||||||
|
preset = NULL; /* for subsequent strtok() calls */
|
||||||
|
if (strcasecmp(resetpin, "cs") == 0)
|
||||||
|
PDATA(pgm)->reset |= BP_RESET_CS;
|
||||||
|
else if (strcasecmp(resetpin, "aux") == 0 || strcasecmp(reset, "aux1") == 0)
|
||||||
|
PDATA(pgm)->reset |= BP_RESET_AUX;
|
||||||
|
else if (strcasecmp(resetpin, "aux2") == 0)
|
||||||
|
PDATA(pgm)->reset |= BP_RESET_AUX2;
|
||||||
|
else {
|
||||||
|
fprintf(stderr, "BusPirate: reset must be either CS or AUX.\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pgm->flag |= BP_FLAG_XPARM_RESET;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
buspirate_verifyconfig(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
/* Default reset pin is CS */
|
||||||
|
if (PDATA(pgm)->reset == 0x00)
|
||||||
|
PDATA(pgm)->reset |= BP_RESET_CS;
|
||||||
|
|
||||||
|
/* reset=AUX2 is only available on HW=v1a and FW>=3.0 */
|
||||||
|
if ((PDATA(pgm)->reset & BP_RESET_AUX2) && !buspirate_has_aux2(pgm)) {
|
||||||
|
fprintf(stderr, "BusPirate: Pin AUX2 is only available in binary mode\n");
|
||||||
|
fprintf(stderr, "BusPirate: with hardware==v1a && firmware>=3.0\n");
|
||||||
|
fprintf(stderr, "BusPirate: Your hardware==%s and firmware==%d.%d\n",
|
||||||
|
PDATA(pgm)->hw_version, PDATA(pgm)->fw_version/100, PDATA(pgm)->fw_version%100);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((PDATA(pgm)->reset != BP_RESET_CS) && buspirate_uses_ascii(pgm)) {
|
||||||
|
fprintf(stderr, "BusPirate: RESET pin other than CS is not supported in ASCII mode\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((pgm->flag & BP_FLAG_XPARM_SPIFREQ) && buspirate_uses_ascii(pgm)) {
|
||||||
|
fprintf(stderr, "BusPirate: SPI speed selection is not supported in ASCII mode\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ====== Programmer methods ======= */
|
||||||
|
static int buspirate_open(struct programmer_t *pgm, char * port)
|
||||||
|
{
|
||||||
|
/* BusPirate runs at 115200 by default */
|
||||||
|
if(pgm->baudrate == 0)
|
||||||
|
pgm->baudrate = 115200;
|
||||||
|
|
||||||
|
strcpy(pgm->port, port);
|
||||||
|
if (serial_open(port, pgm->baudrate, &pgm->fd)==-1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* drain any extraneous input */
|
||||||
|
serial_drain(&pgm->fd, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void buspirate_close(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
serial_close(&pgm->fd);
|
||||||
|
pgm->fd.ifd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void buspirate_reset_from_binmode(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
char buf[10];
|
||||||
|
|
||||||
|
buf[0] = 0x00; /* BinMode: revert to HiZ */
|
||||||
|
buspirate_send_bin(pgm, buf, 1);
|
||||||
|
|
||||||
|
buf[0] = 0x0F; /* BinMode: reset */
|
||||||
|
buspirate_send_bin(pgm, buf, 1);
|
||||||
|
|
||||||
|
/* read back all output */
|
||||||
|
memset(buf, '\0', sizeof(buf));
|
||||||
|
for (;;) {
|
||||||
|
int rc;
|
||||||
|
rc = buspirate_recv_bin(pgm, buf, sizeof(buf) - 1);
|
||||||
|
|
||||||
|
if (buspirate_is_prompt(buf)) {
|
||||||
|
pgm->flag &= ~BP_FLAG_IN_BINMODE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (rc == EOF)
|
||||||
|
break;
|
||||||
|
memset(buf, '\0', sizeof(buf));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
fprintf(stderr, "BusPirate reset failed. You may need to powercycle it.\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr, "BusPirate is back in the text mode\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_start_spi_mode_bin(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
char buf[20] = { '\0' };
|
||||||
|
|
||||||
|
/* == Switch to binmode - send 20x '\0' == */
|
||||||
|
buspirate_send_bin(pgm, buf, sizeof(buf));
|
||||||
|
|
||||||
|
/* Expecting 'BBIOx' reply */
|
||||||
|
memset(buf, 0, sizeof(buf));
|
||||||
|
buspirate_recv_bin(pgm, buf, 5);
|
||||||
|
if (sscanf(buf, "BBIO%d", &PDATA(pgm)->binmode_version) != 1) {
|
||||||
|
fprintf(stderr, "Binary mode not confirmed: '%s'\n", buf);
|
||||||
|
buspirate_reset_from_binmode(pgm);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr, "BusPirate binmode version: %d\n",
|
||||||
|
PDATA(pgm)->binmode_version);
|
||||||
|
|
||||||
|
pgm->flag |= BP_FLAG_IN_BINMODE;
|
||||||
|
|
||||||
|
/* == Enter SPI mode == */
|
||||||
|
buf[0] = 0x01; /* Enter raw SPI mode */
|
||||||
|
buspirate_send_bin(pgm, buf, 1);
|
||||||
|
memset(buf, 0, sizeof(buf));
|
||||||
|
buspirate_recv_bin(pgm, buf, 4);
|
||||||
|
if (sscanf(buf, "SPI%d", &PDATA(pgm)->bin_spi_version) != 1) {
|
||||||
|
fprintf(stderr, "SPI mode not confirmed: '%s'\n", buf);
|
||||||
|
buspirate_reset_from_binmode(pgm);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr, "BusPirate SPI version: %d\n",
|
||||||
|
PDATA(pgm)->bin_spi_version);
|
||||||
|
|
||||||
|
/* 0b0100wxyz - Configure peripherals w=power, x=pull-ups/aux2, y=AUX, z=CS
|
||||||
|
* we want power (0x48) and all reset pins high. */
|
||||||
|
PDATA(pgm)->current_peripherals_config = 0x48;
|
||||||
|
PDATA(pgm)->current_peripherals_config |= BP_RESET_CS;
|
||||||
|
PDATA(pgm)->current_peripherals_config |= BP_RESET_AUX;
|
||||||
|
if (buspirate_has_aux2(pgm))
|
||||||
|
PDATA(pgm)->current_peripherals_config |= BP_RESET_AUX2;
|
||||||
|
buspirate_expect_bin_byte(pgm, PDATA(pgm)->current_peripherals_config, 0x01);
|
||||||
|
usleep(50000); // sleep for 50ms after power up
|
||||||
|
|
||||||
|
/* 01100xxx - SPI speed
|
||||||
|
* xxx = 000=30kHz, 001=125kHz, 010=250kHz, 011=1MHz,
|
||||||
|
* 100=2MHz, 101=2.6MHz, 110=4MHz, 111=8MHz
|
||||||
|
* use 30kHz = 0x60 */
|
||||||
|
buspirate_expect_bin_byte(pgm, 0x60 | PDATA(pgm)->spifreq, 0x01);
|
||||||
|
|
||||||
|
/* 1000wxyz - SPI config, w=HiZ(0)/3.3v(1), x=CLK idle, y=CLK edge, z=SMP sample
|
||||||
|
* we want: 3.3V(1), idle low(0), data change on trailing edge (1),
|
||||||
|
* sample in the middle of the pulse (0)
|
||||||
|
* => 0b10001010 = 0x8a */
|
||||||
|
buspirate_expect_bin_byte(pgm, 0x8A, 0x01);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_start_spi_mode_ascii(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
int spi_cmd = -1;
|
||||||
|
int cmd;
|
||||||
|
char *rcvd, mode[11], buf[5];
|
||||||
|
|
||||||
|
buspirate_send(pgm, "M\n");
|
||||||
|
while(1) {
|
||||||
|
rcvd = buspirate_readline(pgm, NULL, 0);
|
||||||
|
if (spi_cmd == -1 && sscanf(rcvd, "%d. %10s", &cmd, mode)) {
|
||||||
|
if (strcmp(mode, "SPI") == 0)
|
||||||
|
spi_cmd = cmd;
|
||||||
|
}
|
||||||
|
if (buspirate_is_prompt(rcvd))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (spi_cmd == -1) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: SPI mode number not found. Does your BusPirate support SPI?\n",
|
||||||
|
progname);
|
||||||
|
fprintf(stderr, "%s: Try powercycling your BusPirate and try again.\n",
|
||||||
|
progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
snprintf(buf, sizeof(buf), "%d\n", spi_cmd);
|
||||||
|
buspirate_send(pgm, buf);
|
||||||
|
buf[0] = '\0';
|
||||||
|
while (1) {
|
||||||
|
rcvd = buspirate_readline(pgm, NULL, 0);
|
||||||
|
if (strstr(rcvd, "Normal (H=3.3V, L=GND)")) {
|
||||||
|
/* BP firmware 2.1 defaults to Open-drain output.
|
||||||
|
* That doesn't work on my board, even with pull-up
|
||||||
|
* resistors. Select 3.3V output mode instead. */
|
||||||
|
sscanf(rcvd, " %d.", &cmd);
|
||||||
|
snprintf(buf, sizeof(buf), "%d\n", cmd);
|
||||||
|
}
|
||||||
|
if (buspirate_is_prompt(rcvd)) {
|
||||||
|
if (strncmp(rcvd, "SPI>", 4) == 0) {
|
||||||
|
fprintf(stderr, "BusPirate is now configured for SPI\n");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* Not yet 'SPI>' prompt */
|
||||||
|
if (buf[0]) {
|
||||||
|
buspirate_send(pgm, buf);
|
||||||
|
buf[0] = '\0';
|
||||||
|
} else
|
||||||
|
buspirate_send(pgm, "\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void buspirate_enable(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
unsigned char *reset_str = "#\n";
|
||||||
|
unsigned char *accept_str = "y\n";
|
||||||
|
char *rcvd;
|
||||||
|
int fw_v1 = 0, fw_v2 = 0;
|
||||||
|
int rc, print_banner = 0;
|
||||||
|
|
||||||
|
fprintf(stderr, "Detecting BusPirate...\n");
|
||||||
|
|
||||||
|
/* Call buspirate_send_bin() instead of buspirate_send()
|
||||||
|
* because we don't know if BP is in text or bin mode */
|
||||||
|
rc = buspirate_send_bin(pgm, reset_str, strlen(reset_str));
|
||||||
|
if (rc) {
|
||||||
|
fprintf(stderr, "BusPirate is not responding. Serial port error: %d\n", rc);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
while(1) {
|
||||||
|
rcvd = buspirate_readline_noexit(pgm, NULL, 0);
|
||||||
|
if (! rcvd) {
|
||||||
|
fprintf(stderr, "BusPirate is not responding. Attempting reset.\n");
|
||||||
|
buspirate_reset_from_binmode(pgm);
|
||||||
|
/* re-run buspirate_enable() */
|
||||||
|
buspirate_enable(pgm);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (strncmp(rcvd, "Are you sure?", 13) == 0) {
|
||||||
|
buspirate_send_bin(pgm, accept_str, strlen(accept_str));
|
||||||
|
}
|
||||||
|
if (strncmp(rcvd, "RESET", 5) == 0) {
|
||||||
|
print_banner = 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (buspirate_is_prompt(rcvd)) {
|
||||||
|
puts("**");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sscanf(rcvd, "Bus Pirate %9s", PDATA(pgm)->hw_version);
|
||||||
|
sscanf(rcvd, "Firmware v%d.%d", &fw_v1, &fw_v2);
|
||||||
|
if (print_banner)
|
||||||
|
fprintf(stderr, "** %s", rcvd);
|
||||||
|
}
|
||||||
|
|
||||||
|
PDATA(pgm)->fw_version = 100 * fw_v1 + fw_v2;
|
||||||
|
if (PDATA(pgm)->hw_version[0] == 0 || PDATA(pgm)->fw_version == 0) {
|
||||||
|
fprintf(stderr, "BusPirate not detected. Aborting.\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buspirate_verifyconfig(pgm) < 0)
|
||||||
|
exit(1);
|
||||||
|
|
||||||
|
if (PDATA(pgm)->fw_version >= FW_BINMODE_VER && !(pgm->flag & BP_FLAG_XPARM_FORCE_ASCII)) {
|
||||||
|
fprintf(stderr, "BusPirate: using BINARY mode\n");
|
||||||
|
if (buspirate_start_spi_mode_bin(pgm) < 0)
|
||||||
|
fprintf(stderr, "%s: Failed to start binary SPI mode\n", progname);
|
||||||
|
}
|
||||||
|
if (!pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
fprintf(stderr, "BusPirate: using ASCII mode\n");
|
||||||
|
if (buspirate_start_spi_mode_ascii(pgm) < 0) {
|
||||||
|
fprintf(stderr, "%s: Failed to start ascii SPI mode\n", progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void buspirate_disable(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
serial_recv_timeout = 100;
|
||||||
|
buspirate_reset_from_binmode(pgm);
|
||||||
|
} else
|
||||||
|
buspirate_expect(pgm, "#\n", "RESET", 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_initialize(struct programmer_t *pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
pgm->powerup(pgm);
|
||||||
|
|
||||||
|
return pgm->program_enable(pgm, p);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void buspirate_powerup(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
/* Powerup in BinMode is handled in SPI init */
|
||||||
|
return;
|
||||||
|
} else
|
||||||
|
if (buspirate_expect(pgm, "W\n", "POWER SUPPLIES ON", 1))
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: warning: did not get a response to PowerUp command.\n", progname);
|
||||||
|
fprintf(stderr, "%s: warning: Trying to continue anyway...\n", progname);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void buspirate_powerdown(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
/* 0b0100wxyz - Configure peripherals w=power, x=pull-ups, y=AUX, z=CS
|
||||||
|
* we want everything off -- 0b01000000 = 0x40 */
|
||||||
|
if (buspirate_expect_bin_byte(pgm, 0x40, 0x01))
|
||||||
|
return;
|
||||||
|
} else
|
||||||
|
if (buspirate_expect(pgm, "w\n", "POWER SUPPLIES OFF", 1))
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: warning: did not get a response to PowerDown command.\n", progname);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_cmd_bin(struct programmer_t *pgm,
|
||||||
|
unsigned char cmd[4],
|
||||||
|
unsigned char res[4])
|
||||||
|
{
|
||||||
|
/* 0001xxxx - Bulk SPI transfer, send/read 1-16 bytes (0=1byte!)
|
||||||
|
* we are sending 4 bytes -> 0x13 */
|
||||||
|
if (!buspirate_expect_bin_byte(pgm, 0x13, 0x01))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
buspirate_send_bin(pgm, (char *)cmd, 4);
|
||||||
|
buspirate_recv_bin(pgm, (char *)res, 4);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_cmd_ascii(struct programmer_t *pgm,
|
||||||
|
unsigned char cmd[4],
|
||||||
|
unsigned char res[4])
|
||||||
|
{
|
||||||
|
char buf[25];
|
||||||
|
char *rcvd;
|
||||||
|
int spi_write, spi_read, i = 0;
|
||||||
|
|
||||||
|
snprintf(buf, sizeof(buf), "0x%02x 0x%02x 0x%02x 0x%02x\n",
|
||||||
|
cmd[0], cmd[1], cmd[2], cmd[3]);
|
||||||
|
buspirate_send(pgm, buf);
|
||||||
|
while (i < 4) {
|
||||||
|
rcvd = buspirate_readline(pgm, NULL, 0);
|
||||||
|
/* WRITE: 0xAC READ: 0x04 */
|
||||||
|
if (sscanf(rcvd, "WRITE: 0x%x READ: 0x%x", &spi_write, &spi_read) == 2) {
|
||||||
|
res[i++] = spi_read;
|
||||||
|
}
|
||||||
|
if (buspirate_is_prompt(rcvd))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i != 4) {
|
||||||
|
fprintf(stderr, "%s: error: SPI has not read 4 bytes back\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* wait for prompt */
|
||||||
|
while (buspirate_getc(pgm) != '>')
|
||||||
|
/* do nothing */;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_cmd(struct programmer_t *pgm,
|
||||||
|
unsigned char cmd[4],
|
||||||
|
unsigned char res[4])
|
||||||
|
{
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE)
|
||||||
|
return buspirate_cmd_bin(pgm, cmd, res);
|
||||||
|
else
|
||||||
|
return buspirate_cmd_ascii(pgm, cmd, res);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_program_enable(struct programmer_t *pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
unsigned char cmd[4];
|
||||||
|
unsigned char res[4];
|
||||||
|
|
||||||
|
if (pgm->flag & BP_FLAG_IN_BINMODE) {
|
||||||
|
/* Clear configured reset pin(s): CS and/or AUX and/or AUX2 */
|
||||||
|
PDATA(pgm)->current_peripherals_config &= ~PDATA(pgm)->reset;
|
||||||
|
buspirate_expect_bin_byte(pgm, PDATA(pgm)->current_peripherals_config, 0x01);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
buspirate_expect(pgm, "{\n", "CS ENABLED", 1);
|
||||||
|
|
||||||
|
if (p->op[AVR_OP_PGM_ENABLE] == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"program enable instruction not defined for part \"%s\"\n",
|
||||||
|
p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(cmd));
|
||||||
|
avr_set_bits(p->op[AVR_OP_PGM_ENABLE], cmd);
|
||||||
|
pgm->cmd(pgm, cmd, res);
|
||||||
|
|
||||||
|
if (res[2] != cmd[1])
|
||||||
|
return -2;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int buspirate_chip_erase(struct programmer_t *pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
unsigned char cmd[4];
|
||||||
|
unsigned char res[4];
|
||||||
|
|
||||||
|
if (p->op[AVR_OP_CHIP_ERASE] == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"chip erase instruction not defined for part \"%s\"\n",
|
||||||
|
p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, ON);
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(cmd));
|
||||||
|
|
||||||
|
avr_set_bits(p->op[AVR_OP_CHIP_ERASE], cmd);
|
||||||
|
pgm->cmd(pgm, cmd, res);
|
||||||
|
usleep(p->chip_erase_delay);
|
||||||
|
pgm->initialize(pgm, p);
|
||||||
|
|
||||||
|
pgm->pgm_led(pgm, OFF);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void buspirate_initpgm(struct programmer_t *pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "BusPirate");
|
||||||
|
|
||||||
|
pgm->display = buspirate_dummy_6;
|
||||||
|
|
||||||
|
/* BusPirate itself related methods */
|
||||||
|
pgm->open = buspirate_open;
|
||||||
|
pgm->close = buspirate_close;
|
||||||
|
pgm->enable = buspirate_enable;
|
||||||
|
pgm->disable = buspirate_disable;
|
||||||
|
pgm->initialize = buspirate_initialize;
|
||||||
|
|
||||||
|
/* Chip related methods */
|
||||||
|
pgm->powerup = buspirate_powerup;
|
||||||
|
pgm->powerdown = buspirate_powerdown;
|
||||||
|
pgm->program_enable = buspirate_program_enable;
|
||||||
|
pgm->chip_erase = buspirate_chip_erase;
|
||||||
|
pgm->cmd = buspirate_cmd;
|
||||||
|
pgm->read_byte = avr_read_byte_default;
|
||||||
|
pgm->write_byte = avr_write_byte_default;
|
||||||
|
|
||||||
|
/* Support functions */
|
||||||
|
pgm->parseextparams = buspirate_parseextparms;
|
||||||
|
|
||||||
|
/* Allocate private data */
|
||||||
|
if ((pgm->cookie = calloc(1, sizeof(struct pdata))) == 0) {
|
||||||
|
fprintf(stderr, "%s: buspirate_initpgm(): Out of memory allocating private data\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,30 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
*
|
||||||
|
* avrdude support for The Bus Pirate - universal serial interface
|
||||||
|
*
|
||||||
|
* Copyright (C) 2009 Michal Ludvig <mludvig@logix.net.nz>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef buspirate_h
|
||||||
|
#define buspirate_h
|
||||||
|
|
||||||
|
void buspirate_initpgm (struct programmer_t *pgm);
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,747 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
* Copyright (C) 2005, 2007 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avrdude interface for the serial programming mode of the Atmel butterfly
|
||||||
|
* evaluation board. This board features a bootloader which uses a protocol
|
||||||
|
* very similar, but not identical, to the one described in application note
|
||||||
|
* avr910.
|
||||||
|
*
|
||||||
|
* Actually, the butterfly uses a predecessor of the avr910 protocol
|
||||||
|
* which is described in application notes avr109 (generic AVR
|
||||||
|
* bootloader) and avr911 (opensource programmer). This file now
|
||||||
|
* fully handles the features present in avr109. It should probably
|
||||||
|
* be renamed to avr109, but we rather stick with the old name inside
|
||||||
|
* the file. We'll provide aliases for "avr109" and "avr911" in
|
||||||
|
* avrdude.conf so users could call it by these name as well.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "butterfly.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Private data for this programmer.
|
||||||
|
*/
|
||||||
|
struct pdata
|
||||||
|
{
|
||||||
|
char has_auto_incr_addr;
|
||||||
|
unsigned int buffersize;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define PDATA(pgm) ((struct pdata *)(pgm->cookie))
|
||||||
|
|
||||||
|
static void butterfly_setup(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
if ((pgm->cookie = malloc(sizeof(struct pdata))) == 0) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: butterfly_setup(): Out of memory allocating private data\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
memset(pgm->cookie, 0, sizeof(struct pdata));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void butterfly_teardown(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
free(pgm->cookie);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int butterfly_send(PROGRAMMER * pgm, char * buf, size_t len)
|
||||||
|
{
|
||||||
|
return serial_send(&pgm->fd, (unsigned char *)buf, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_recv(PROGRAMMER * pgm, char * buf, size_t len)
|
||||||
|
{
|
||||||
|
int rv;
|
||||||
|
|
||||||
|
rv = serial_recv(&pgm->fd, (unsigned char *)buf, len);
|
||||||
|
if (rv < 0) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: butterfly_recv(): programmer is not responding\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_drain(PROGRAMMER * pgm, int display)
|
||||||
|
{
|
||||||
|
return serial_drain(&pgm->fd, display);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_vfy_cmd_sent(PROGRAMMER * pgm, char * errmsg)
|
||||||
|
{
|
||||||
|
char c;
|
||||||
|
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
if (c != '\r') {
|
||||||
|
fprintf(stderr, "%s: error: programmer did not respond to command: %s\n",
|
||||||
|
progname, errmsg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_rdy_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_err_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_pgm_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_vfy_led(PROGRAMMER * pgm, int value)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* issue the 'chip erase' command to the butterfly board
|
||||||
|
*/
|
||||||
|
static int butterfly_chip_erase(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
butterfly_send(pgm, "e", 1);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "chip erase");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_enter_prog_mode(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
butterfly_send(pgm, "P", 1);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "enter prog mode");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_leave_prog_mode(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
butterfly_send(pgm, "L", 1);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "leave prog mode");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* issue the 'program enable' command to the AVR device
|
||||||
|
*/
|
||||||
|
static int butterfly_program_enable(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* apply power to the AVR processor
|
||||||
|
*/
|
||||||
|
static void butterfly_powerup(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* remove power from the AVR processor
|
||||||
|
*/
|
||||||
|
static void butterfly_powerdown(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/* Do nothing. */
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define IS_BUTTERFLY_MK 0x0001
|
||||||
|
|
||||||
|
/*
|
||||||
|
* initialize the AVR device and prepare it to accept commands
|
||||||
|
*/
|
||||||
|
static int butterfly_initialize(PROGRAMMER * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
char id[8];
|
||||||
|
char sw[2];
|
||||||
|
char hw[2];
|
||||||
|
char buf[10];
|
||||||
|
char type;
|
||||||
|
char c, devtype_1st;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Send some ESC to activate butterfly bootloader. This is not needed
|
||||||
|
* for plain avr109 bootloaders but does not harm there either.
|
||||||
|
*/
|
||||||
|
fprintf(stderr, "Connecting to programmer: ");
|
||||||
|
if (pgm->flag & IS_BUTTERFLY_MK)
|
||||||
|
{
|
||||||
|
char mk_reset_cmd[6] = {"#aR@S\r"};
|
||||||
|
unsigned char mk_timeout = 0;
|
||||||
|
|
||||||
|
putc('.', stderr);
|
||||||
|
butterfly_send(pgm, mk_reset_cmd, sizeof(mk_reset_cmd));
|
||||||
|
usleep(20000);
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
c = 27;
|
||||||
|
butterfly_send(pgm, &c, 1);
|
||||||
|
usleep(20000);
|
||||||
|
c = 0xaa;
|
||||||
|
usleep(80000);
|
||||||
|
butterfly_send(pgm, &c, 1);
|
||||||
|
if (mk_timeout % 10 == 0) putc('.', stderr);
|
||||||
|
} while (mk_timeout++ < 10);
|
||||||
|
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
if ( c != 'M' && c != '?')
|
||||||
|
{
|
||||||
|
fprintf(stderr, "\nConnection FAILED.");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
id[0] = 'M'; id[1] = 'K'; id[2] = '2'; id[3] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
do {
|
||||||
|
putc('.', stderr);
|
||||||
|
butterfly_send(pgm, "\033", 1);
|
||||||
|
butterfly_drain(pgm, 0);
|
||||||
|
butterfly_send(pgm, "S", 1);
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
if (c != '?') {
|
||||||
|
putc('\n', stderr);
|
||||||
|
/*
|
||||||
|
* Got a useful response, continue getting the programmer
|
||||||
|
* identifier. Programmer returns exactly 7 chars _without_
|
||||||
|
* the null.
|
||||||
|
*/
|
||||||
|
id[0] = c;
|
||||||
|
butterfly_recv(pgm, &id[1], sizeof(id)-2);
|
||||||
|
id[sizeof(id)-1] = '\0';
|
||||||
|
}
|
||||||
|
} while (c == '?');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get the HW and SW versions to see if the programmer is present. */
|
||||||
|
butterfly_drain(pgm, 0);
|
||||||
|
|
||||||
|
butterfly_send(pgm, "V", 1);
|
||||||
|
butterfly_recv(pgm, sw, sizeof(sw));
|
||||||
|
|
||||||
|
butterfly_send(pgm, "v", 1);
|
||||||
|
butterfly_recv(pgm, hw, 1); /* first, read only _one_ byte */
|
||||||
|
if (hw[0]!='?') {
|
||||||
|
butterfly_recv(pgm, &hw[1], 1);/* now, read second byte */
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Get the programmer type (serial or parallel). Expect serial. */
|
||||||
|
|
||||||
|
butterfly_send(pgm, "p", 1);
|
||||||
|
butterfly_recv(pgm, &type, 1);
|
||||||
|
|
||||||
|
fprintf(stderr, "Found programmer: Id = \"%s\"; type = %c\n", id, type);
|
||||||
|
fprintf(stderr, " Software Version = %c.%c; ", sw[0], sw[1]);
|
||||||
|
if (hw[0]=='?') {
|
||||||
|
fprintf(stderr, "No Hardware Version given.\n");
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "Hardware Version = %c.%c\n", hw[0], hw[1]);
|
||||||
|
};
|
||||||
|
|
||||||
|
/* See if programmer supports autoincrement of address. */
|
||||||
|
|
||||||
|
butterfly_send(pgm, "a", 1);
|
||||||
|
butterfly_recv(pgm, &PDATA(pgm)->has_auto_incr_addr, 1);
|
||||||
|
if (PDATA(pgm)->has_auto_incr_addr == 'Y')
|
||||||
|
fprintf(stderr, "Programmer supports auto addr increment.\n");
|
||||||
|
|
||||||
|
/* Check support for buffered memory access, abort if not available */
|
||||||
|
|
||||||
|
butterfly_send(pgm, "b", 1);
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
if (c != 'Y') {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: error: buffered memory access not supported. Maybe it isn't\n"\
|
||||||
|
"a butterfly/AVR109 but a AVR910 device?\n", progname);
|
||||||
|
exit(1);
|
||||||
|
};
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
PDATA(pgm)->buffersize = (unsigned int)(unsigned char)c<<8;
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
PDATA(pgm)->buffersize += (unsigned int)(unsigned char)c;
|
||||||
|
fprintf(stderr,
|
||||||
|
"Programmer supports buffered memory access with buffersize=%i bytes.\n",
|
||||||
|
PDATA(pgm)->buffersize);
|
||||||
|
|
||||||
|
/* Get list of devices that the programmer supports. */
|
||||||
|
|
||||||
|
butterfly_send(pgm, "t", 1);
|
||||||
|
fprintf(stderr, "\nProgrammer supports the following devices:\n");
|
||||||
|
devtype_1st = 0;
|
||||||
|
while (1) {
|
||||||
|
butterfly_recv(pgm, &c, 1);
|
||||||
|
if (devtype_1st == 0)
|
||||||
|
devtype_1st = c;
|
||||||
|
|
||||||
|
if (c == 0)
|
||||||
|
break;
|
||||||
|
fprintf(stderr, " Device code: 0x%02x\n", (unsigned int)(unsigned char)c);
|
||||||
|
};
|
||||||
|
fprintf(stderr,"\n");
|
||||||
|
|
||||||
|
/* Tell the programmer which part we selected.
|
||||||
|
According to the AVR109 code, this is ignored by the bootloader. As
|
||||||
|
some early versions might not properly ignore it, rather pick up the
|
||||||
|
first device type as reported above than anything out of avrdude.conf,
|
||||||
|
so to avoid a potential conflict. There appears to be no general
|
||||||
|
agreement on AVR910 device IDs beyond the ones from the original
|
||||||
|
appnote 910. */
|
||||||
|
|
||||||
|
buf[0] = 'T';
|
||||||
|
buf[1] = devtype_1st;
|
||||||
|
|
||||||
|
butterfly_send(pgm, buf, 2);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "select device");
|
||||||
|
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: devcode selected: 0x%02x\n",
|
||||||
|
progname, (unsigned)buf[1]);
|
||||||
|
|
||||||
|
butterfly_enter_prog_mode(pgm);
|
||||||
|
butterfly_drain(pgm, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_disable(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
butterfly_leave_prog_mode(pgm);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_enable(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_open(PROGRAMMER * pgm, char * port)
|
||||||
|
{
|
||||||
|
strcpy(pgm->port, port);
|
||||||
|
/*
|
||||||
|
* If baudrate was not specified use 19200 Baud
|
||||||
|
*/
|
||||||
|
if(pgm->baudrate == 0) {
|
||||||
|
pgm->baudrate = 19200;
|
||||||
|
}
|
||||||
|
if (serial_open(port, pgm->baudrate, &pgm->fd)==-1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* drain any extraneous input
|
||||||
|
*/
|
||||||
|
butterfly_drain (pgm, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_close(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/* "exit programmer" */
|
||||||
|
butterfly_send(pgm, "E", 1);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "exit bootloader");
|
||||||
|
|
||||||
|
serial_close(&pgm->fd);
|
||||||
|
pgm->fd.ifd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_display(PROGRAMMER * pgm, const char * p)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_set_addr(PROGRAMMER * pgm, unsigned long addr)
|
||||||
|
{
|
||||||
|
char cmd[3];
|
||||||
|
|
||||||
|
cmd[0] = 'A';
|
||||||
|
cmd[1] = (addr >> 8) & 0xff;
|
||||||
|
cmd[2] = addr & 0xff;
|
||||||
|
|
||||||
|
butterfly_send(pgm, cmd, sizeof(cmd));
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "set addr");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void butterfly_set_extaddr(PROGRAMMER * pgm, unsigned long addr)
|
||||||
|
{
|
||||||
|
char cmd[4];
|
||||||
|
|
||||||
|
cmd[0] = 'H';
|
||||||
|
cmd[1] = (addr >> 16) & 0xff;
|
||||||
|
cmd[2] = (addr >> 8) & 0xff;
|
||||||
|
cmd[3] = addr & 0xff;
|
||||||
|
|
||||||
|
butterfly_send(pgm, cmd, sizeof(cmd));
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "set extaddr");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_write_byte(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char value)
|
||||||
|
{
|
||||||
|
char cmd[6];
|
||||||
|
int size;
|
||||||
|
int use_ext_addr = m->op[AVR_OP_LOAD_EXT_ADDR] != NULL;
|
||||||
|
|
||||||
|
if ((strcmp(m->desc, "flash") == 0) || (strcmp(m->desc, "eeprom") == 0))
|
||||||
|
{
|
||||||
|
cmd[0] = 'B';
|
||||||
|
cmd[1] = 0;
|
||||||
|
if ((cmd[3] = toupper((int)(m->desc[0]))) == 'E') { /* write to eeprom */
|
||||||
|
cmd[2] = 1;
|
||||||
|
cmd[4] = value;
|
||||||
|
size = 5;
|
||||||
|
} else { /* write to flash */
|
||||||
|
/* @@@ not yet implemented */
|
||||||
|
cmd[2] = 2;
|
||||||
|
size = 6;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (use_ext_addr) {
|
||||||
|
butterfly_set_extaddr(pgm, addr);
|
||||||
|
} else {
|
||||||
|
butterfly_set_addr(pgm, addr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (strcmp(m->desc, "lock") == 0)
|
||||||
|
{
|
||||||
|
cmd[0] = 'l';
|
||||||
|
cmd[1] = value;
|
||||||
|
size = 2;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
butterfly_send(pgm, cmd, size);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "write byte");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_read_byte_flash(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
static int cached = 0;
|
||||||
|
static unsigned char cvalue;
|
||||||
|
static unsigned long caddr;
|
||||||
|
int use_ext_addr = m->op[AVR_OP_LOAD_EXT_ADDR] != NULL;
|
||||||
|
|
||||||
|
if (cached && ((caddr + 1) == addr)) {
|
||||||
|
*value = cvalue;
|
||||||
|
cached = 0;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
char buf[2];
|
||||||
|
|
||||||
|
if (use_ext_addr) {
|
||||||
|
butterfly_set_extaddr(pgm, addr >> 1);
|
||||||
|
} else {
|
||||||
|
butterfly_set_addr(pgm, addr >> 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
butterfly_send(pgm, "g\000\002F", 4);
|
||||||
|
|
||||||
|
/* Read back the program mem word (MSB first) */
|
||||||
|
butterfly_recv(pgm, buf, sizeof(buf));
|
||||||
|
|
||||||
|
if ((addr & 0x01) == 0) {
|
||||||
|
*value = buf[1];
|
||||||
|
cached = 1;
|
||||||
|
cvalue = buf[0];
|
||||||
|
caddr = addr;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
*value = buf[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_read_byte_eeprom(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
butterfly_set_addr(pgm, addr);
|
||||||
|
butterfly_send(pgm, "g\000\001E", 4);
|
||||||
|
butterfly_recv(pgm, (char *)value, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_read_byte(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
char cmd;
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "flash") == 0) {
|
||||||
|
return butterfly_read_byte_flash(pgm, p, m, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "eeprom") == 0) {
|
||||||
|
return butterfly_read_byte_eeprom(pgm, p, m, addr, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "lfuse") == 0) {
|
||||||
|
cmd = 'F';
|
||||||
|
}
|
||||||
|
else if (strcmp(m->desc, "hfuse") == 0) {
|
||||||
|
cmd = 'N';
|
||||||
|
}
|
||||||
|
else if (strcmp(m->desc, "efuse") == 0) {
|
||||||
|
cmd = 'Q';
|
||||||
|
}
|
||||||
|
else if (strcmp(m->desc, "lock") == 0) {
|
||||||
|
cmd = 'r';
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
butterfly_send(pgm, &cmd, 1);
|
||||||
|
butterfly_recv(pgm, (char *)value, 1);
|
||||||
|
|
||||||
|
return *value == '?'? -1: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_paged_write(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes)
|
||||||
|
{
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr = n_bytes;
|
||||||
|
char *cmd;
|
||||||
|
unsigned int blocksize = PDATA(pgm)->buffersize;
|
||||||
|
int use_ext_addr = m->op[AVR_OP_LOAD_EXT_ADDR] != NULL;
|
||||||
|
|
||||||
|
if (strcmp(m->desc, "flash") && strcmp(m->desc, "eeprom"))
|
||||||
|
return -2;
|
||||||
|
|
||||||
|
if (m->desc[0] == 'e')
|
||||||
|
blocksize = 1; /* Write to eeprom single bytes only */
|
||||||
|
|
||||||
|
if (use_ext_addr) {
|
||||||
|
butterfly_set_extaddr(pgm, addr);
|
||||||
|
} else {
|
||||||
|
butterfly_set_addr(pgm, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
usleep(1000000);
|
||||||
|
butterfly_send(pgm, "y", 1);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "clear LED");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
cmd = malloc(4+blocksize);
|
||||||
|
if (!cmd) return -1;
|
||||||
|
cmd[0] = 'B';
|
||||||
|
cmd[3] = toupper((int)(m->desc[0]));
|
||||||
|
|
||||||
|
while (addr < max_addr) {
|
||||||
|
if ((max_addr - addr) < blocksize) {
|
||||||
|
blocksize = max_addr - addr;
|
||||||
|
};
|
||||||
|
memcpy(&cmd[4], &m->buf[addr], blocksize);
|
||||||
|
cmd[1] = (blocksize >> 8) & 0xff;
|
||||||
|
cmd[2] = blocksize & 0xff;
|
||||||
|
|
||||||
|
butterfly_send(pgm, cmd, 4+blocksize);
|
||||||
|
butterfly_vfy_cmd_sent(pgm, "write block");
|
||||||
|
|
||||||
|
addr += blocksize;
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
} /* while */
|
||||||
|
free(cmd);
|
||||||
|
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static int butterfly_paged_load(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes)
|
||||||
|
{
|
||||||
|
unsigned int addr = 0;
|
||||||
|
unsigned int max_addr = n_bytes;
|
||||||
|
int rd_size = 1;
|
||||||
|
int use_ext_addr = m->op[AVR_OP_LOAD_EXT_ADDR] != NULL;
|
||||||
|
|
||||||
|
/* check parameter syntax: only "flash" or "eeprom" is allowed */
|
||||||
|
if (strcmp(m->desc, "flash") && strcmp(m->desc, "eeprom"))
|
||||||
|
return -2;
|
||||||
|
|
||||||
|
{ /* use buffered mode */
|
||||||
|
char cmd[4];
|
||||||
|
int blocksize = PDATA(pgm)->buffersize;
|
||||||
|
|
||||||
|
cmd[0] = 'g';
|
||||||
|
cmd[3] = toupper((int)(m->desc[0]));
|
||||||
|
|
||||||
|
if (use_ext_addr) {
|
||||||
|
butterfly_set_extaddr(pgm, addr);
|
||||||
|
} else {
|
||||||
|
butterfly_set_addr(pgm, addr);
|
||||||
|
}
|
||||||
|
while (addr < max_addr) {
|
||||||
|
if ((max_addr - addr) < blocksize) {
|
||||||
|
blocksize = max_addr - addr;
|
||||||
|
};
|
||||||
|
cmd[1] = (blocksize >> 8) & 0xff;
|
||||||
|
cmd[2] = blocksize & 0xff;
|
||||||
|
|
||||||
|
butterfly_send(pgm, cmd, 4);
|
||||||
|
butterfly_recv(pgm, (char *)&m->buf[addr], blocksize);
|
||||||
|
|
||||||
|
addr += blocksize;
|
||||||
|
|
||||||
|
report_progress (addr, max_addr, NULL);
|
||||||
|
} /* while */
|
||||||
|
}
|
||||||
|
|
||||||
|
return addr * rd_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Signature byte reads are always 3 bytes. */
|
||||||
|
static int butterfly_read_sig_bytes(PROGRAMMER * pgm, AVRPART * p, AVRMEM * m)
|
||||||
|
{
|
||||||
|
unsigned char tmp;
|
||||||
|
|
||||||
|
if (m->size < 3) {
|
||||||
|
fprintf(stderr, "%s: memsize too small for sig byte read", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
butterfly_send(pgm, "s", 1);
|
||||||
|
butterfly_recv(pgm, (char *)m->buf, 3);
|
||||||
|
/* Returned signature has wrong order. */
|
||||||
|
tmp = m->buf[2];
|
||||||
|
m->buf[2] = m->buf[0];
|
||||||
|
m->buf[0] = tmp;
|
||||||
|
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void butterfly_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "butterfly");
|
||||||
|
|
||||||
|
/*
|
||||||
|
* mandatory functions
|
||||||
|
*/
|
||||||
|
pgm->rdy_led = butterfly_rdy_led;
|
||||||
|
pgm->err_led = butterfly_err_led;
|
||||||
|
pgm->pgm_led = butterfly_pgm_led;
|
||||||
|
pgm->vfy_led = butterfly_vfy_led;
|
||||||
|
pgm->initialize = butterfly_initialize;
|
||||||
|
pgm->display = butterfly_display;
|
||||||
|
pgm->enable = butterfly_enable;
|
||||||
|
pgm->disable = butterfly_disable;
|
||||||
|
pgm->powerup = butterfly_powerup;
|
||||||
|
pgm->powerdown = butterfly_powerdown;
|
||||||
|
pgm->program_enable = butterfly_program_enable;
|
||||||
|
pgm->chip_erase = butterfly_chip_erase;
|
||||||
|
pgm->open = butterfly_open;
|
||||||
|
pgm->close = butterfly_close;
|
||||||
|
pgm->read_byte = butterfly_read_byte;
|
||||||
|
pgm->write_byte = butterfly_write_byte;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* optional functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
pgm->paged_write = butterfly_paged_write;
|
||||||
|
pgm->paged_load = butterfly_paged_load;
|
||||||
|
|
||||||
|
pgm->read_sig_bytes = butterfly_read_sig_bytes;
|
||||||
|
|
||||||
|
pgm->setup = butterfly_setup;
|
||||||
|
pgm->teardown = butterfly_teardown;
|
||||||
|
pgm->flag = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void butterfly_mk_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
butterfly_initpgm(pgm);
|
||||||
|
strcpy(pgm->type, "butterfly_mk");
|
||||||
|
pgm->flag = IS_BUTTERFLY_MK;
|
||||||
|
}
|
|
@ -0,0 +1,36 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef butterfly_h
|
||||||
|
#define butterfly_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void butterfly_initpgm (PROGRAMMER * pgm);
|
||||||
|
void butterfly_mk_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* butterfly_h */
|
|
@ -0,0 +1,307 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "config.h"
|
||||||
|
#include "config_gram.h"
|
||||||
|
|
||||||
|
char default_programmer[MAX_STR_CONST];
|
||||||
|
char default_parallel[PATH_MAX];
|
||||||
|
char default_serial[PATH_MAX];
|
||||||
|
double default_bitclock;
|
||||||
|
|
||||||
|
char string_buf[MAX_STR_CONST];
|
||||||
|
char *string_buf_ptr;
|
||||||
|
|
||||||
|
LISTID string_list;
|
||||||
|
LISTID number_list;
|
||||||
|
PROGRAMMER * current_prog;
|
||||||
|
AVRPART * current_part;
|
||||||
|
AVRMEM * current_mem;
|
||||||
|
LISTID part_list;
|
||||||
|
LISTID programmers;
|
||||||
|
|
||||||
|
int lineno;
|
||||||
|
const char * infile;
|
||||||
|
|
||||||
|
extern char * yytext;
|
||||||
|
|
||||||
|
#define DEBUG 0
|
||||||
|
|
||||||
|
|
||||||
|
int init_config(void)
|
||||||
|
{
|
||||||
|
string_list = lcreat(NULL, 0);
|
||||||
|
number_list = lcreat(NULL, 0);
|
||||||
|
current_prog = NULL;
|
||||||
|
current_part = NULL;
|
||||||
|
current_mem = 0;
|
||||||
|
part_list = lcreat(NULL, 0);
|
||||||
|
programmers = lcreat(NULL, 0);
|
||||||
|
|
||||||
|
lineno = 1;
|
||||||
|
infile = NULL;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
int yywrap()
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int yyerror(char * errmsg)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s at %s:%d\n", errmsg, infile, lineno);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TOKEN * new_token(int primary)
|
||||||
|
{
|
||||||
|
TOKEN * tkn;
|
||||||
|
|
||||||
|
tkn = (TOKEN *)malloc(sizeof(TOKEN));
|
||||||
|
if (tkn == NULL) {
|
||||||
|
fprintf(stderr, "new_token(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(tkn, 0, sizeof(TOKEN));
|
||||||
|
|
||||||
|
tkn->primary = primary;
|
||||||
|
|
||||||
|
return tkn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void free_token(TOKEN * tkn)
|
||||||
|
{
|
||||||
|
if (tkn) {
|
||||||
|
switch (tkn->primary) {
|
||||||
|
case TKN_STRING:
|
||||||
|
case TKN_ID:
|
||||||
|
if (tkn->value.string)
|
||||||
|
free(tkn->value.string);
|
||||||
|
tkn->value.string = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(tkn);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void free_tokens(int n, ...)
|
||||||
|
{
|
||||||
|
TOKEN * t;
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start(ap, n);
|
||||||
|
while (n--) {
|
||||||
|
t = va_arg(ap, TOKEN *);
|
||||||
|
free_token(t);
|
||||||
|
}
|
||||||
|
va_end(ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
TOKEN * number(char * text)
|
||||||
|
{
|
||||||
|
struct token_t * tkn;
|
||||||
|
|
||||||
|
tkn = new_token(TKN_NUMBER);
|
||||||
|
tkn->value.type = V_NUM;
|
||||||
|
tkn->value.number = atof(text);
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
|
fprintf(stderr, "NUMBER(%g)\n", tkn->value.number);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return tkn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TOKEN * hexnumber(char * text)
|
||||||
|
{
|
||||||
|
struct token_t * tkn;
|
||||||
|
char * e;
|
||||||
|
|
||||||
|
tkn = new_token(TKN_NUMBER);
|
||||||
|
tkn->value.type = V_NUM;
|
||||||
|
tkn->value.number = strtoul(text, &e, 16);
|
||||||
|
if ((e == text) || (*e != 0)) {
|
||||||
|
fprintf(stderr, "error at %s:%d: can't scan hex number \"%s\"\n",
|
||||||
|
infile, lineno, text);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
|
fprintf(stderr, "HEXNUMBER(%g)\n", tkn->value.number);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return tkn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TOKEN * string(char * text)
|
||||||
|
{
|
||||||
|
struct token_t * tkn;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
tkn = new_token(TKN_STRING);
|
||||||
|
|
||||||
|
len = strlen(text);
|
||||||
|
|
||||||
|
tkn->value.type = V_STR;
|
||||||
|
tkn->value.string = (char *) malloc(len+1);
|
||||||
|
if (tkn->value.string == NULL) {
|
||||||
|
fprintf(stderr, "id(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
strcpy(tkn->value.string, text);
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
|
fprintf(stderr, "STRING(%s)\n", tkn->value.string);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return tkn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TOKEN * id(char * text)
|
||||||
|
{
|
||||||
|
struct token_t * tkn;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
tkn = new_token(TKN_ID);
|
||||||
|
|
||||||
|
len = strlen(text);
|
||||||
|
|
||||||
|
tkn->value.type = V_STR;
|
||||||
|
tkn->value.string = (char *) malloc(len+1);
|
||||||
|
if (tkn->value.string == NULL) {
|
||||||
|
fprintf(stderr, "id(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
strcpy(tkn->value.string, text);
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
|
fprintf(stderr, "ID(%s)\n", tkn->value.string);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return tkn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TOKEN * keyword(int primary)
|
||||||
|
{
|
||||||
|
struct token_t * tkn;
|
||||||
|
|
||||||
|
tkn = new_token(primary);
|
||||||
|
|
||||||
|
return tkn;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void print_token(TOKEN * tkn)
|
||||||
|
{
|
||||||
|
if (!tkn)
|
||||||
|
return;
|
||||||
|
|
||||||
|
fprintf(stderr, "token = %d = ", tkn->primary);
|
||||||
|
switch (tkn->primary) {
|
||||||
|
case TKN_NUMBER:
|
||||||
|
fprintf(stderr, "NUMBER, value=%g", tkn->value.number);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TKN_STRING:
|
||||||
|
fprintf(stderr, "STRING, value=%s", tkn->value.string);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TKN_ID:
|
||||||
|
fprintf(stderr, "ID, value=%s", tkn->value.string);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "<other>");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void pyytext(void)
|
||||||
|
{
|
||||||
|
#if DEBUG
|
||||||
|
fprintf(stderr, "TOKEN: \"%s\"\n", yytext);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
char * dup_string(const char * str)
|
||||||
|
{
|
||||||
|
char * s;
|
||||||
|
|
||||||
|
s = strdup(str);
|
||||||
|
if (s == NULL) {
|
||||||
|
fprintf(stderr, "dup_string(): out of memory\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
int read_config(const char * file)
|
||||||
|
{
|
||||||
|
FILE * f;
|
||||||
|
|
||||||
|
f = fopen(file, "r");
|
||||||
|
if (f == NULL) {
|
||||||
|
fprintf(stderr, "%s: can't open config file \"%s\": %s\n",
|
||||||
|
progname, file, strerror(errno));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
lineno = 1;
|
||||||
|
infile = file;
|
||||||
|
yyin = f;
|
||||||
|
|
||||||
|
yyparse();
|
||||||
|
|
||||||
|
fclose(f);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
|
@ -0,0 +1,109 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef config_h
|
||||||
|
#define config_h
|
||||||
|
|
||||||
|
#include "lists.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "avr.h"
|
||||||
|
|
||||||
|
|
||||||
|
#define MAX_STR_CONST 1024
|
||||||
|
|
||||||
|
enum { V_NONE, V_NUM, V_STR };
|
||||||
|
typedef struct value_t {
|
||||||
|
int type;
|
||||||
|
double number;
|
||||||
|
char * string;
|
||||||
|
} VALUE;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct token_t {
|
||||||
|
int primary;
|
||||||
|
VALUE value;
|
||||||
|
} TOKEN;
|
||||||
|
typedef struct token_t *token_p;
|
||||||
|
|
||||||
|
|
||||||
|
extern FILE * yyin;
|
||||||
|
extern PROGRAMMER * current_prog;
|
||||||
|
extern AVRPART * current_part;
|
||||||
|
extern AVRMEM * current_mem;
|
||||||
|
extern int lineno;
|
||||||
|
extern const char * infile;
|
||||||
|
extern LISTID string_list;
|
||||||
|
extern LISTID number_list;
|
||||||
|
extern LISTID part_list;
|
||||||
|
extern LISTID programmers;
|
||||||
|
extern char default_programmer[];
|
||||||
|
extern char default_parallel[];
|
||||||
|
extern char default_serial[];
|
||||||
|
extern double default_bitclock;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#if !defined(HAS_YYSTYPE)
|
||||||
|
#define YYSTYPE token_p
|
||||||
|
#endif
|
||||||
|
extern YYSTYPE yylval;
|
||||||
|
|
||||||
|
extern char string_buf[MAX_STR_CONST];
|
||||||
|
extern char *string_buf_ptr;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int yyparse(void);
|
||||||
|
|
||||||
|
|
||||||
|
int init_config(void);
|
||||||
|
|
||||||
|
TOKEN * new_token(int primary);
|
||||||
|
|
||||||
|
void free_token(TOKEN * tkn);
|
||||||
|
|
||||||
|
void free_tokens(int n, ...);
|
||||||
|
|
||||||
|
TOKEN * number(char * text);
|
||||||
|
|
||||||
|
TOKEN * hexnumber(char * text);
|
||||||
|
|
||||||
|
TOKEN * string(char * text);
|
||||||
|
|
||||||
|
TOKEN * id(char * text);
|
||||||
|
|
||||||
|
TOKEN * keyword(int primary);
|
||||||
|
|
||||||
|
void print_token(TOKEN * tkn);
|
||||||
|
|
||||||
|
void pyytext(void);
|
||||||
|
|
||||||
|
char * dup_string(const char * str);
|
||||||
|
|
||||||
|
int read_config(const char * file);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,385 @@
|
||||||
|
#
|
||||||
|
# avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
# Copyright (C) 2003, 2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
#
|
||||||
|
# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# $Id$
|
||||||
|
#
|
||||||
|
|
||||||
|
# Process this file with autoconf to produce a configure script.
|
||||||
|
|
||||||
|
AC_PREREQ(2.57)
|
||||||
|
AC_INIT(avrdude, 5.11, avrdude-dev@nongnu.org)
|
||||||
|
|
||||||
|
AC_CANONICAL_BUILD
|
||||||
|
AC_CANONICAL_HOST
|
||||||
|
AC_CANONICAL_TARGET
|
||||||
|
|
||||||
|
AC_CONFIG_SRCDIR([main.c])
|
||||||
|
AM_INIT_AUTOMAKE
|
||||||
|
AM_CONFIG_HEADER(ac_cfg.h)
|
||||||
|
|
||||||
|
# Checks for programs.
|
||||||
|
AC_PROG_CC
|
||||||
|
AC_PROG_INSTALL
|
||||||
|
AC_PROG_YACC
|
||||||
|
AC_PROG_LEX
|
||||||
|
AC_PROG_RANLIB
|
||||||
|
|
||||||
|
dnl Makefile.am:77: compiling `config_gram.c' with per-target flags requires `AM_PROG_CC_C_O' in `configure.ac'
|
||||||
|
AM_PROG_CC_C_O
|
||||||
|
|
||||||
|
# Checks for libraries.
|
||||||
|
AC_CHECK_LIB([termcap], [tputs])
|
||||||
|
AC_CHECK_LIB([ncurses], [tputs])
|
||||||
|
AC_CHECK_LIB([readline], [readline])
|
||||||
|
AC_SEARCH_LIBS([gethostent], [nsl])
|
||||||
|
AC_SEARCH_LIBS([setsockopt], [socket])
|
||||||
|
AH_TEMPLATE([HAVE_LIBUSB],
|
||||||
|
[Define if USB support is enabled via libusb])
|
||||||
|
AC_CHECK_LIB([usb], [usb_get_string_simple], [have_libusb=yes])
|
||||||
|
if test x$have_libusb = xyes; then
|
||||||
|
case $target in
|
||||||
|
*-*-darwin*)
|
||||||
|
LIBUSB="-lusb -framework CoreFoundation -framework IOKit"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
LIBUSB="-lusb"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
AC_DEFINE([HAVE_LIBUSB])
|
||||||
|
fi
|
||||||
|
AC_SUBST(LIBUSB, $LIBUSB)
|
||||||
|
|
||||||
|
AH_TEMPLATE([HAVE_LIBUSB_1_0],
|
||||||
|
[Define if USB support is enabled via libusb 1.0])
|
||||||
|
AC_CHECK_LIB([usb-1.0], [libusb_init], [have_libusb_1_0=yes])
|
||||||
|
if test x$have_libusb_1_0 = xyes; then
|
||||||
|
case $target in
|
||||||
|
*-*-darwin*)
|
||||||
|
LIBUSB_1_0="-lusb-1.0 -framework CoreFoundation -framework IOKit"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
LIBUSB_1_0="-lusb-1.0"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
AC_DEFINE([HAVE_LIBUSB_1_0])
|
||||||
|
AC_CHECK_HEADERS([libusb-1.0/libusb.h])
|
||||||
|
AC_CHECK_HEADERS([libusb.h])
|
||||||
|
fi
|
||||||
|
AH_TEMPLATE([HAVE_LIBUSB_1_0],
|
||||||
|
[Define if USB support is enabled via a libusb-1.0 compatible libusb])
|
||||||
|
AC_CHECK_LIB([usb], [libusb_init], [have_libusb_1_0=yes])
|
||||||
|
if test x$have_libusb_1_0 = xyes; then
|
||||||
|
case $target in
|
||||||
|
*-*-freebsd*)
|
||||||
|
# FreeBSD 8+ has a native libusb-1.0 API compatible
|
||||||
|
# library offered by -lusb (which is also libusb-0.1
|
||||||
|
# compatible). FreeBSD <8 does not have a libusb-1.0
|
||||||
|
# at all so probing will fail but we do not have to
|
||||||
|
# special-case that.
|
||||||
|
LIBUSB_1_0="-lusb"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
LIBUSB_1_0="-lusb-1.0"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
AC_DEFINE([HAVE_LIBUSB_1_0])
|
||||||
|
AC_CHECK_HEADERS([libusb.h])
|
||||||
|
fi
|
||||||
|
AC_SUBST(LIBUSB_1_0, $LIBUSB_1_0)
|
||||||
|
AH_TEMPLATE([HAVE_LIBFTDI],
|
||||||
|
[Define if FTDI support is enabled via libftdi])
|
||||||
|
AC_CHECK_LIB([ftdi], [ftdi_init], [have_libftdi=yes], [], [-lusb])
|
||||||
|
if test x$have_libftdi = xyes; then
|
||||||
|
LIBFTDI="-lftdi -lusb"
|
||||||
|
AC_DEFINE([HAVE_LIBFTDI])
|
||||||
|
fi
|
||||||
|
AC_SUBST(LIBFTDI, $LIBFTDI)
|
||||||
|
# Checks for header files.
|
||||||
|
AC_CHECK_HEADERS([limits.h stdlib.h string.h])
|
||||||
|
AC_CHECK_HEADERS([fcntl.h sys/ioctl.h sys/time.h termios.h unistd.h])
|
||||||
|
AC_CHECK_HEADERS([ddk/hidsdi.h],,,[#include <windows.h>
|
||||||
|
#include <setupapi.h>])
|
||||||
|
|
||||||
|
|
||||||
|
# Checks for typedefs, structures, and compiler characteristics.
|
||||||
|
AC_C_CONST
|
||||||
|
AC_HEADER_TIME
|
||||||
|
|
||||||
|
# Checks for library functions.
|
||||||
|
AC_CHECK_FUNCS([memset select strcasecmp strdup strerror strncasecmp strtol strtoul gettimeofday usleep])
|
||||||
|
|
||||||
|
AC_MSG_CHECKING([for a Win32 HID libray])
|
||||||
|
SAVED_LIBS="${LIBS}"
|
||||||
|
case $target in
|
||||||
|
*-*-mingw32* | *-*-cygwin* | *-*-windows*)
|
||||||
|
LIBHID="-lhid -lsetupapi"
|
||||||
|
if test $ac_cv_header_ddk_hidsdi_h = yes
|
||||||
|
then
|
||||||
|
HIDINCLUDE="#include <ddk/hidsdi.h>"
|
||||||
|
else
|
||||||
|
HIDINCLUDE="#include \"my_ddk_hidsdi.h\""
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
LIBHID=""
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
LIBS="${LIBS} ${LIBHID}"
|
||||||
|
|
||||||
|
AH_TEMPLATE([HAVE_LIBHID],
|
||||||
|
[Define if HID support is enabled via the Win32 DDK])
|
||||||
|
AC_TRY_RUN([#include <windows.h>
|
||||||
|
#include <setupapi.h>
|
||||||
|
$HIDINCLUDE
|
||||||
|
|
||||||
|
int
|
||||||
|
main(void)
|
||||||
|
{
|
||||||
|
GUID hidGuid;
|
||||||
|
HidD_GetHidGuid(&hidGuid);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
], [have_libhid=yes], [have_libhid=no], [have_libhid=no])
|
||||||
|
AC_MSG_RESULT([$have_libhid])
|
||||||
|
if test x$have_libhid = xyes; then
|
||||||
|
AC_DEFINE([HAVE_LIBHID])
|
||||||
|
else
|
||||||
|
LIBHID=""
|
||||||
|
fi
|
||||||
|
LIBS="${SAVED_LIBS}"
|
||||||
|
AC_SUBST(LIBHID, $LIBHID)
|
||||||
|
|
||||||
|
# Check for types
|
||||||
|
|
||||||
|
# Solaris has uint_t and ulong_t typedefs in <sys/types.h>, avoid
|
||||||
|
# the redeclaration in usbtiny.c.
|
||||||
|
AC_CHECK_TYPES([uint_t], [], [], [#include <sys/types.h>])
|
||||||
|
AC_CHECK_TYPES([ulong_t], [], [], [#include <sys/types.h>])
|
||||||
|
|
||||||
|
# Checks for misc stuff.
|
||||||
|
|
||||||
|
AC_ARG_ENABLE(
|
||||||
|
[versioned-doc],
|
||||||
|
AC_HELP_STRING(
|
||||||
|
[--enable-versioned-doc],
|
||||||
|
[install docs in directory with version name (default)]),
|
||||||
|
[case "${enableval}" in
|
||||||
|
yes) versioned_doc=yes ;;
|
||||||
|
no) versioned_doc=no ;;
|
||||||
|
*) AC_MSG_ERROR(bad value ${enableval} for versioned-doc option) ;;
|
||||||
|
esac],
|
||||||
|
[versioned_doc=yes])
|
||||||
|
|
||||||
|
if test "$versioned_doc" = "yes"; then
|
||||||
|
DOC_INST_DIR='$(DESTDIR)$(datadir)/doc/avrdude-$(VERSION)'
|
||||||
|
else
|
||||||
|
DOC_INST_DIR='$(DESTDIR)$(datadir)/doc/avrdude'
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_ARG_ENABLE(
|
||||||
|
[doc],
|
||||||
|
AC_HELP_STRING(
|
||||||
|
[--enable-doc],
|
||||||
|
[Enable building documents]),
|
||||||
|
[case "${enableval}" in
|
||||||
|
yes) enabled_doc=yes ;;
|
||||||
|
no) enabled_doc=no ;;
|
||||||
|
*) AC_MSG_ERROR(bad value ${enableval} for disable-doc option) ;;
|
||||||
|
esac],
|
||||||
|
[enabled_doc=no])
|
||||||
|
|
||||||
|
AC_ARG_ENABLE(
|
||||||
|
[parport],
|
||||||
|
AC_HELP_STRING(
|
||||||
|
[--enable-parport],
|
||||||
|
[Enable accessing parallel ports(default)]),
|
||||||
|
[case "${enableval}" in
|
||||||
|
yes) enabled_parport=yes ;;
|
||||||
|
no) enabled_parport=no ;;
|
||||||
|
*) AC_MSG_ERROR(bad value ${enableval} for enable-parport option) ;;
|
||||||
|
esac],
|
||||||
|
[enabled_parport=yes])
|
||||||
|
|
||||||
|
DIST_SUBDIRS_AC='doc windows'
|
||||||
|
|
||||||
|
if test "$enabled_doc" = "yes"; then
|
||||||
|
SUBDIRS_AC='doc'
|
||||||
|
else
|
||||||
|
SUBDIRS_AC=''
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_SUBST(DOC_INST_DIR, $DOC_INST_DIR)
|
||||||
|
AC_SUBST(SUBDIRS_AC, $SUBDIRS_AC)
|
||||||
|
AC_SUBST(DIST_SUBDIRS_AC, $DIST_SUBDIRS_AC)
|
||||||
|
|
||||||
|
|
||||||
|
# Find the parallel serial device files based on target system
|
||||||
|
# If a system doesn't have a PC style parallel, mark it as unknown.
|
||||||
|
case $target in
|
||||||
|
i[[3456]]86-*-linux*|x86_64-*-linux*)
|
||||||
|
DEFAULT_PAR_PORT="/dev/parport0"
|
||||||
|
DEFAULT_SER_PORT="/dev/ttyS0"
|
||||||
|
;;
|
||||||
|
*-*-linux*)
|
||||||
|
DEFAULT_PAR_PORT="unknown"
|
||||||
|
DEFAULT_SER_PORT="/dev/ttyS0"
|
||||||
|
;;
|
||||||
|
i[[3456]]86-*-*freebsd*|amd64-*-*freebsd*)
|
||||||
|
DEFAULT_PAR_PORT="/dev/ppi0"
|
||||||
|
DEFAULT_SER_PORT="/dev/cuad0"
|
||||||
|
;;
|
||||||
|
*-*-*freebsd*)
|
||||||
|
DEFAULT_PAR_PORT="unknown"
|
||||||
|
DEFAULT_SER_PORT="/dev/cuad0"
|
||||||
|
;;
|
||||||
|
*-*-solaris*)
|
||||||
|
DEFAULT_PAR_PORT="/dev/printers/0"
|
||||||
|
DEFAULT_SER_PORT="/dev/term/a"
|
||||||
|
;;
|
||||||
|
*-*-msdos* | *-*-mingw32* | *-*-cygwin* | *-*-windows*)
|
||||||
|
DEFAULT_PAR_PORT="lpt1"
|
||||||
|
DEFAULT_SER_PORT="com1"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
DEFAULT_PAR_PORT="unknown"
|
||||||
|
DEFAULT_SER_PORT="unknown"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
if test "$enabled_parport" = "yes"; then
|
||||||
|
AC_MSG_CHECKING([for parallel device])
|
||||||
|
if test "$DEFAULT_PAR_PORT" = "unknown"; then
|
||||||
|
AC_MSG_NOTICE([parallel port access disabled for this system])
|
||||||
|
enabled_parport=no
|
||||||
|
else
|
||||||
|
AC_MSG_RESULT([$DEFAULT_PAR_PORT])
|
||||||
|
fi
|
||||||
|
AC_SUBST(DEFAULT_PAR_PORT, $DEFAULT_PAR_PORT)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_MSG_CHECKING([for serial device])
|
||||||
|
AC_MSG_RESULT([$DEFAULT_SER_PORT])
|
||||||
|
AC_SUBST(DEFAULT_SER_PORT, $DEFAULT_SER_PORT)
|
||||||
|
|
||||||
|
if test "$enabled_parport" = "yes"; then
|
||||||
|
AC_DEFINE(HAVE_PARPORT, 1, [parallel port access enabled])
|
||||||
|
confsubst="-e /^@HAVE_PARPORT_/d"
|
||||||
|
else
|
||||||
|
confsubst="-e /^@HAVE_PARPORT_BEGIN@/,/^@HAVE_PARPORT_END@/d"
|
||||||
|
fi
|
||||||
|
export confsubst
|
||||||
|
|
||||||
|
# If we are compiling with gcc, enable all warning and make warnings errors.
|
||||||
|
if test "$GCC" = yes; then
|
||||||
|
ENABLE_WARNINGS="-Wall"
|
||||||
|
|
||||||
|
# does this compiler support -Wno-pointer-sign ?
|
||||||
|
AC_MSG_CHECKING([if gcc accepts -Wno-pointer-sign ])
|
||||||
|
|
||||||
|
safe_CFLAGS=$CFLAGS
|
||||||
|
CFLAGS="$ENABLE_WARNINGS -Wno-pointer-sign"
|
||||||
|
|
||||||
|
AC_TRY_COMPILE(, [ int main () { return 0 ; } ], [
|
||||||
|
no_pointer_sign=yes
|
||||||
|
AC_MSG_RESULT([yes])
|
||||||
|
], [
|
||||||
|
no_pointer_sign=no
|
||||||
|
AC_MSG_RESULT([no])
|
||||||
|
])
|
||||||
|
CFLAGS=$safe_CFLAGS
|
||||||
|
|
||||||
|
if test x$no_pointer_sign = xyes; then
|
||||||
|
ENABLE_WARNINGS="$ENABLE_WARNINGS -Wno-pointer-sign"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
AC_SUBST(ENABLE_WARNINGS,$ENABLE_WARNINGS)
|
||||||
|
|
||||||
|
# See if we need to drop into the windows subdir.
|
||||||
|
case $target in
|
||||||
|
*-*-mingw32* | *-*-cygwin* | *-*-windows*)
|
||||||
|
if test "$GCC" = yes -a \( "$CC" = "cc" -o "$CC" = "gcc" \); then
|
||||||
|
# does this compiler support -mno-cygwin?
|
||||||
|
AC_MSG_CHECKING([if $CC accepts -mno-cygwin])
|
||||||
|
|
||||||
|
safe_CFLAGS="$CFLAGS"
|
||||||
|
CFLAGS="$ENABLE_WARNINGS -mno-cygwin"
|
||||||
|
|
||||||
|
AC_TRY_COMPILE(, [ int main () { return 0 ; } ], [
|
||||||
|
no_cygwin=yes
|
||||||
|
AC_MSG_RESULT([yes])
|
||||||
|
], [
|
||||||
|
no_cygwin=no
|
||||||
|
AC_MSG_RESULT([no])
|
||||||
|
])
|
||||||
|
CFLAGS="$safe_CFLAGS"
|
||||||
|
|
||||||
|
if test x$no_cygwin = xyes; then
|
||||||
|
CFLAGS="${CFLAGS} -mno-cygwin"
|
||||||
|
else
|
||||||
|
AC_MSG_NOTICE([Your compiler does not understand the -mno-cygwin option.])
|
||||||
|
AC_MSG_NOTICE([You might want to select an alternative compiler, like])
|
||||||
|
AC_MSG_NOTICE([])
|
||||||
|
AC_MSG_NOTICE([CC=mingw32-gcc ./configure])
|
||||||
|
AC_MSG_NOTICE([])
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_MSG_CHECKING([if linker accepts -static])
|
||||||
|
|
||||||
|
safe_LDFLAGS="$LDFLAGS"
|
||||||
|
LDFLAGS="${LDFLAGS} -static"
|
||||||
|
AC_TRY_LINK(, [ int main () { return 0 ; } ], [
|
||||||
|
can_link_static=yes
|
||||||
|
AC_MSG_RESULT([yes])
|
||||||
|
], [
|
||||||
|
can_link_static_cygwin=no
|
||||||
|
AC_MSG_RESULT([no])
|
||||||
|
])
|
||||||
|
LDFLAGS="$safe_LDFLAGS"
|
||||||
|
|
||||||
|
if test x$can_link_static = xyes; then
|
||||||
|
LDFLAGS="${LDFLAGS} -static"
|
||||||
|
fi
|
||||||
|
|
||||||
|
WINDOWS_DIRS="windows"
|
||||||
|
CFLAGS="${CFLAGS} -DWIN32NATIVE"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
AC_SUBST(WINDOWS_DIRS,$WINDOWS_DIRS)
|
||||||
|
|
||||||
|
AC_CONFIG_FILES([
|
||||||
|
doc/Makefile
|
||||||
|
windows/Makefile
|
||||||
|
avrdude.spec
|
||||||
|
Makefile
|
||||||
|
])
|
||||||
|
|
||||||
|
# The procedure to create avrdude.conf involves two steps. First,
|
||||||
|
# normal autoconf substitution will be applied, resulting in
|
||||||
|
# avrdude.conf.tmp. Finally, a sed command will be applied to filter
|
||||||
|
# out unwanted parts (currently the parallel port programmer types)
|
||||||
|
# based on previous configuration results, thereby producing the final
|
||||||
|
# avrdude.conf file.
|
||||||
|
|
||||||
|
AC_CONFIG_FILES([avrdude.conf.tmp:avrdude.conf.in],
|
||||||
|
[sed $confsubst avrdude.conf.tmp > avrdude.conf])
|
||||||
|
|
||||||
|
AC_OUTPUT
|
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Eric B. Weddington <eric@ecentral.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
#include <limits.h>
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
|
||||||
|
static char *filename;
|
||||||
|
|
||||||
|
|
||||||
|
void win_sys_config_set(char sys_config[PATH_MAX])
|
||||||
|
{
|
||||||
|
sys_config[0] = 0;
|
||||||
|
|
||||||
|
/* Use Windows API call to search for the Windows default system config file.*/
|
||||||
|
SearchPath(NULL, "avrdude.conf", NULL, PATH_MAX, sys_config, &filename);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void win_usr_config_set(char usr_config[PATH_MAX])
|
||||||
|
{
|
||||||
|
usr_config[0] = 0;
|
||||||
|
|
||||||
|
/* Use Windows API call to search for the Windows default user config file. */
|
||||||
|
SearchPath(NULL, "avrdude.rc", NULL, PATH_MAX, usr_config, &filename);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Eric B. Weddington <eric@ecentral.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
#ifndef confwin_h
|
||||||
|
#define confwin_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void win_sys_config_set(char sys_config[PATH_MAX]);
|
||||||
|
void win_usr_config_set(char usr_config[PATH_MAX]);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,83 @@
|
||||||
|
/*
|
||||||
|
* Derived from CRC algorithm for JTAG ICE mkII, published in Atmel
|
||||||
|
* Appnote AVR067. Converted from C++ to C.
|
||||||
|
*/
|
||||||
|
#include "crc16.h"
|
||||||
|
|
||||||
|
/* CRC16 Definitions */
|
||||||
|
static const unsigned short crc_table[256] = {
|
||||||
|
0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
|
||||||
|
0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,
|
||||||
|
0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,
|
||||||
|
0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,
|
||||||
|
0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,
|
||||||
|
0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,
|
||||||
|
0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,
|
||||||
|
0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,
|
||||||
|
0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,
|
||||||
|
0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,
|
||||||
|
0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,
|
||||||
|
0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,
|
||||||
|
0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,
|
||||||
|
0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,
|
||||||
|
0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,
|
||||||
|
0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,
|
||||||
|
0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,
|
||||||
|
0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,
|
||||||
|
0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,
|
||||||
|
0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,
|
||||||
|
0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,
|
||||||
|
0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,
|
||||||
|
0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,
|
||||||
|
0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,
|
||||||
|
0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,
|
||||||
|
0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,
|
||||||
|
0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,
|
||||||
|
0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,
|
||||||
|
0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,
|
||||||
|
0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,
|
||||||
|
0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,
|
||||||
|
0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78
|
||||||
|
};
|
||||||
|
|
||||||
|
/* CRC calculation macros */
|
||||||
|
#define CRC_INIT 0xFFFF
|
||||||
|
#define CRC(crcval,newchar) crcval = (crcval >> 8) ^ \
|
||||||
|
crc_table[(crcval ^ newchar) & 0x00ff]
|
||||||
|
|
||||||
|
unsigned short
|
||||||
|
crcsum(const unsigned char* message, unsigned long length,
|
||||||
|
unsigned short crc)
|
||||||
|
{
|
||||||
|
unsigned long i;
|
||||||
|
|
||||||
|
for(i = 0; i < length; i++)
|
||||||
|
{
|
||||||
|
CRC(crc, message[i]);
|
||||||
|
}
|
||||||
|
return crc;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
crcverify(const unsigned char* message, unsigned long length)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Returns true if the last two bytes in a message is the crc of the
|
||||||
|
* preceding bytes.
|
||||||
|
*/
|
||||||
|
unsigned short expected;
|
||||||
|
|
||||||
|
expected = crcsum(message, length - 2, CRC_INIT);
|
||||||
|
return (expected & 0xff) == message[length - 2] &&
|
||||||
|
((expected >> 8) & 0xff) == message[length - 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
crcappend(unsigned char* message, unsigned long length)
|
||||||
|
{
|
||||||
|
unsigned long crc;
|
||||||
|
|
||||||
|
crc = crcsum(message, length, CRC_INIT);
|
||||||
|
message[length] = (unsigned char)(crc & 0xff);
|
||||||
|
message[length+1] = (unsigned char)((crc >> 8) & 0xff);
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
#ifndef CRC16_H
|
||||||
|
#define CRC16_H
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Derived from CRC algorithm for JTAG ICE mkII, published in Atmel
|
||||||
|
* Appnote AVR067. Converted from C++ to C.
|
||||||
|
*/
|
||||||
|
|
||||||
|
extern unsigned short crcsum(const unsigned char* message,
|
||||||
|
unsigned long length,
|
||||||
|
unsigned short crc);
|
||||||
|
/*
|
||||||
|
* Verify that the last two bytes is a (LSB first) valid CRC of the
|
||||||
|
* message.
|
||||||
|
*/
|
||||||
|
extern int crcverify(const unsigned char* message,
|
||||||
|
unsigned long length);
|
||||||
|
/*
|
||||||
|
* Append a two byte CRC (LSB first) to message. length is size of
|
||||||
|
* message excluding crc. Space for the CRC bytes must be allocated
|
||||||
|
* in advance!
|
||||||
|
*/
|
||||||
|
extern void crcappend(unsigned char* message,
|
||||||
|
unsigned long length);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,22 @@
|
||||||
|
.cvsignore
|
||||||
|
Makefile
|
||||||
|
Makefile.in
|
||||||
|
avrdude-html
|
||||||
|
avrdude.aux
|
||||||
|
avrdude.cp
|
||||||
|
avrdude.cps
|
||||||
|
avrdude.dvi
|
||||||
|
avrdude.fn
|
||||||
|
avrdude.info
|
||||||
|
avrdude.ky
|
||||||
|
avrdude.log
|
||||||
|
avrdude.pdf
|
||||||
|
avrdude.pg
|
||||||
|
avrdude.ps
|
||||||
|
avrdude.toc
|
||||||
|
avrdude.tp
|
||||||
|
avrdude.vr
|
||||||
|
mdate-sh
|
||||||
|
stamp-vti
|
||||||
|
texinfo.tex
|
||||||
|
version.texi
|
|
@ -0,0 +1,60 @@
|
||||||
|
#
|
||||||
|
# avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
# Copyright (C) 2003 Theodore A. Roth <troth@openavr.org>
|
||||||
|
#
|
||||||
|
# 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# $Id$
|
||||||
|
#
|
||||||
|
|
||||||
|
CLEANFILES = \
|
||||||
|
version.texi \
|
||||||
|
stamp-vti
|
||||||
|
|
||||||
|
info_TEXINFOS = avrdude.texi
|
||||||
|
|
||||||
|
all-local: info html ps pdf
|
||||||
|
|
||||||
|
html: avrdude-html/avrdude.html
|
||||||
|
|
||||||
|
avrdude-html/avrdude.html: $(srcdir)/$(info_TEXINFOS)
|
||||||
|
texi2html -split_node $(srcdir)/$(info_TEXINFOS)
|
||||||
|
if [ -e ./avrdude.html -o -e ./avrdude_1.html ]; then \
|
||||||
|
mkdir -p avrdude-html ; \
|
||||||
|
mv -f *.html avrdude-html ; \
|
||||||
|
else \
|
||||||
|
mv -f avrdude avrdude-html; \
|
||||||
|
fi;
|
||||||
|
|
||||||
|
clean-local:
|
||||||
|
rm -rf avrdude-html *.info
|
||||||
|
|
||||||
|
install-data-local: install-docs
|
||||||
|
|
||||||
|
install-docs: html ps pdf
|
||||||
|
$(mkinstalldirs) $(DOC_INST_DIR)
|
||||||
|
$(INSTALL_DATA) avrdude.ps $(DOC_INST_DIR)/avrdude.ps
|
||||||
|
$(INSTALL_DATA) avrdude.pdf $(DOC_INST_DIR)/avrdude.pdf
|
||||||
|
$(mkinstalldirs) $(DOC_INST_DIR)/avrdude-html
|
||||||
|
@list=`echo avrdude-html/*.html`; \
|
||||||
|
for file in $$list; \
|
||||||
|
do \
|
||||||
|
$(INSTALL_DATA) $$file $(DOC_INST_DIR)/$$file; \
|
||||||
|
done
|
||||||
|
|
||||||
|
uninstall-local:
|
||||||
|
rm -rf $(DOC_INST_DIR)
|
|
@ -0,0 +1,26 @@
|
||||||
|
|
||||||
|
- Man page needs updated for avr910 info.
|
||||||
|
|
||||||
|
- Website needs to link to docs:
|
||||||
|
http://savannah.nongnu.org/download/avrdude/doc/avrdude-html/
|
||||||
|
|
||||||
|
- Add "skip empty pages" optimization on avr910 paged write. The stk500 has
|
||||||
|
this optimization already.
|
||||||
|
|
||||||
|
- Fix "overfull \hbox" issues in building documentation.
|
||||||
|
|
||||||
|
- FIXME: term.c: terminal_get_input(): strip newlines in non-readline input
|
||||||
|
code.
|
||||||
|
|
||||||
|
- FIXME: avr910.c: avr910_cmd(): Insert version check here.
|
||||||
|
|
||||||
|
- FIXME: ser_posix.c: serial_close(): Should really restore the terminal to
|
||||||
|
original state here.
|
||||||
|
|
||||||
|
- FIXME: main.c, par.c: exitspecs don't work if RESET-pin is controlled over
|
||||||
|
PPICTRL.
|
||||||
|
|
||||||
|
- transfer ppi-speedtuning to the windows version (CAVEAT: This will make
|
||||||
|
programming too fast for chips with 500kHz clock)
|
||||||
|
|
||||||
|
- make SCK-period configurable for PPI-programmers
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,65 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef fileio_h
|
||||||
|
#define fileio_h
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
FMT_AUTO,
|
||||||
|
FMT_SREC,
|
||||||
|
FMT_IHEX,
|
||||||
|
FMT_RBIN,
|
||||||
|
FMT_IMM,
|
||||||
|
FMT_HEX,
|
||||||
|
FMT_DEC,
|
||||||
|
FMT_OCT,
|
||||||
|
FMT_BIN
|
||||||
|
} FILEFMT;
|
||||||
|
|
||||||
|
struct fioparms {
|
||||||
|
int op;
|
||||||
|
char * mode;
|
||||||
|
char * iodesc;
|
||||||
|
char * dir;
|
||||||
|
char * rw;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
FIO_READ,
|
||||||
|
FIO_WRITE
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
char * fmtstr(FILEFMT format);
|
||||||
|
|
||||||
|
int fileio_setparms(int op, struct fioparms * fp);
|
||||||
|
|
||||||
|
int fileio(int op, char * filename, FILEFMT format,
|
||||||
|
struct avrpart * p, char * memtype, int size);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2005 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef freebsd_ppi_h
|
||||||
|
#define freebsd_ppi_h
|
||||||
|
|
||||||
|
#include <dev/ppbus/ppi.h>
|
||||||
|
|
||||||
|
#define ppi_claim(fd) {}
|
||||||
|
|
||||||
|
#define ppi_release(fd) {}
|
||||||
|
|
||||||
|
#define DO_PPI_READ(fd, reg, valp) \
|
||||||
|
(void)ioctl(fd, \
|
||||||
|
(reg) == PPIDATA? PPIGDATA: ((reg) == PPICTRL? PPIGCTRL: PPIGSTATUS), \
|
||||||
|
valp)
|
||||||
|
#define DO_PPI_WRITE(fd, reg, valp) \
|
||||||
|
(void)ioctl(fd, \
|
||||||
|
(reg) == PPIDATA? PPISDATA: ((reg) == PPICTRL? PPISCTRL: PPISSTATUS), \
|
||||||
|
valp)
|
||||||
|
|
||||||
|
#endif /* freebsd_ppi_h */
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,36 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2002-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef jtagmkI_h
|
||||||
|
#define jtagmkI_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void jtagmkI_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,57 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2002-2004, 2006 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef jtagmkII_h
|
||||||
|
#define jtagmkII_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int jtagmkII_send(PROGRAMMER * pgm, unsigned char * data, size_t len);
|
||||||
|
int jtagmkII_recv(PROGRAMMER * pgm, unsigned char **msg);
|
||||||
|
void jtagmkII_close(PROGRAMMER * pgm);
|
||||||
|
int jtagmkII_getsync(PROGRAMMER * pgm, int mode);
|
||||||
|
int jtagmkII_getparm(PROGRAMMER * pgm, unsigned char parm,
|
||||||
|
unsigned char * value);
|
||||||
|
|
||||||
|
void jtagmkII_initpgm (PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_avr32_initpgm (PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_dw_initpgm (PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_pdi_initpgm (PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_dragon_initpgm (PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_dragon_dw_initpgm (PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_dragon_pdi_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* These functions are referenced from stk500v2.c for JTAG ICE mkII
|
||||||
|
* and AVR Dragon programmers running in one of the STK500v2
|
||||||
|
* modi.
|
||||||
|
*/
|
||||||
|
void jtagmkII_setup(PROGRAMMER * pgm);
|
||||||
|
void jtagmkII_teardown(PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -0,0 +1,363 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2005, 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* JTAG ICE mkII definitions
|
||||||
|
* Taken from Appnote AVR067
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !defined(JTAGMKII_PRIVATE_EXPORTED)
|
||||||
|
/*
|
||||||
|
* Communication with the JTAG ICE works in frames. The protocol
|
||||||
|
* somewhat resembles the STK500v2 protocol, yet it is sufficiently
|
||||||
|
* different to prevent a direct code reuse. :-(
|
||||||
|
*
|
||||||
|
* Frame format:
|
||||||
|
*
|
||||||
|
* +---------------------------------------------------------------+
|
||||||
|
* | 0 | 1 . 2 | 3 . 4 . 5 . 6 | 7 | ... | N-1 . N |
|
||||||
|
* | | | | | | |
|
||||||
|
* | start | LSB MSB | LSB ....... MSB | token | msg | LSB MSB |
|
||||||
|
* | 0x1B | sequence# | message size | 0x0E | | CRC16 |
|
||||||
|
* +---------------------------------------------------------------+
|
||||||
|
*
|
||||||
|
* Each request message will be returned by a response with a matching
|
||||||
|
* sequence #. Sequence # 0xffff is reserved for asynchronous event
|
||||||
|
* notifications that will be sent by the ICE without a request
|
||||||
|
* message (e.g. when the target hit a breakpoint).
|
||||||
|
*
|
||||||
|
* The message size excludes the framing overhead (10 bytes).
|
||||||
|
*
|
||||||
|
* The first byte of the message is always the request or response
|
||||||
|
* code, which is roughly classified as:
|
||||||
|
*
|
||||||
|
* . Messages (commands) use 0x00 through 0x3f. (The documentation
|
||||||
|
* claims that messages start at 0x01, but actually CMND_SIGN_OFF is
|
||||||
|
* 0x00.)
|
||||||
|
* . Internal commands use 0x40 through 0x7f (not documented).
|
||||||
|
* . Success responses use 0x80 through 0x9f.
|
||||||
|
* . Failure responses use 0xa0 through 0xbf.
|
||||||
|
* . Events use 0xe0 through 0xff.
|
||||||
|
*/
|
||||||
|
#define MESSAGE_START 0x1b
|
||||||
|
#define TOKEN 0x0e
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Max message size we are willing to accept. Prevents us from trying
|
||||||
|
* to allocate too much VM in case we received a nonsensical packet
|
||||||
|
* length. We have to allocate the buffer as soon as we've got the
|
||||||
|
* length information (and thus have to trust that information by that
|
||||||
|
* time at first), as the final CRC check can only be done once the
|
||||||
|
* entire packet came it.
|
||||||
|
*/
|
||||||
|
#define MAX_MESSAGE 100000
|
||||||
|
|
||||||
|
#endif /* JTAGMKII_PRIVATE_EXPORTED */
|
||||||
|
|
||||||
|
/* ICE command codes */
|
||||||
|
#define CMND_CHIP_ERASE 0x13
|
||||||
|
#define CMND_CLEAR_EVENTS 0x22
|
||||||
|
#define CMND_CLR_BREAK 0x1A
|
||||||
|
#define CMND_ENTER_PROGMODE 0x14
|
||||||
|
#define CMND_ERASEPAGE_SPM 0x0D
|
||||||
|
#define CMND_FORCED_STOP 0x0A
|
||||||
|
#define CMND_GET_BREAK 0x12
|
||||||
|
#define CMND_GET_PARAMETER 0x03
|
||||||
|
#define CMND_GET_SIGN_ON 0x01
|
||||||
|
#define CMND_GET_SYNC 0x0f
|
||||||
|
#define CMND_GO 0x08
|
||||||
|
#define CMND_ISP_PACKET 0x2F
|
||||||
|
#define CMND_LEAVE_PROGMODE 0x15
|
||||||
|
#define CMND_READ_MEMORY 0x05
|
||||||
|
#define CMND_READ_PC 0x07
|
||||||
|
#define CMND_RESET 0x0B
|
||||||
|
#define CMND_RESTORE_TARGET 0x23
|
||||||
|
#define CMND_RUN_TO_ADDR 0x1C
|
||||||
|
#define CMND_SELFTEST 0x10
|
||||||
|
#define CMND_SET_BREAK 0x11
|
||||||
|
#define CMND_SET_DEVICE_DESCRIPTOR 0x0C
|
||||||
|
#define CMND_SET_N_PARAMETERS 0x16
|
||||||
|
#define CMND_SET_PARAMETER 0x02
|
||||||
|
#define CMND_SIGN_OFF 0x00
|
||||||
|
#define CMND_SINGLE_STEP 0x09
|
||||||
|
#define CMND_SPI_CMD 0x1D
|
||||||
|
#define CMND_WRITE_MEMORY 0x04
|
||||||
|
#define CMND_WRITE_PC 0x06
|
||||||
|
#define CMND_XMEGA_ERASE 0x34
|
||||||
|
// AVR32 - DFH
|
||||||
|
#define CMND_GET_IR 0x24
|
||||||
|
#define CMND_GET_xxx 0x25
|
||||||
|
#define CMND_WRITE_SAB 0x28
|
||||||
|
#define CMND_READ_SAB 0x29
|
||||||
|
#define CMND_RESET_AVR 0x2B
|
||||||
|
#define CMND_READ_MEMORY32 0x2C
|
||||||
|
#define CMND_WRITE_MEMORY32 0x2D
|
||||||
|
|
||||||
|
|
||||||
|
/* ICE responses */
|
||||||
|
#define RSP_DEBUGWIRE_SYNC_FAILED 0xAC
|
||||||
|
#define RSP_FAILED 0xA0
|
||||||
|
#define RSP_GET_BREAK 0x83
|
||||||
|
#define RSP_ILLEGAL_BREAKPOINT 0xA8
|
||||||
|
#define RSP_ILLEGAL_COMMAND 0xAA
|
||||||
|
#define RSP_ILLEGAL_EMULATOR_MODE 0xA4
|
||||||
|
#define RSP_ILLEGAL_JTAG_ID 0xA9
|
||||||
|
#define RSP_ILLEGAL_MCU_STATE 0xA5
|
||||||
|
#define RSP_ILLEGAL_MEMORY_TYPE 0xA2
|
||||||
|
#define RSP_ILLEGAL_MEMORY_RANGE 0xA3
|
||||||
|
#define RSP_ILLEGAL_PARAMETER 0xA1
|
||||||
|
#define RSP_ILLEGAL_POWER_STATE 0xAD
|
||||||
|
#define RSP_ILLEGAL_VALUE 0xA6
|
||||||
|
#define RSP_MEMORY 0x82
|
||||||
|
#define RSP_NO_TARGET_POWER 0xAB
|
||||||
|
#define RSP_OK 0x80
|
||||||
|
#define RSP_PARAMETER 0x81
|
||||||
|
#define RSP_PC 0x84
|
||||||
|
#define RSP_SELFTEST 0x85
|
||||||
|
#define RSP_SET_N_PARAMETERS 0xA7
|
||||||
|
#define RSP_SIGN_ON 0x86
|
||||||
|
#define RSP_SPI_DATA 0x88
|
||||||
|
|
||||||
|
/* ICE events */
|
||||||
|
#define EVT_BREAK 0xE0
|
||||||
|
#define EVT_DEBUG 0xE6
|
||||||
|
#define EVT_ERROR_PHY_FORCE_BREAK_TIMEOUT 0xE2
|
||||||
|
#define EVT_ERROR_PHY_MAX_BIT_LENGTH_DIFF 0xED
|
||||||
|
#define EVT_ERROR_PHY_OPT_RECEIVE_TIMEOUT 0xF9
|
||||||
|
#define EVT_ERROR_PHY_OPT_RECEIVED_BREAK 0xFA
|
||||||
|
#define EVT_ERROR_PHY_RECEIVED_BREAK 0xF8
|
||||||
|
#define EVT_ERROR_PHY_RECEIVE_TIMEOUT 0xF7
|
||||||
|
#define EVT_ERROR_PHY_RELEASE_BREAK_TIMEOUT 0xE3
|
||||||
|
#define EVT_ERROR_PHY_SYNC_OUT_OF_RANGE 0xF5
|
||||||
|
#define EVT_ERROR_PHY_SYNC_TIMEOUT 0xF0
|
||||||
|
#define EVT_ERROR_PHY_SYNC_TIMEOUT_BAUD 0xF4
|
||||||
|
#define EVT_ERROR_PHY_SYNC_WAIT_TIMEOUT 0xF6
|
||||||
|
#define EVT_RESULT_PHY_NO_ACTIVITY 0xFB
|
||||||
|
#define EVT_EXT_RESET 0xE7
|
||||||
|
#define EVT_ICE_POWER_ERROR_STATE 0xEA
|
||||||
|
#define EVT_ICE_POWER_OK 0xEB
|
||||||
|
#define EVT_IDR_DIRTY 0xEC
|
||||||
|
#define EVT_NONE 0xEF
|
||||||
|
#define EVT_PDSB_BREAK 0xF2
|
||||||
|
#define EVT_PDSMB_BREAK 0xF3
|
||||||
|
#define EVT_PROGRAM_BREAK 0xF1
|
||||||
|
#define EVT_RUN 0xE1
|
||||||
|
#define EVT_TARGET_POWER_OFF 0xE5
|
||||||
|
#define EVT_TARGET_POWER_ON 0xE4
|
||||||
|
#define EVT_TARGET_SLEEP 0xE8
|
||||||
|
#define EVT_TARGET_WAKEUP 0xE9
|
||||||
|
|
||||||
|
/* memory types for CMND_{READ,WRITE}_MEMORY */
|
||||||
|
#define MTYPE_IO_SHADOW 0x30 /* cached IO registers? */
|
||||||
|
#define MTYPE_SRAM 0x20 /* target's SRAM or [ext.] IO registers */
|
||||||
|
#define MTYPE_EEPROM 0x22 /* EEPROM, what way? */
|
||||||
|
#define MTYPE_EVENT 0x60 /* ICE event memory */
|
||||||
|
#define MTYPE_SPM 0xA0 /* flash through LPM/SPM */
|
||||||
|
#define MTYPE_FLASH_PAGE 0xB0 /* flash in programming mode */
|
||||||
|
#define MTYPE_EEPROM_PAGE 0xB1 /* EEPROM in programming mode */
|
||||||
|
#define MTYPE_FUSE_BITS 0xB2 /* fuse bits in programming mode */
|
||||||
|
#define MTYPE_LOCK_BITS 0xB3 /* lock bits in programming mode */
|
||||||
|
#define MTYPE_SIGN_JTAG 0xB4 /* signature in programming mode */
|
||||||
|
#define MTYPE_OSCCAL_BYTE 0xB5 /* osccal cells in programming mode */
|
||||||
|
#define MTYPE_CAN 0xB6 /* CAN mailbox */
|
||||||
|
#define MTYPE_FLASH 0xc0 /* xmega flash - undocumented in AVR067 */
|
||||||
|
#define MTYPE_USERSIG 0xc5 /* xmega user signature - undocumented in AVR067 */
|
||||||
|
#define MTYPE_PRODSIG 0xc6 /* xmega production signature - undocumented in AVR067 */
|
||||||
|
|
||||||
|
/* (some) ICE parameters, for CMND_{GET,SET}_PARAMETER */
|
||||||
|
#define PAR_HW_VERSION 0x01
|
||||||
|
#define PAR_FW_VERSION 0x02
|
||||||
|
#define PAR_EMULATOR_MODE 0x03
|
||||||
|
# define EMULATOR_MODE_DEBUGWIRE 0x00
|
||||||
|
# define EMULATOR_MODE_JTAG 0x01
|
||||||
|
# define EMULATOR_MODE_HV 0x02 /* HVSP or PP mode of AVR Dragon */
|
||||||
|
# define EMULATOR_MODE_SPI 0x03
|
||||||
|
# define EMULATOR_MODE_JTAG_AVR32 0x04
|
||||||
|
# define EMULATOR_MODE_JTAG_XMEGA 0x05
|
||||||
|
# define EMULATOR_MODE_PDI 0x06
|
||||||
|
#define PAR_IREG 0x04
|
||||||
|
#define PAR_BAUD_RATE 0x05
|
||||||
|
# define PAR_BAUD_2400 0x01
|
||||||
|
# define PAR_BAUD_4800 0x02
|
||||||
|
# define PAR_BAUD_9600 0x03
|
||||||
|
# define PAR_BAUD_19200 0x04 /* default */
|
||||||
|
# define PAR_BAUD_38400 0x05
|
||||||
|
# define PAR_BAUD_57600 0x06
|
||||||
|
# define PAR_BAUD_115200 0x07
|
||||||
|
# define PAR_BAUD_14400 0x08
|
||||||
|
#define PAR_OCD_VTARGET 0x06
|
||||||
|
#define PAR_OCD_JTAG_CLK 0x07
|
||||||
|
#define PAR_OCD_BREAK_CAUSE 0x08
|
||||||
|
#define PAR_TIMERS_RUNNING 0x09
|
||||||
|
#define PAR_BREAK_ON_CHANGE_FLOW 0x0A
|
||||||
|
#define PAR_BREAK_ADDR1 0x0B
|
||||||
|
#define PAR_BREAK_ADDR2 0x0C
|
||||||
|
#define PAR_COMBBREAKCTRL 0x0D
|
||||||
|
#define PAR_JTAGID 0x0E
|
||||||
|
#define PAR_UNITS_BEFORE 0x0F
|
||||||
|
#define PAR_UNITS_AFTER 0x10
|
||||||
|
#define PAR_BIT_BEFORE 0x11
|
||||||
|
#define PAR_BIT_ATER 0x12
|
||||||
|
#define PAR_EXTERNAL_RESET 0x13
|
||||||
|
#define PAR_FLASH_PAGE_SIZE 0x14
|
||||||
|
#define PAR_EEPROM_PAGE_SIZE 0x15
|
||||||
|
#define PAR_UNUSED1 0x16
|
||||||
|
#define PAR_PSB0 0x17
|
||||||
|
#define PAR_PSB1 0x18
|
||||||
|
#define PAR_PROTOCOL_DEBUG_EVENT 0x19
|
||||||
|
#define PAR_MCU_STATE 0x1A
|
||||||
|
# define STOPPED 0x00
|
||||||
|
# define RUNNING 0x01
|
||||||
|
# define PROGRAMMING 0x02
|
||||||
|
#define PAR_DAISY_CHAIN_INFO 0x1B
|
||||||
|
#define PAR_BOOT_ADDRESS 0x1C
|
||||||
|
#define PAR_TARGET_SIGNATURE 0x1D
|
||||||
|
#define PAR_DEBUGWIRE_BAUDRATE 0x1E
|
||||||
|
#define PAR_PROGRAM_ENTRY_POINT 0x1F
|
||||||
|
#define PAR_PDI_OFFSET_START 0x32
|
||||||
|
#define PAR_PDI_OFFSET_END 0x33
|
||||||
|
#define PAR_PACKET_PARSING_ERRORS 0x40
|
||||||
|
#define PAR_VALID_PACKETS_RECEIVED 0x41
|
||||||
|
#define PAR_INTERCOMMUNICATION_TX_FAILURES 0x42
|
||||||
|
#define PAR_INTERCOMMUNICATION_RX_FAILURES 0x43
|
||||||
|
#define PAR_CRC_ERRORS 0x44
|
||||||
|
#define PAR_POWER_SOURCE 0x45
|
||||||
|
# define POWER_EXTERNAL 0x00
|
||||||
|
# define POWER_USB 0x01
|
||||||
|
#define PAR_CAN_FLAG 0x22
|
||||||
|
# define DONT_READ_CAN_MAILBOX 0x00
|
||||||
|
# define READ_CAN_MAILBOX 0x01
|
||||||
|
#define PAR_ENABLE_IDR_IN_RUN_MODE 0x23
|
||||||
|
# define ACCESS_OSCCAL 0x00
|
||||||
|
# define ACCESS_IDR 0x01
|
||||||
|
#define PAR_ALLOW_PAGEPROGRAMMING_IN_SCANCHAIN 0x24
|
||||||
|
# define PAGEPROG_NOT_ALLOWED 0x00
|
||||||
|
# define PAGEPROG_ALLOWED 0x01
|
||||||
|
|
||||||
|
/* Xmega erase memory types, for CMND_XMEGA_ERASE */
|
||||||
|
#define XMEGA_ERASE_CHIP 0x00
|
||||||
|
#define XMEGA_ERASE_APP 0x01
|
||||||
|
#define XMEGA_ERASE_BOOT 0x02
|
||||||
|
#define XMEGA_ERASE_EEPROM 0x03
|
||||||
|
#define XMEGA_ERASE_APP_PAGE 0x04
|
||||||
|
#define XMEGA_ERASE_BOOT_PAGE 0x05
|
||||||
|
#define XMEGA_ERASE_EEPROM_PAGE 0x06
|
||||||
|
#define XMEGA_ERASE_USERSIG 0x07
|
||||||
|
|
||||||
|
/* AVR32 related definitions */
|
||||||
|
#define AVR32_FLASHC_FCR 0xFFFE1400
|
||||||
|
#define AVR32_FLASHC_FCMD 0xFFFE1404
|
||||||
|
#define AVR32_FLASHC_FCMD_KEY 0xA5000000
|
||||||
|
#define AVR32_FLASHC_FCMD_WRITE_PAGE 1
|
||||||
|
#define AVR32_FLASHC_FCMD_ERASE_PAGE 2
|
||||||
|
#define AVR32_FLASHC_FCMD_CLEAR_PAGE_BUFFER 3
|
||||||
|
#define AVR32_FLASHC_FCMD_LOCK 4
|
||||||
|
#define AVR32_FLASHC_FCMD_UNLOCK 5
|
||||||
|
#define AVR32_FLASHC_FSR 0xFFFE1408
|
||||||
|
#define AVR32_FLASHC_FSR_RDY 0x00000001
|
||||||
|
#define AVR32_FLASHC_FSR_ERR 0x00000008
|
||||||
|
#define AVR32_FLASHC_FGPFRHI 0xFFFE140C
|
||||||
|
#define AVR32_FLASHC_FGPFRLO 0xFFFE1410
|
||||||
|
|
||||||
|
#define AVR32_DC 0x00000008
|
||||||
|
#define AVR32_DS 0x00000010
|
||||||
|
#define AVR32_DINST 0x00000104
|
||||||
|
#define AVR32_DCCPU 0x00000110
|
||||||
|
#define AVR32_DCEMU 0x00000114
|
||||||
|
#define AVR32_DCSR 0x00000118
|
||||||
|
|
||||||
|
#define AVR32_DC_ABORT 0x80000000
|
||||||
|
#define AVR32_DC_RESET 0x40000000
|
||||||
|
#define AVR32_DC_DBE 0x00002000
|
||||||
|
#define AVR32_DC_DBR 0x00001000
|
||||||
|
|
||||||
|
#define AVR32_RESET_READ 0x0001
|
||||||
|
#define AVR32_RESET_WRITE 0x0002
|
||||||
|
#define AVR32_RESET_CHIP_ERASE 0x0004
|
||||||
|
#define AVR32_SET4RUNNING 0x0008
|
||||||
|
//#define AVR32_RESET_COMMON (AVR32_RESET_READ | AVR32_RESET_WRITE | AVR32_RESET_CHIP_ERASE )
|
||||||
|
|
||||||
|
|
||||||
|
#if !defined(JTAGMKII_PRIVATE_EXPORTED)
|
||||||
|
/*
|
||||||
|
* In appnote AVR067, struct device_descriptor is written with
|
||||||
|
* int/long field types. We cannot use them directly, as they were
|
||||||
|
* neither properly aligned for portability, nor did they care for
|
||||||
|
* endianess issues. We thus use arrays of unsigned chars, plus
|
||||||
|
* conversion macros.
|
||||||
|
*/
|
||||||
|
struct device_descriptor
|
||||||
|
{
|
||||||
|
unsigned char ucReadIO[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucReadIOShadow[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucWriteIO[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucWriteIOShadow[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucReadExtIO[52]; /*LSB = IOloc 96, MSB = IOloc511 */
|
||||||
|
unsigned char ucReadIOExtShadow[52]; /*LSB = IOloc 96, MSB = IOloc511 */
|
||||||
|
unsigned char ucWriteExtIO[52]; /*LSB = IOloc 96, MSB = IOloc511 */
|
||||||
|
unsigned char ucWriteIOExtShadow[52];/*LSB = IOloc 96, MSB = IOloc511 */
|
||||||
|
unsigned char ucIDRAddress; /*IDR address */
|
||||||
|
unsigned char ucSPMCRAddress; /*SPMCR Register address and dW BasePC */
|
||||||
|
unsigned char ucRAMPZAddress; /*RAMPZ Register address in SRAM I/O */
|
||||||
|
/*space */
|
||||||
|
unsigned char uiFlashPageSize[2]; /*Device Flash Page Size, Size = */
|
||||||
|
/*2 exp ucFlashPageSize */
|
||||||
|
unsigned char ucEepromPageSize; /*Device Eeprom Page Size in bytes */
|
||||||
|
unsigned char ulBootAddress[4]; /*Device Boot Loader Start Address */
|
||||||
|
unsigned char uiUpperExtIOLoc[2]; /*Topmost (last) extended I/O */
|
||||||
|
/*location, 0 if no external I/O */
|
||||||
|
unsigned char ulFlashSize[4]; /*Device Flash Size */
|
||||||
|
unsigned char ucEepromInst[20]; /*Instructions for W/R EEPROM */
|
||||||
|
unsigned char ucFlashInst[3]; /*Instructions for W/R FLASH */
|
||||||
|
unsigned char ucSPHaddr; /* stack pointer high */
|
||||||
|
unsigned char ucSPLaddr; /* stack pointer low */
|
||||||
|
/* new as of 16-02-2004 */
|
||||||
|
unsigned char uiFlashpages[2]; /* number of pages in flash */
|
||||||
|
unsigned char ucDWDRAddress; /* DWDR register address */
|
||||||
|
unsigned char ucDWBasePC; /* base/mask value of the PC */
|
||||||
|
/* new as of 30-04-2004 */
|
||||||
|
unsigned char ucAllowFullPageBitstream; /* FALSE on ALL new */
|
||||||
|
/*parts */
|
||||||
|
unsigned char uiStartSmallestBootLoaderSection[2]; /* */
|
||||||
|
/* new as of 18-10-2004 */
|
||||||
|
unsigned char EnablePageProgramming; /* For JTAG parts only, */
|
||||||
|
/* default TRUE */
|
||||||
|
unsigned char ucCacheType; /* CacheType_Normal 0x00, */
|
||||||
|
/* CacheType_CAN 0x01, */
|
||||||
|
/* CacheType_HEIMDALL 0x02 */
|
||||||
|
/* new as of 27-10-2004 */
|
||||||
|
unsigned char uiSramStartAddr[2]; /* Start of SRAM */
|
||||||
|
unsigned char ucResetType; /* Selects reset type. ResetNormal = 0x00 */
|
||||||
|
/* ResetAT76CXXX = 0x01 */
|
||||||
|
unsigned char ucPCMaskExtended; /* For parts with extended PC */
|
||||||
|
unsigned char ucPCMaskHigh; /* PC high mask */
|
||||||
|
unsigned char ucEindAddress; /* Selects reset type. [EIND address...] */
|
||||||
|
/* new as of early 2005, firmware 4.x */
|
||||||
|
unsigned char EECRAddress[2]; /* EECR memory-mapped IO address */
|
||||||
|
};
|
||||||
|
#endif /* JTAGMKII_PRIVATE_EXPORTED */
|
||||||
|
|
||||||
|
/* return code from jtagmkII_getsync() to indicate a "graceful"
|
||||||
|
* failure, i.e. an attempt to enable ISP failed and should be
|
||||||
|
* eventually retried */
|
||||||
|
#define JTAGII_GETSYNC_FAIL_GRACEFUL (-2)
|
|
@ -0,0 +1,169 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2005 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* JTAG ICE mkI definitions
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* ICE command codes */
|
||||||
|
/* 0x20 Get Synch [Resp_OK] */
|
||||||
|
#define CMD_GET_SYNC ' '
|
||||||
|
|
||||||
|
/* 0x31 Single Step [Sync_CRC/EOP] [Resp_OK] */
|
||||||
|
/* 0x32 Read PC [Sync_CRC/EOP] [Resp_OK] [program counter]
|
||||||
|
* [Resp_OK] */
|
||||||
|
/* 0x33 Write PC [program counter] [Sync_CRC/EOP] [Resp_OK]
|
||||||
|
* [Resp_OK] */
|
||||||
|
/* 0xA2 Firmware Upgrade [upgrade string] [Sync_CRC/EOP] [Resp_OK]
|
||||||
|
* [Resp_OK] */
|
||||||
|
/* 0xA0 Set Device Descriptor [device info] [Sync_CRC/EOP] [Resp_OK]
|
||||||
|
* [Resp_OK] */
|
||||||
|
#define CMD_SET_DEVICE_DESCRIPTOR 0xA0
|
||||||
|
|
||||||
|
/* 0x42 Set Parameter [parameter] [setting] [Sync_CRC/EOP] [Resp_OK]
|
||||||
|
* [Resp_OK] */
|
||||||
|
#define CMD_SET_PARAM 'B'
|
||||||
|
|
||||||
|
/* 0x46 Forced Stop [Sync_CRC/EOP] [Resp_OK] [checksum][program
|
||||||
|
* counter] [Resp_OK] */
|
||||||
|
#define CMD_STOP 'F'
|
||||||
|
|
||||||
|
/* 0x47 Go [Sync_CRC/EOP] [Resp_OK] */
|
||||||
|
#define CMD_GO 'G'
|
||||||
|
|
||||||
|
/* 0x52 Read Memory [memory type] [word count] [start address]
|
||||||
|
* [Sync_CRC/EOP] [Resp_OK] [word 0] ... [word n] [checksum]
|
||||||
|
* [Resp_OK] */
|
||||||
|
#define CMD_READ_MEM 'R'
|
||||||
|
|
||||||
|
/* 0x53 Get Sign On [Sync_CRC/EOP] [Resp_OK] ["AVRNOCD"] [Resp_OK] */
|
||||||
|
#define CMD_GET_SIGNON 'S'
|
||||||
|
|
||||||
|
/* 0XA1 Erase Page spm [address] [Sync_CRC/EOP] [Resp_OK] [Resp_OK] */
|
||||||
|
|
||||||
|
/* 0x57 Write Memory [memory type] [word count] [start address]
|
||||||
|
* [Sync_CRC/EOP] [Resp_OK] [Cmd_DATA] [word 0] ... [word n] */
|
||||||
|
#define CMD_WRITE_MEM 'W'
|
||||||
|
|
||||||
|
/* Second half of write memory: the data command. Undocumented. */
|
||||||
|
#define CMD_DATA 'h'
|
||||||
|
|
||||||
|
/* 0x64 Get Debug Info [Sync_CRC/EOP] [Resp_OK] [0x00] [Resp_OK] */
|
||||||
|
/* 0x71 Get Parameter [parameter] [Sync_CRC/EOP] [Resp_OK] [setting]
|
||||||
|
* [Resp_OK] */
|
||||||
|
#define CMD_GET_PARAM 'q'
|
||||||
|
|
||||||
|
/* 0x78 Reset [Sync_CRC/EOP] [Resp_OK] [Resp_OK] */
|
||||||
|
#define CMD_RESET 'x'
|
||||||
|
|
||||||
|
/* 0xA3 Enter Progmode [Sync_CRC/EOP] [Resp_OK] [Resp_OK] */
|
||||||
|
#define CMD_ENTER_PROGMODE 0xa3
|
||||||
|
|
||||||
|
/* 0xA4 Leave Progmode [Sync_CRC/EOP] [Resp_OK] [Resp_OK] */
|
||||||
|
#define CMD_LEAVE_PROGMODE 0xa4
|
||||||
|
|
||||||
|
/* 0xA5 Chip Erase [Sync_CRC/EOP] [Resp_OK] [Resp_OK] */
|
||||||
|
#define CMD_CHIP_ERASE 0xa5
|
||||||
|
|
||||||
|
|
||||||
|
/* ICE responses */
|
||||||
|
#define RESP_OK 'A'
|
||||||
|
#define RESP_BREAK 'B'
|
||||||
|
#define RESP_INFO 'G'
|
||||||
|
#define RESP_FAILED 'F'
|
||||||
|
#define RESP_SYNC_ERROR 'E'
|
||||||
|
#define RESP_SLEEP 'H'
|
||||||
|
#define RESP_POWER 'I'
|
||||||
|
|
||||||
|
#define PARM_BITRATE 'b'
|
||||||
|
#define PARM_SW_VERSION 0x7b
|
||||||
|
#define PARM_HW_VERSION 0x7a
|
||||||
|
#define PARM_IREG_HIGH 0x81
|
||||||
|
#define PARM_IREG_LOW 0x82
|
||||||
|
#define PARM_OCD_VTARGET 0x84
|
||||||
|
#define PARM_OCD_BREAK_CAUSE 0x85
|
||||||
|
#define PARM_CLOCK 0x86
|
||||||
|
#define PARM_EXTERNAL_RESET 0x8b
|
||||||
|
#define PARM_FLASH_PAGESIZE_LOW 0x88
|
||||||
|
#define PARM_FLASH_PAGESIZE_HIGH 0x89
|
||||||
|
#define PARM_EEPROM_PAGESIZE 0x8a
|
||||||
|
#define PARM_TIMERS_RUNNING 0xa0
|
||||||
|
#define PARM_BP_FLOW 0xa1
|
||||||
|
#define PARM_BP_X_HIGH 0xa2
|
||||||
|
#define PARM_BP_X_LOW 0xa3
|
||||||
|
#define PARM_BP_Y_HIGH 0xa4
|
||||||
|
#define PARM_BP_Y_LOW 0xa5
|
||||||
|
#define PARM_BP_MODE 0xa6
|
||||||
|
#define PARM_JTAGID_BYTE0 0xa7
|
||||||
|
#define PARM_JTAGID_BYTE1 0xa8
|
||||||
|
#define PARM_JTAGID_BYTE2 0xa9
|
||||||
|
#define PARM_JTAGID_BYTE3 0xaa
|
||||||
|
#define PARM_UNITS_BEFORE 0xab
|
||||||
|
#define PARM_UNITS_AFTER 0xac
|
||||||
|
#define PARM_BIT_BEFORE 0xad
|
||||||
|
#define PARM_BIT_AFTER 0xae
|
||||||
|
#define PARM_PSB0_LOW 0xaf
|
||||||
|
#define PARM_PSBO_HIGH 0xb0
|
||||||
|
#define PARM_PSB1_LOW 0xb1
|
||||||
|
#define PARM_PSB1_HIGH 0xb2
|
||||||
|
#define PARM_MCU_MODE 0xb3
|
||||||
|
|
||||||
|
#define JTAG_BITRATE_1_MHz 0xff
|
||||||
|
#define JTAG_BITRATE_500_kHz 0xfe
|
||||||
|
#define JTAG_BITRATE_250_kHz 0xfd
|
||||||
|
#define JTAG_BITRATE_125_kHz 0xfb
|
||||||
|
|
||||||
|
/* memory types for CMND_{READ,WRITE}_MEMORY */
|
||||||
|
#define MTYPE_IO_SHADOW 0x30 /* cached IO registers? */
|
||||||
|
#define MTYPE_SRAM 0x20 /* target's SRAM or [ext.] IO registers */
|
||||||
|
#define MTYPE_EEPROM 0x22 /* EEPROM, what way? */
|
||||||
|
#define MTYPE_EVENT 0x60 /* ICE event memory */
|
||||||
|
#define MTYPE_SPM 0xA0 /* flash through LPM/SPM */
|
||||||
|
#define MTYPE_FLASH_PAGE 0xB0 /* flash in programming mode */
|
||||||
|
#define MTYPE_EEPROM_PAGE 0xB1 /* EEPROM in programming mode */
|
||||||
|
#define MTYPE_FUSE_BITS 0xB2 /* fuse bits in programming mode */
|
||||||
|
#define MTYPE_LOCK_BITS 0xB3 /* lock bits in programming mode */
|
||||||
|
#define MTYPE_SIGN_JTAG 0xB4 /* signature in programming mode */
|
||||||
|
#define MTYPE_OSCCAL_BYTE 0xB5 /* osccal cells in programming mode */
|
||||||
|
|
||||||
|
struct device_descriptor
|
||||||
|
{
|
||||||
|
unsigned char ucReadIO[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucWriteIO[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucReadIOShadow[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucWriteIOShadow[8]; /*LSB = IOloc 0, MSB = IOloc63 */
|
||||||
|
unsigned char ucReadExtIO[20]; /*LSB = IOloc 96, MSB = IOloc255 */
|
||||||
|
unsigned char ucWriteExtIO[20]; /*LSB = IOloc 96, MSB = IOloc255 */
|
||||||
|
unsigned char ucReadIOExtShadow[20]; /*LSB = IOloc 96, MSB = IOloc255 */
|
||||||
|
unsigned char ucWriteIOExtShadow[20];/*LSB = IOloc 96, MSB = IOloc255 */
|
||||||
|
unsigned char ucIDRAddress; /*IDR address */
|
||||||
|
unsigned char ucSPMCRAddress; /*SPMCR Register address and dW BasePC */
|
||||||
|
unsigned char ucRAMPZAddress; /*RAMPZ Register address in SRAM I/O */
|
||||||
|
/*space */
|
||||||
|
unsigned char uiFlashPageSize[2]; /*Device Flash Page Size, Size = */
|
||||||
|
/*2 exp ucFlashPageSize */
|
||||||
|
unsigned char ucEepromPageSize; /*Device Eeprom Page Size in bytes */
|
||||||
|
unsigned char ulBootAddress[4]; /*Device Boot Loader Start Address */
|
||||||
|
unsigned char uiUpperExtIOLoc; /*Topmost (last) extended I/O */
|
||||||
|
/*location, 0 if no external I/O */
|
||||||
|
};
|
|
@ -0,0 +1,291 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
%{
|
||||||
|
/* need this for the call to atof() below */
|
||||||
|
#include <math.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <limits.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "config_gram.h"
|
||||||
|
#include "lists.h"
|
||||||
|
|
||||||
|
#define YY_NO_UNPUT
|
||||||
|
|
||||||
|
%}
|
||||||
|
|
||||||
|
DIGIT [0-9]
|
||||||
|
HEXDIGIT [0-9a-fA-F]
|
||||||
|
ID [_a-zA-Z][_a-zA-Z0-9]*
|
||||||
|
SIGN [+-]
|
||||||
|
|
||||||
|
%x strng
|
||||||
|
%x incl
|
||||||
|
%x comment
|
||||||
|
|
||||||
|
/* Bump resources for classic lex. */
|
||||||
|
%e2000
|
||||||
|
%p10000
|
||||||
|
%n1000
|
||||||
|
|
||||||
|
%%
|
||||||
|
|
||||||
|
{SIGN}*{DIGIT}+ { yylval = number(yytext); return TKN_NUMBER; }
|
||||||
|
{SIGN}*{DIGIT}+"."{DIGIT}* { yylval = number(yytext); return TKN_NUMBER; }
|
||||||
|
{SIGN}*"."{DIGIT}* { yylval = number(yytext); return TKN_NUMBER; }
|
||||||
|
|
||||||
|
"\"" { string_buf_ptr = string_buf; BEGIN(strng); }
|
||||||
|
|
||||||
|
0x{HEXDIGIT}+ { yylval = hexnumber(yytext); return TKN_NUMBER; }
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# { /* The following eats '#' style comments to end of line */
|
||||||
|
BEGIN(comment); }
|
||||||
|
<comment>[^\n] { /* eat comments */ }
|
||||||
|
<comment>\n { lineno++; BEGIN(INITIAL); }
|
||||||
|
|
||||||
|
|
||||||
|
"/*" { /* The following eats multiline C style comments */
|
||||||
|
int c;
|
||||||
|
int comment_start;
|
||||||
|
|
||||||
|
comment_start = lineno;
|
||||||
|
while (1) {
|
||||||
|
while (((c = input()) != '*') && (c != EOF)) {
|
||||||
|
/* eat up text of comment, but keep counting lines */
|
||||||
|
if (c == '\n')
|
||||||
|
lineno++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (c == '*') {
|
||||||
|
while ((c = input()) == '*')
|
||||||
|
;
|
||||||
|
if (c == '/')
|
||||||
|
break; /* found the end */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (c == EOF) {
|
||||||
|
fprintf(stderr, "error at %s:%d: EOF in comment\n",
|
||||||
|
infile, lineno);
|
||||||
|
fprintf(stderr, " comment started on line %d\n",
|
||||||
|
comment_start);
|
||||||
|
exit(1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
<strng>\" { *string_buf_ptr = 0; string_buf_ptr = string_buf;
|
||||||
|
yylval = string(string_buf_ptr); BEGIN(INITIAL); return TKN_STRING; }
|
||||||
|
<strng>\\n *string_buf_ptr++ = '\n';
|
||||||
|
<strng>\\t *string_buf_ptr++ = '\t';
|
||||||
|
<strng>\\r *string_buf_ptr++ = '\r';
|
||||||
|
<strng>\\b *string_buf_ptr++ = '\b';
|
||||||
|
<strng>\\f *string_buf_ptr++ = '\f';
|
||||||
|
<strng>\\(.|\n) *(string_buf_ptr++) = yytext[1];
|
||||||
|
<strng>[^\\\n\"]+ { char *yptr = yytext; while (*yptr)
|
||||||
|
*(string_buf_ptr++) = *(yptr++); }
|
||||||
|
<strng>\n { fprintf(stderr, "error at line %d: unterminated character constant\n",
|
||||||
|
lineno);
|
||||||
|
exit(1); }
|
||||||
|
|
||||||
|
allowfullpagebitstream { yylval=NULL; return K_ALLOWFULLPAGEBITSTREAM; }
|
||||||
|
arduino { yylval=NULL; return K_ARDUINO; }
|
||||||
|
avr910 { yylval=NULL; return K_AVR910; }
|
||||||
|
avr910_devcode { yylval=NULL; return K_AVR910_DEVCODE; }
|
||||||
|
avrftdi { yylval=NULL; return K_AVRFTDI; }
|
||||||
|
usbasp { yylval=NULL; return K_USBASP; }
|
||||||
|
usbtiny { yylval=NULL; return K_USBTINY; }
|
||||||
|
bank_size { yylval=NULL; return K_PAGE_SIZE; }
|
||||||
|
banked { yylval=NULL; return K_PAGED; }
|
||||||
|
baudrate { yylval=NULL; return K_BAUDRATE; }
|
||||||
|
bs2 { yylval=NULL; return K_BS2; }
|
||||||
|
buff { yylval=NULL; return K_BUFF; }
|
||||||
|
buspirate { yylval=NULL; return K_BUSPIRATE; }
|
||||||
|
butterfly { yylval=NULL; return K_BUTTERFLY; }
|
||||||
|
butterfly_mk { yylval=NULL; return K_BUTTERFLY_MK; }
|
||||||
|
chip_erase_delay { yylval=NULL; return K_CHIP_ERASE_DELAY; }
|
||||||
|
desc { yylval=NULL; return K_DESC; }
|
||||||
|
default_parallel { yylval=NULL; return K_DEFAULT_PARALLEL; }
|
||||||
|
default_programmer { yylval=NULL; return K_DEFAULT_PROGRAMMER; }
|
||||||
|
default_serial { yylval=NULL; return K_DEFAULT_SERIAL; }
|
||||||
|
default_bitclock { yylval=NULL; return K_DEFAULT_BITCLOCK; }
|
||||||
|
devicecode { yylval=NULL; return K_DEVICECODE; }
|
||||||
|
dragon_dw { yylval=NULL; return K_DRAGON_DW; }
|
||||||
|
dragon_hvsp { yylval=NULL; return K_DRAGON_HVSP; }
|
||||||
|
dragon_isp { yylval=NULL; return K_DRAGON_ISP; }
|
||||||
|
dragon_jtag { yylval=NULL; return K_DRAGON_JTAG; }
|
||||||
|
dragon_pdi { yylval=NULL; return K_DRAGON_PDI; }
|
||||||
|
dragon_pp { yylval=NULL; return K_DRAGON_PP; }
|
||||||
|
eecr { yylval=NULL; return K_EECR; }
|
||||||
|
eeprom { yylval=NULL; return K_EEPROM; }
|
||||||
|
enablepageprogramming { yylval=NULL; return K_ENABLEPAGEPROGRAMMING; }
|
||||||
|
errled { yylval=NULL; return K_ERRLED; }
|
||||||
|
flash { yylval=NULL; return K_FLASH; }
|
||||||
|
has_jtag { yylval=NULL; return K_HAS_JTAG; }
|
||||||
|
has_debugwire { yylval=NULL; return K_HAS_DW; }
|
||||||
|
has_pdi { yylval=NULL; return K_HAS_PDI; }
|
||||||
|
has_tpi { yylval=NULL; return K_HAS_TPI; }
|
||||||
|
id { yylval=NULL; return K_ID; }
|
||||||
|
idr { yylval=NULL; return K_IDR; }
|
||||||
|
is_avr32 { yylval=NULL; return K_IS_AVR32; }
|
||||||
|
jtagmki { yylval=NULL; return K_JTAG_MKI; }
|
||||||
|
jtagmkii { yylval=NULL; return K_JTAG_MKII; }
|
||||||
|
jtagmkii_avr32 { yylval=NULL; return K_JTAG_MKII_AVR32; }
|
||||||
|
jtagmkii_dw { yylval=NULL; return K_JTAG_MKII_DW; }
|
||||||
|
jtagmkii_isp { yylval=NULL; return K_JTAG_MKII_ISP; }
|
||||||
|
jtagmkii_pdi { yylval=NULL; return K_JTAG_MKII_PDI; }
|
||||||
|
max_write_delay { yylval=NULL; return K_MAX_WRITE_DELAY; }
|
||||||
|
memory { yylval=NULL; return K_MEMORY; }
|
||||||
|
min_write_delay { yylval=NULL; return K_MIN_WRITE_DELAY; }
|
||||||
|
miso { yylval=NULL; return K_MISO; }
|
||||||
|
mosi { yylval=NULL; return K_MOSI; }
|
||||||
|
num_banks { yylval=NULL; return K_NUM_PAGES; }
|
||||||
|
num_pages { yylval=NULL; return K_NUM_PAGES; }
|
||||||
|
nvm_base { yylval=NULL; return K_NVM_BASE; }
|
||||||
|
offset { yylval=NULL; return K_OFFSET; }
|
||||||
|
page_size { yylval=NULL; return K_PAGE_SIZE; }
|
||||||
|
paged { yylval=NULL; return K_PAGED; }
|
||||||
|
pagel { yylval=NULL; return K_PAGEL; }
|
||||||
|
par { yylval=NULL; return K_PAR; }
|
||||||
|
parallel { yylval=NULL; return K_PARALLEL; }
|
||||||
|
part { yylval=NULL; return K_PART; }
|
||||||
|
pgmled { yylval=NULL; return K_PGMLED; }
|
||||||
|
programmer { yylval=NULL; return K_PROGRAMMER; }
|
||||||
|
pwroff_after_write { yylval=NULL; return K_PWROFF_AFTER_WRITE; }
|
||||||
|
rampz { yylval=NULL; return K_RAMPZ; }
|
||||||
|
rdyled { yylval=NULL; return K_RDYLED; }
|
||||||
|
readback_p1 { yylval=NULL; return K_READBACK_P1; }
|
||||||
|
readback_p2 { yylval=NULL; return K_READBACK_P2; }
|
||||||
|
retry_pulse { yylval=NULL; return K_RETRY_PULSE; }
|
||||||
|
serbb { yylval=NULL; return K_SERBB; }
|
||||||
|
serial { yylval=NULL; return K_SERIAL; }
|
||||||
|
signature { yylval=NULL; return K_SIGNATURE; }
|
||||||
|
size { yylval=NULL; return K_SIZE; }
|
||||||
|
spmcr { yylval=NULL; return K_SPMCR; }
|
||||||
|
stk500 { yylval=NULL; return K_STK500; }
|
||||||
|
stk500hvsp { yylval=NULL; return K_STK500HVSP; }
|
||||||
|
stk500pp { yylval=NULL; return K_STK500PP; }
|
||||||
|
stk500v2 { yylval=NULL; return K_STK500V2; }
|
||||||
|
stk500generic { yylval=NULL; return K_STK500GENERIC; }
|
||||||
|
stk500_devcode { yylval=NULL; return K_STK500_DEVCODE; }
|
||||||
|
stk600 { yylval=NULL; return K_STK600; }
|
||||||
|
stk600hvsp { yylval=NULL; return K_STK600HVSP; }
|
||||||
|
stk600pp { yylval=NULL; return K_STK600PP; }
|
||||||
|
type { yylval=NULL; return K_TYPE; }
|
||||||
|
usbdev { yylval=NULL; return K_USBDEV; }
|
||||||
|
usbpid { yylval=NULL; return K_USBPID; }
|
||||||
|
usbproduct { yylval=NULL; return K_USBPRODUCT; }
|
||||||
|
usbsn { yylval=NULL; return K_USBSN; }
|
||||||
|
usbvendor { yylval=NULL; return K_USBVENDOR; }
|
||||||
|
usbvid { yylval=NULL; return K_USBVID; }
|
||||||
|
vcc { yylval=NULL; return K_VCC; }
|
||||||
|
vfyled { yylval=NULL; return K_VFYLED; }
|
||||||
|
wiring { yylval=NULL; return K_WIRING; }
|
||||||
|
|
||||||
|
timeout { yylval=NULL; return K_TIMEOUT; }
|
||||||
|
stabdelay { yylval=NULL; return K_STABDELAY; }
|
||||||
|
cmdexedelay { yylval=NULL; return K_CMDEXEDELAY; }
|
||||||
|
hvspcmdexedelay { yylval=NULL; return K_HVSPCMDEXEDELAY; }
|
||||||
|
synchloops { yylval=NULL; return K_SYNCHLOOPS; }
|
||||||
|
bytedelay { yylval=NULL; return K_BYTEDELAY; }
|
||||||
|
pollvalue { yylval=NULL; return K_POLLVALUE; }
|
||||||
|
pollindex { yylval=NULL; return K_POLLINDEX; }
|
||||||
|
predelay { yylval=NULL; return K_PREDELAY; }
|
||||||
|
postdelay { yylval=NULL; return K_POSTDELAY; }
|
||||||
|
pollmethod { yylval=NULL; return K_POLLMETHOD; }
|
||||||
|
mode { yylval=NULL; return K_MODE; }
|
||||||
|
delay { yylval=NULL; return K_DELAY; }
|
||||||
|
blocksize { yylval=NULL; return K_BLOCKSIZE; }
|
||||||
|
readsize { yylval=NULL; return K_READSIZE; }
|
||||||
|
pp_controlstack { yylval=NULL; return K_PP_CONTROLSTACK; }
|
||||||
|
hvsp_controlstack { yylval=NULL; return K_HVSP_CONTROLSTACK; }
|
||||||
|
hventerstabdelay { yylval=NULL; return K_HVENTERSTABDELAY; }
|
||||||
|
progmodedelay { yylval=NULL; return K_PROGMODEDELAY; }
|
||||||
|
latchcycles { yylval=NULL; return K_LATCHCYCLES; }
|
||||||
|
togglevtg { yylval=NULL; return K_TOGGLEVTG; }
|
||||||
|
poweroffdelay { yylval=NULL; return K_POWEROFFDELAY; }
|
||||||
|
resetdelayms { yylval=NULL; return K_RESETDELAYMS; }
|
||||||
|
resetdelayus { yylval=NULL; return K_RESETDELAYUS; }
|
||||||
|
hvleavestabdelay { yylval=NULL; return K_HVLEAVESTABDELAY; }
|
||||||
|
resetdelay { yylval=NULL; return K_RESETDELAY; }
|
||||||
|
synchcycles { yylval=NULL; return K_SYNCHCYCLES; }
|
||||||
|
chiperasepulsewidth { yylval=NULL; return K_CHIPERASEPULSEWIDTH; }
|
||||||
|
chiperasepolltimeout { yylval=NULL; return K_CHIPERASEPOLLTIMEOUT; }
|
||||||
|
chiperasetime { yylval=NULL; return K_CHIPERASETIME; }
|
||||||
|
programfusepulsewidth { yylval=NULL; return K_PROGRAMFUSEPULSEWIDTH; }
|
||||||
|
programfusepolltimeout { yylval=NULL; return K_PROGRAMFUSEPOLLTIMEOUT; }
|
||||||
|
programlockpulsewidth { yylval=NULL; return K_PROGRAMLOCKPULSEWIDTH; }
|
||||||
|
programlockpolltimeout { yylval=NULL; return K_PROGRAMLOCKPOLLTIMEOUT; }
|
||||||
|
flash_instr { yylval=NULL; return K_FLASH_INSTR; }
|
||||||
|
eeprom_instr { yylval=NULL; return K_EEPROM_INSTR; }
|
||||||
|
|
||||||
|
dedicated { yylval=new_token(K_DEDICATED); return K_DEDICATED; }
|
||||||
|
io { yylval=new_token(K_IO); return K_IO; }
|
||||||
|
pseudo { yylval=new_token(K_PSEUDO); return K_PSEUDO; }
|
||||||
|
|
||||||
|
reset { yylval=new_token(K_RESET); return K_RESET; }
|
||||||
|
sck { yylval=new_token(K_SCK); return K_SCK; }
|
||||||
|
|
||||||
|
read { yylval=new_token(K_READ); return K_READ; }
|
||||||
|
write { yylval=new_token(K_WRITE); return K_WRITE; }
|
||||||
|
read_lo { yylval=new_token(K_READ_LO); return K_READ_LO; }
|
||||||
|
read_hi { yylval=new_token(K_READ_HI); return K_READ_HI; }
|
||||||
|
write_lo { yylval=new_token(K_WRITE_LO); return K_WRITE_LO; }
|
||||||
|
write_hi { yylval=new_token(K_WRITE_HI); return K_WRITE_HI; }
|
||||||
|
loadpage_lo { yylval=new_token(K_LOADPAGE_LO); return K_LOADPAGE_LO; }
|
||||||
|
loadpage_hi { yylval=new_token(K_LOADPAGE_HI); return K_LOADPAGE_HI; }
|
||||||
|
load_ext_addr { yylval=new_token(K_LOAD_EXT_ADDR); return K_LOAD_EXT_ADDR; }
|
||||||
|
writepage { yylval=new_token(K_WRITEPAGE); return K_WRITEPAGE; }
|
||||||
|
chip_erase { yylval=new_token(K_CHIP_ERASE); return K_CHIP_ERASE; }
|
||||||
|
pgm_enable { yylval=new_token(K_PGM_ENABLE); return K_PGM_ENABLE; }
|
||||||
|
|
||||||
|
no { yylval=new_token(K_NO); return K_NO; }
|
||||||
|
yes { yylval=new_token(K_YES); return K_YES; }
|
||||||
|
|
||||||
|
"," { yylval = NULL; pyytext(); return TKN_COMMA; }
|
||||||
|
"=" { yylval = NULL; pyytext(); return TKN_EQUAL; }
|
||||||
|
";" { yylval = NULL; pyytext(); return TKN_SEMI; }
|
||||||
|
"~" { yylval = NULL; pyytext(); return TKN_TILDE; }
|
||||||
|
|
||||||
|
"\n" { lineno++; }
|
||||||
|
[ \r\t]+ { /* ignore whitespace */ }
|
||||||
|
|
||||||
|
c: { fprintf(stderr, "error at %s:%d: possible old-style config file entry\n",
|
||||||
|
infile, lineno);
|
||||||
|
fprintf(stderr, " Update your config file (see %s%s for a sample)\n",
|
||||||
|
CONFIG_DIR, "/avrdude.conf.sample");
|
||||||
|
exit(1); }
|
||||||
|
|
||||||
|
. { fprintf(stderr, "error at %s:%d unrecognized character: \"%s\"\n",
|
||||||
|
infile, lineno, yytext); exit(1); }
|
||||||
|
|
||||||
|
%%
|
||||||
|
|
|
@ -0,0 +1,57 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003, 2005 Theodore A. Roth
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef linux_ppdev_h
|
||||||
|
#define linux_ppdev_h
|
||||||
|
|
||||||
|
#define OBSOLETE__IOW _IOW
|
||||||
|
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
#include <linux/parport.h>
|
||||||
|
#include <linux/ppdev.h>
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#define ppi_claim(fd) \
|
||||||
|
if (ioctl(fd, PPCLAIM)) { \
|
||||||
|
fprintf(stderr, "%s: can't claim device \"%s\": %s\n\n", \
|
||||||
|
progname, port, strerror(errno)); \
|
||||||
|
close(fd); \
|
||||||
|
exit(1); \
|
||||||
|
}
|
||||||
|
|
||||||
|
#define ppi_release(fd) \
|
||||||
|
if (ioctl(fd, PPRELEASE)) { \
|
||||||
|
fprintf(stderr, "%s: can't release device: %s\n\n", \
|
||||||
|
progname, strerror(errno)); \
|
||||||
|
exit(1); \
|
||||||
|
}
|
||||||
|
|
||||||
|
#define DO_PPI_READ(fd, reg, valp) \
|
||||||
|
(void)ioctl(fd, \
|
||||||
|
(reg) == PPIDATA? PPRDATA: ((reg) == PPICTRL? PPRCONTROL: PPRSTATUS), \
|
||||||
|
valp)
|
||||||
|
#define DO_PPI_WRITE(fd, reg, valp) \
|
||||||
|
(void)ioctl(fd, \
|
||||||
|
(reg) == PPIDATA? PPWDATA: ((reg) == PPICTRL? PPWCONTROL: PPWSTATUS), \
|
||||||
|
valp)
|
||||||
|
|
||||||
|
#endif /* linux_ppdev_h */
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,115 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 1990-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*----------------------------------------------------------------------
|
||||||
|
Id: lists.h,v 1.2 2001/08/19 23:13:17 bsd Exp $
|
||||||
|
----------------------------------------------------------------------*/
|
||||||
|
/*----------------------------------------------------------------------
|
||||||
|
General purpose linked list routines - header file declarations.
|
||||||
|
|
||||||
|
Author : Brian Dean
|
||||||
|
Date : 10 January, 1990
|
||||||
|
----------------------------------------------------------------------*/
|
||||||
|
#ifndef lists_h
|
||||||
|
#define lists_h
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
typedef void * LISTID;
|
||||||
|
typedef void * LNODEID;
|
||||||
|
|
||||||
|
|
||||||
|
/*----------------------------------------------------------------------
|
||||||
|
several defines to access the LIST structure as as stack or a queue
|
||||||
|
--- use for program readability
|
||||||
|
----------------------------------------------------------------------*/
|
||||||
|
#define STACKID LISTID
|
||||||
|
#define SNODEID LNODEID
|
||||||
|
#define QUEUEID LISTID
|
||||||
|
#define QNODEID LNODEID
|
||||||
|
|
||||||
|
|
||||||
|
#define PUSH(s,d) lins_n(s,d,1) /* push 'd' onto the stack */
|
||||||
|
#define POP(s) lrmv_n(s,1) /* pop the stack */
|
||||||
|
#define LOOKSTACK(s) lget_n(s,1) /* look at the top of the stack,
|
||||||
|
but don't pop */
|
||||||
|
|
||||||
|
|
||||||
|
#define ENQUEUE(q,d) lins_n(q,d,1) /* put 'd' on the end of the queue */
|
||||||
|
#define DEQUEUE(q) lrmv(q) /* remove next item from the front of
|
||||||
|
the queue */
|
||||||
|
#define REQUEUE(q,d) ladd(q,d) /* re-insert (push) item back on the
|
||||||
|
front of the queue */
|
||||||
|
#define LOOKQUEUE(q) lget(q) /* return next item on the queue,
|
||||||
|
but don't dequeue */
|
||||||
|
#define QUEUELEN(q) lsize(q) /* length of the queue */
|
||||||
|
|
||||||
|
|
||||||
|
#define LISTADD(l,d) ladd(l,d) /* add to end of the list */
|
||||||
|
#define LISTRMV(l,d) lrmv_d(l,d) /* remove from end of the list */
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* .................... Function Prototypes .................... */
|
||||||
|
|
||||||
|
LISTID lcreat ( void * liststruct, int poolsize );
|
||||||
|
void ldestroy ( LISTID lid );
|
||||||
|
void ldestroy_cb ( LISTID lid, void (*ucleanup)() );
|
||||||
|
|
||||||
|
LNODEID lfirst ( LISTID ); /* head of the list */
|
||||||
|
LNODEID llast ( LISTID ); /* tail of the list */
|
||||||
|
LNODEID lnext ( LNODEID ); /* next item in the list */
|
||||||
|
LNODEID lprev ( LNODEID ); /* previous item in the list */
|
||||||
|
void * ldata ( LNODEID ); /* data at the current position */
|
||||||
|
int lsize ( LISTID ); /* number of elements in the list */
|
||||||
|
|
||||||
|
int ladd ( LISTID lid, void * p );
|
||||||
|
int laddo ( LISTID lid, void *p,
|
||||||
|
int (*compare)(const void *p1,const void *p2),
|
||||||
|
LNODEID * firstdup );
|
||||||
|
int laddu ( LISTID lid, void * p,
|
||||||
|
int (*compare)(const void *p1,const void *p2));
|
||||||
|
int lins_n ( LISTID lid, void * d, unsigned int n );
|
||||||
|
int lins_ln ( LISTID lid, LNODEID lnid, void * data_ptr );
|
||||||
|
|
||||||
|
void * lget ( LISTID lid );
|
||||||
|
void * lget_n ( LISTID lid, unsigned int n );
|
||||||
|
LNODEID lget_ln ( LISTID lid, unsigned int n );
|
||||||
|
|
||||||
|
void * lrmv ( LISTID lid );
|
||||||
|
void * lrmv_n ( LISTID lid, unsigned int n );
|
||||||
|
void * lrmv_ln ( LISTID lid, LNODEID lnid );
|
||||||
|
void * lrmv_d ( LISTID lid, void * data_ptr );
|
||||||
|
|
||||||
|
LISTID lcat ( LISTID lid1, LISTID lid2 );
|
||||||
|
|
||||||
|
void * lsrch ( LISTID lid, void * p, int (*compare)(void *p1,void *p2));
|
||||||
|
|
||||||
|
int lprint ( FILE * f, LISTID lid );
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,49 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2006 Christian Starkjohann
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
The following is a replacement for hidsdi.h from the Windows DDK. It defines some
|
||||||
|
of the types and function prototypes of this header for our project. If you
|
||||||
|
have the Windows DDK version of this file or a version shipped with MinGW, use
|
||||||
|
that instead.
|
||||||
|
*/
|
||||||
|
#ifndef MY_DDK_HIDSDI_H
|
||||||
|
#define MY_DDK_HIDSDI_H
|
||||||
|
#include <pshpack4.h>
|
||||||
|
#include <ddk/hidusage.h>
|
||||||
|
#include <ddk/hidpi.h>
|
||||||
|
typedef struct{
|
||||||
|
ULONG Size;
|
||||||
|
USHORT VendorID;
|
||||||
|
USHORT ProductID;
|
||||||
|
USHORT VersionNumber;
|
||||||
|
}HIDD_ATTRIBUTES;
|
||||||
|
void __stdcall HidD_GetHidGuid(OUT LPGUID hidGuid);
|
||||||
|
BOOLEAN __stdcall HidD_GetAttributes(IN HANDLE device, OUT HIDD_ATTRIBUTES *attributes);
|
||||||
|
BOOLEAN __stdcall HidD_GetManufacturerString(IN HANDLE device, OUT void *buffer, IN ULONG bufferLen);
|
||||||
|
BOOLEAN __stdcall HidD_GetProductString(IN HANDLE device, OUT void *buffer, IN ULONG bufferLen);
|
||||||
|
BOOLEAN __stdcall HidD_GetSerialNumberString(IN HANDLE device, OUT void *buffer, IN ULONG bufferLen);
|
||||||
|
BOOLEAN __stdcall HidD_GetFeature(IN HANDLE device, OUT void *reportBuffer, IN ULONG bufferLen);
|
||||||
|
BOOLEAN __stdcall HidD_SetFeature(IN HANDLE device, IN void *reportBuffer, IN ULONG bufferLen);
|
||||||
|
BOOLEAN __stdcall HidD_GetNumInputBuffers(IN HANDLE device, OUT ULONG *numBuffers);
|
||||||
|
BOOLEAN __stdcall HidD_SetNumInputBuffers(IN HANDLE device, OUT ULONG numBuffers);
|
||||||
|
#include <poppack.h>
|
||||||
|
#endif /* MY_DDK_HIDSDI_H */
|
|
@ -0,0 +1,462 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2006 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
|
||||||
|
# include "freebsd_ppi.h"
|
||||||
|
#elif defined(__linux__)
|
||||||
|
# include "linux_ppdev.h"
|
||||||
|
#elif defined(__sun__) || defined(__sun) /* Solaris */
|
||||||
|
# include "solaris_ecpp.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "ppi.h"
|
||||||
|
#include "bitbang.h"
|
||||||
|
|
||||||
|
#if HAVE_PARPORT
|
||||||
|
|
||||||
|
struct ppipins_t {
|
||||||
|
int pin;
|
||||||
|
int reg;
|
||||||
|
int bit;
|
||||||
|
int inverted;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct ppipins_t ppipins[] = {
|
||||||
|
{ 1, PPICTRL, 0x01, 1 },
|
||||||
|
{ 2, PPIDATA, 0x01, 0 },
|
||||||
|
{ 3, PPIDATA, 0x02, 0 },
|
||||||
|
{ 4, PPIDATA, 0x04, 0 },
|
||||||
|
{ 5, PPIDATA, 0x08, 0 },
|
||||||
|
{ 6, PPIDATA, 0x10, 0 },
|
||||||
|
{ 7, PPIDATA, 0x20, 0 },
|
||||||
|
{ 8, PPIDATA, 0x40, 0 },
|
||||||
|
{ 9, PPIDATA, 0x80, 0 },
|
||||||
|
{ 10, PPISTATUS, 0x40, 0 },
|
||||||
|
{ 11, PPISTATUS, 0x80, 1 },
|
||||||
|
{ 12, PPISTATUS, 0x20, 0 },
|
||||||
|
{ 13, PPISTATUS, 0x10, 0 },
|
||||||
|
{ 14, PPICTRL, 0x02, 1 },
|
||||||
|
{ 15, PPISTATUS, 0x08, 0 },
|
||||||
|
{ 16, PPICTRL, 0x04, 0 },
|
||||||
|
{ 17, PPICTRL, 0x08, 1 }
|
||||||
|
};
|
||||||
|
|
||||||
|
#define NPINS (sizeof(ppipins)/sizeof(struct ppipins_t))
|
||||||
|
|
||||||
|
static int par_setpin(PROGRAMMER * pgm, int pin, int value)
|
||||||
|
{
|
||||||
|
int inverted;
|
||||||
|
|
||||||
|
inverted = pin & PIN_INVERSE;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
|
||||||
|
if (pin < 1 || pin > 17)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
pin--;
|
||||||
|
|
||||||
|
if (ppipins[pin].inverted)
|
||||||
|
inverted = !inverted;
|
||||||
|
|
||||||
|
if (inverted)
|
||||||
|
value = !value;
|
||||||
|
|
||||||
|
if (value)
|
||||||
|
ppi_set(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
else
|
||||||
|
ppi_clr(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void par_setmany(PROGRAMMER * pgm, unsigned int pinset, int value)
|
||||||
|
{
|
||||||
|
int pin;
|
||||||
|
|
||||||
|
for (pin = 1; pin <= 17; pin++) {
|
||||||
|
if (pinset & (1 << pin))
|
||||||
|
par_setpin(pgm, pin, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int par_getpin(PROGRAMMER * pgm, int pin)
|
||||||
|
{
|
||||||
|
int value;
|
||||||
|
int inverted;
|
||||||
|
|
||||||
|
inverted = pin & PIN_INVERSE;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
|
||||||
|
if (pin < 1 || pin > 17)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
pin--;
|
||||||
|
|
||||||
|
value = ppi_get(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
|
||||||
|
if (value)
|
||||||
|
value = 1;
|
||||||
|
|
||||||
|
if (ppipins[pin].inverted)
|
||||||
|
inverted = !inverted;
|
||||||
|
|
||||||
|
if (inverted)
|
||||||
|
value = !value;
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int par_highpulsepin(PROGRAMMER * pgm, int pin)
|
||||||
|
{
|
||||||
|
int inverted;
|
||||||
|
|
||||||
|
inverted = pin & PIN_INVERSE;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
|
||||||
|
if (pin < 1 || pin > 17)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
pin--;
|
||||||
|
|
||||||
|
if (ppipins[pin].inverted)
|
||||||
|
inverted = !inverted;
|
||||||
|
|
||||||
|
if (inverted) {
|
||||||
|
ppi_clr(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
|
||||||
|
ppi_set(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
} else {
|
||||||
|
ppi_set(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
|
||||||
|
ppi_clr(&pgm->fd, ppipins[pin].reg, ppipins[pin].bit);
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char * pins_to_str(unsigned int pmask)
|
||||||
|
{
|
||||||
|
static char buf[64];
|
||||||
|
int pin;
|
||||||
|
char b2[8];
|
||||||
|
|
||||||
|
buf[0] = 0;
|
||||||
|
for (pin = 1; pin <= 17; pin++) {
|
||||||
|
if (pmask & (1 << pin)) {
|
||||||
|
sprintf(b2, "%d", pin);
|
||||||
|
if (buf[0] != 0)
|
||||||
|
strcat(buf, ",");
|
||||||
|
strcat(buf, b2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* apply power to the AVR processor
|
||||||
|
*/
|
||||||
|
static void par_powerup(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_VCC], 1); /* power up */
|
||||||
|
usleep(100000);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* remove power from the AVR processor
|
||||||
|
*/
|
||||||
|
static void par_powerdown(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_VCC], 0); /* power down */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void par_disable(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_BUFF], 1); /* turn off */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void par_enable(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Prepare to start talking to the connected device - pull reset low
|
||||||
|
* first, delay a few milliseconds, then enable the buffer. This
|
||||||
|
* sequence allows the AVR to be reset before the buffer is enabled
|
||||||
|
* to avoid a short period of time where the AVR may be driving the
|
||||||
|
* programming lines at the same time the programmer tries to. Of
|
||||||
|
* course, if a buffer is being used, then the /RESET line from the
|
||||||
|
* programmer needs to be directly connected to the AVR /RESET line
|
||||||
|
* and not via the buffer chip.
|
||||||
|
*/
|
||||||
|
|
||||||
|
par_setpin(pgm, pgm->pinno[PIN_AVR_RESET], 0);
|
||||||
|
usleep(1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* enable the 74367 buffer, if connected; this signal is active low
|
||||||
|
*/
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_BUFF], 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int par_open(PROGRAMMER * pgm, char * port)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
bitbang_check_prerequisites(pgm);
|
||||||
|
|
||||||
|
ppi_open(port, &pgm->fd);
|
||||||
|
if (pgm->fd.ifd < 0) {
|
||||||
|
fprintf(stderr, "%s: failed to open parallel port \"%s\"\n\n",
|
||||||
|
progname, port);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* save pin values, so they can be restored when device is closed
|
||||||
|
*/
|
||||||
|
rc = ppi_getall(&pgm->fd, PPIDATA);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: error reading status of ppi data port\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
pgm->ppidata = rc;
|
||||||
|
|
||||||
|
rc = ppi_getall(&pgm->fd, PPICTRL);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: error reading status of ppi ctrl port\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
pgm->ppictrl = rc;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void par_close(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Restore pin values before closing,
|
||||||
|
* but ensure that buffers are turned off.
|
||||||
|
*/
|
||||||
|
ppi_setall(&pgm->fd, PPIDATA, pgm->ppidata);
|
||||||
|
ppi_setall(&pgm->fd, PPICTRL, pgm->ppictrl);
|
||||||
|
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_BUFF], 1);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Handle exit specs.
|
||||||
|
*/
|
||||||
|
switch (pgm->exit_reset) {
|
||||||
|
case EXIT_RESET_ENABLED:
|
||||||
|
par_setpin(pgm, pgm->pinno[PIN_AVR_RESET], 0);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EXIT_RESET_DISABLED:
|
||||||
|
par_setpin(pgm, pgm->pinno[PIN_AVR_RESET], 1);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EXIT_RESET_UNSPEC:
|
||||||
|
/* Leave it alone. */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (pgm->exit_datahigh) {
|
||||||
|
case EXIT_DATAHIGH_ENABLED:
|
||||||
|
ppi_setall(&pgm->fd, PPIDATA, 0xff);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EXIT_DATAHIGH_DISABLED:
|
||||||
|
ppi_setall(&pgm->fd, PPIDATA, 0x00);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EXIT_DATAHIGH_UNSPEC:
|
||||||
|
/* Leave it alone. */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (pgm->exit_vcc) {
|
||||||
|
case EXIT_VCC_ENABLED:
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_VCC], 1);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EXIT_VCC_DISABLED:
|
||||||
|
par_setmany(pgm, pgm->pinno[PPI_AVR_VCC], 0);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case EXIT_VCC_UNSPEC:
|
||||||
|
/* Leave it alone. */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ppi_close(&pgm->fd);
|
||||||
|
pgm->fd.ifd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void par_display(PROGRAMMER * pgm, const char * p)
|
||||||
|
{
|
||||||
|
char vccpins[64];
|
||||||
|
char buffpins[64];
|
||||||
|
|
||||||
|
if (pgm->pinno[PPI_AVR_VCC]) {
|
||||||
|
snprintf(vccpins, sizeof(vccpins), "%s",
|
||||||
|
pins_to_str(pgm->pinno[PPI_AVR_VCC]));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
strcpy(vccpins, " (not used)");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pgm->pinno[PPI_AVR_BUFF]) {
|
||||||
|
snprintf(buffpins, sizeof(buffpins), "%s",
|
||||||
|
pins_to_str(pgm->pinno[PPI_AVR_BUFF]));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
strcpy(buffpins, " (not used)");
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s VCC = %s\n"
|
||||||
|
"%s BUFF = %s\n"
|
||||||
|
"%s RESET = %d\n"
|
||||||
|
"%s SCK = %d\n"
|
||||||
|
"%s MOSI = %d\n"
|
||||||
|
"%s MISO = %d\n"
|
||||||
|
"%s ERR LED = %d\n"
|
||||||
|
"%s RDY LED = %d\n"
|
||||||
|
"%s PGM LED = %d\n"
|
||||||
|
"%s VFY LED = %d\n",
|
||||||
|
|
||||||
|
p, vccpins,
|
||||||
|
p, buffpins,
|
||||||
|
p, pgm->pinno[PIN_AVR_RESET],
|
||||||
|
p, pgm->pinno[PIN_AVR_SCK],
|
||||||
|
p, pgm->pinno[PIN_AVR_MOSI],
|
||||||
|
p, pgm->pinno[PIN_AVR_MISO],
|
||||||
|
p, pgm->pinno[PIN_LED_ERR],
|
||||||
|
p, pgm->pinno[PIN_LED_RDY],
|
||||||
|
p, pgm->pinno[PIN_LED_PGM],
|
||||||
|
p, pgm->pinno[PIN_LED_VFY]);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* parse the -E string
|
||||||
|
*/
|
||||||
|
static int par_parseexitspecs(PROGRAMMER * pgm, char *s)
|
||||||
|
{
|
||||||
|
char *cp;
|
||||||
|
|
||||||
|
while ((cp = strtok(s, ","))) {
|
||||||
|
if (strcmp(cp, "reset") == 0) {
|
||||||
|
pgm->exit_reset = EXIT_RESET_ENABLED;
|
||||||
|
}
|
||||||
|
else if (strcmp(cp, "noreset") == 0) {
|
||||||
|
pgm->exit_reset = EXIT_RESET_DISABLED;
|
||||||
|
}
|
||||||
|
else if (strcmp(cp, "vcc") == 0) {
|
||||||
|
pgm->exit_vcc = EXIT_VCC_ENABLED;
|
||||||
|
}
|
||||||
|
else if (strcmp(cp, "novcc") == 0) {
|
||||||
|
pgm->exit_vcc = EXIT_VCC_DISABLED;
|
||||||
|
}
|
||||||
|
else if (strcmp(cp, "d_high") == 0) {
|
||||||
|
pgm->exit_datahigh = EXIT_DATAHIGH_ENABLED;
|
||||||
|
}
|
||||||
|
else if (strcmp(cp, "d_low") == 0) {
|
||||||
|
pgm->exit_datahigh = EXIT_DATAHIGH_DISABLED;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
s = 0; /* strtok() should be called with the actual string only once */
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void par_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "PPI");
|
||||||
|
|
||||||
|
pgm->exit_vcc = EXIT_VCC_UNSPEC;
|
||||||
|
pgm->exit_reset = EXIT_RESET_UNSPEC;
|
||||||
|
pgm->exit_datahigh = EXIT_DATAHIGH_UNSPEC;
|
||||||
|
|
||||||
|
pgm->rdy_led = bitbang_rdy_led;
|
||||||
|
pgm->err_led = bitbang_err_led;
|
||||||
|
pgm->pgm_led = bitbang_pgm_led;
|
||||||
|
pgm->vfy_led = bitbang_vfy_led;
|
||||||
|
pgm->initialize = bitbang_initialize;
|
||||||
|
pgm->display = par_display;
|
||||||
|
pgm->enable = par_enable;
|
||||||
|
pgm->disable = par_disable;
|
||||||
|
pgm->powerup = par_powerup;
|
||||||
|
pgm->powerdown = par_powerdown;
|
||||||
|
pgm->program_enable = bitbang_program_enable;
|
||||||
|
pgm->chip_erase = bitbang_chip_erase;
|
||||||
|
pgm->cmd = bitbang_cmd;
|
||||||
|
pgm->cmd_tpi = bitbang_cmd_tpi;
|
||||||
|
pgm->spi = bitbang_spi;
|
||||||
|
pgm->open = par_open;
|
||||||
|
pgm->close = par_close;
|
||||||
|
pgm->setpin = par_setpin;
|
||||||
|
pgm->getpin = par_getpin;
|
||||||
|
pgm->highpulsepin = par_highpulsepin;
|
||||||
|
pgm->parseexitspecs = par_parseexitspecs;
|
||||||
|
pgm->read_byte = avr_read_byte_default;
|
||||||
|
pgm->write_byte = avr_write_byte_default;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* !HAVE_PARPORT */
|
||||||
|
|
||||||
|
void par_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: parallel port access not available in this configuration\n",
|
||||||
|
progname);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* HAVE_PARPORT */
|
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef par_h
|
||||||
|
#define par_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void par_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,228 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2002-2004 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright 2007 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
|
||||||
|
static int pgm_default_2 (struct programmer_t *, AVRPART *);
|
||||||
|
static int pgm_default_3 (struct programmer_t * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char * value);
|
||||||
|
static void pgm_default_4 (struct programmer_t *);
|
||||||
|
static int pgm_default_5 (struct programmer_t * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char data);
|
||||||
|
static void pgm_default_6 (struct programmer_t *, const char *);
|
||||||
|
|
||||||
|
|
||||||
|
static int pgm_default_open (struct programmer_t *pgm, char * name)
|
||||||
|
{
|
||||||
|
fprintf (stderr, "\n%s: Fatal error: Programmer does not support open()",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int pgm_default_led (struct programmer_t * pgm, int value)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* If programmer has no LEDs, just do nothing.
|
||||||
|
*/
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void pgm_default_powerup_powerdown (struct programmer_t * pgm)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* If programmer does not support powerup/down, just do nothing.
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
PROGRAMMER * pgm_new(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
PROGRAMMER * pgm;
|
||||||
|
|
||||||
|
pgm = (PROGRAMMER *)malloc(sizeof(*pgm));
|
||||||
|
if (pgm == NULL) {
|
||||||
|
fprintf(stderr, "%s: out of memory allocating programmer structure\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(pgm, 0, sizeof(*pgm));
|
||||||
|
|
||||||
|
pgm->id = lcreat(NULL, 0);
|
||||||
|
pgm->desc[0] = 0;
|
||||||
|
pgm->type[0] = 0;
|
||||||
|
pgm->config_file[0] = 0;
|
||||||
|
pgm->lineno = 0;
|
||||||
|
pgm->baudrate = 0;
|
||||||
|
|
||||||
|
for (i=0; i<N_PINS; i++)
|
||||||
|
pgm->pinno[i] = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* mandatory functions - these are called without checking to see
|
||||||
|
* whether they are assigned or not
|
||||||
|
*/
|
||||||
|
pgm->initialize = pgm_default_2;
|
||||||
|
pgm->display = pgm_default_6;
|
||||||
|
pgm->enable = pgm_default_4;
|
||||||
|
pgm->disable = pgm_default_4;
|
||||||
|
pgm->powerup = pgm_default_powerup_powerdown;
|
||||||
|
pgm->powerdown = pgm_default_powerup_powerdown;
|
||||||
|
pgm->program_enable = pgm_default_2;
|
||||||
|
pgm->chip_erase = pgm_default_2;
|
||||||
|
pgm->open = pgm_default_open;
|
||||||
|
pgm->close = pgm_default_4;
|
||||||
|
pgm->read_byte = pgm_default_3;
|
||||||
|
pgm->write_byte = pgm_default_5;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* predefined functions - these functions have a valid default
|
||||||
|
* implementation. Hence, they don't need to be defined in
|
||||||
|
* the programmer.
|
||||||
|
*/
|
||||||
|
pgm->rdy_led = pgm_default_led;
|
||||||
|
pgm->err_led = pgm_default_led;
|
||||||
|
pgm->pgm_led = pgm_default_led;
|
||||||
|
pgm->vfy_led = pgm_default_led;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* optional functions - these are checked to make sure they are
|
||||||
|
* assigned before they are called
|
||||||
|
*/
|
||||||
|
pgm->cmd = NULL;
|
||||||
|
pgm->cmd_tpi = NULL;
|
||||||
|
pgm->spi = NULL;
|
||||||
|
pgm->paged_write = NULL;
|
||||||
|
pgm->paged_load = NULL;
|
||||||
|
pgm->write_setup = NULL;
|
||||||
|
pgm->read_sig_bytes = NULL;
|
||||||
|
pgm->set_vtarget = NULL;
|
||||||
|
pgm->set_varef = NULL;
|
||||||
|
pgm->set_fosc = NULL;
|
||||||
|
pgm->perform_osccal = NULL;
|
||||||
|
pgm->parseextparams = NULL;
|
||||||
|
pgm->setup = NULL;
|
||||||
|
pgm->teardown = NULL;
|
||||||
|
|
||||||
|
return pgm;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void pgm_default(void)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: programmer operation not supported\n", progname);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int pgm_default_2 (struct programmer_t * pgm, AVRPART * p)
|
||||||
|
{
|
||||||
|
pgm_default();
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int pgm_default_3 (struct programmer_t * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char * value)
|
||||||
|
{
|
||||||
|
pgm_default();
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void pgm_default_4 (struct programmer_t * pgm)
|
||||||
|
{
|
||||||
|
pgm_default();
|
||||||
|
}
|
||||||
|
|
||||||
|
static int pgm_default_5 (struct programmer_t * pgm, AVRPART * p, AVRMEM * mem,
|
||||||
|
unsigned long addr, unsigned char data)
|
||||||
|
{
|
||||||
|
pgm_default();
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void pgm_default_6 (struct programmer_t * pgm, const char * p)
|
||||||
|
{
|
||||||
|
pgm_default();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void programmer_display(PROGRAMMER * pgm, const char * p)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%sProgrammer Type : %s\n", p, pgm->type);
|
||||||
|
fprintf(stderr, "%sDescription : %s\n", p, pgm->desc);
|
||||||
|
|
||||||
|
pgm->display(pgm, p);
|
||||||
|
}
|
||||||
|
|
||||||
|
PROGRAMMER * locate_programmer(LISTID programmers, const char * configid)
|
||||||
|
{
|
||||||
|
LNODEID ln1, ln2;
|
||||||
|
PROGRAMMER * p = NULL;
|
||||||
|
const char * id;
|
||||||
|
int found;
|
||||||
|
|
||||||
|
found = 0;
|
||||||
|
|
||||||
|
for (ln1=lfirst(programmers); ln1 && !found; ln1=lnext(ln1)) {
|
||||||
|
p = ldata(ln1);
|
||||||
|
for (ln2=lfirst(p->id); ln2 && !found; ln2=lnext(ln2)) {
|
||||||
|
id = ldata(ln2);
|
||||||
|
if (strcasecmp(configid, id) == 0)
|
||||||
|
found = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (found)
|
||||||
|
return p;
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Iterate over the list of programmers given as "programmers", and
|
||||||
|
* call the callback function cb for each entry found. cb is being
|
||||||
|
* passed the following arguments:
|
||||||
|
* . the name of the programmer (for -c)
|
||||||
|
* . the descriptive text given in the config file
|
||||||
|
* . the name of the config file this programmer has been defined in
|
||||||
|
* . the line number of the config file this programmer has been defined at
|
||||||
|
* . the "cookie" passed into walk_programmers() (opaque client data)
|
||||||
|
*/
|
||||||
|
void walk_programmers(LISTID programmers, walk_programmers_cb cb, void *cookie)
|
||||||
|
{
|
||||||
|
LNODEID ln1;
|
||||||
|
PROGRAMMER * p;
|
||||||
|
|
||||||
|
for (ln1 = lfirst(programmers); ln1; ln1 = lnext(ln1)) {
|
||||||
|
p = ldata(ln1);
|
||||||
|
cb((char *)ldata(lfirst(p->id)), p->desc, p->config_file, p->lineno, cookie);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,145 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2002-2004 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright 2007 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef pgm_h
|
||||||
|
#define pgm_h
|
||||||
|
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
|
#include "avrpart.h"
|
||||||
|
#include "lists.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
#define ON 1
|
||||||
|
#define OFF 0
|
||||||
|
|
||||||
|
#define PGM_DESCLEN 80
|
||||||
|
#define PGM_PORTLEN PATH_MAX
|
||||||
|
#define PGM_TYPELEN 32
|
||||||
|
#define PGM_USBSTRINGLEN 256
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
EXIT_VCC_UNSPEC,
|
||||||
|
EXIT_VCC_ENABLED,
|
||||||
|
EXIT_VCC_DISABLED
|
||||||
|
} exit_vcc_t;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
EXIT_RESET_UNSPEC,
|
||||||
|
EXIT_RESET_ENABLED,
|
||||||
|
EXIT_RESET_DISABLED
|
||||||
|
} exit_reset_t;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
EXIT_DATAHIGH_UNSPEC,
|
||||||
|
EXIT_DATAHIGH_ENABLED,
|
||||||
|
EXIT_DATAHIGH_DISABLED
|
||||||
|
} exit_datahigh_t;
|
||||||
|
|
||||||
|
typedef struct programmer_t {
|
||||||
|
LISTID id;
|
||||||
|
char desc[PGM_DESCLEN];
|
||||||
|
char type[PGM_TYPELEN];
|
||||||
|
char port[PGM_PORTLEN];
|
||||||
|
unsigned int pinno[N_PINS];
|
||||||
|
exit_vcc_t exit_vcc;
|
||||||
|
exit_reset_t exit_reset;
|
||||||
|
exit_datahigh_t exit_datahigh;
|
||||||
|
int ppidata;
|
||||||
|
int ppictrl;
|
||||||
|
int baudrate;
|
||||||
|
int usbvid, usbpid;
|
||||||
|
char usbdev[PGM_USBSTRINGLEN], usbsn[PGM_USBSTRINGLEN];
|
||||||
|
char usbvendor[PGM_USBSTRINGLEN], usbproduct[PGM_USBSTRINGLEN];
|
||||||
|
double bitclock; /* JTAG ICE clock period in microseconds */
|
||||||
|
int ispdelay; /* ISP clock delay */
|
||||||
|
union filedescriptor fd;
|
||||||
|
int page_size; /* page size if the programmer supports paged write/load */
|
||||||
|
int (*rdy_led) (struct programmer_t * pgm, int value);
|
||||||
|
int (*err_led) (struct programmer_t * pgm, int value);
|
||||||
|
int (*pgm_led) (struct programmer_t * pgm, int value);
|
||||||
|
int (*vfy_led) (struct programmer_t * pgm, int value);
|
||||||
|
int (*initialize) (struct programmer_t * pgm, AVRPART * p);
|
||||||
|
void (*display) (struct programmer_t * pgm, const char * p);
|
||||||
|
void (*enable) (struct programmer_t * pgm);
|
||||||
|
void (*disable) (struct programmer_t * pgm);
|
||||||
|
void (*powerup) (struct programmer_t * pgm);
|
||||||
|
void (*powerdown) (struct programmer_t * pgm);
|
||||||
|
int (*program_enable) (struct programmer_t * pgm, AVRPART * p);
|
||||||
|
int (*chip_erase) (struct programmer_t * pgm, AVRPART * p);
|
||||||
|
int (*cmd) (struct programmer_t * pgm, unsigned char cmd[4],
|
||||||
|
unsigned char res[4]);
|
||||||
|
int (*cmd_tpi) (struct programmer_t * pgm, unsigned char cmd[],
|
||||||
|
int cmd_len, unsigned char res[], int res_len);
|
||||||
|
int (*spi) (struct programmer_t * pgm, unsigned char cmd[],
|
||||||
|
unsigned char res[], int count);
|
||||||
|
int (*open) (struct programmer_t * pgm, char * port);
|
||||||
|
void (*close) (struct programmer_t * pgm);
|
||||||
|
int (*paged_write) (struct programmer_t * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes);
|
||||||
|
int (*paged_load) (struct programmer_t * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
int page_size, int n_bytes);
|
||||||
|
void (*write_setup) (struct programmer_t * pgm, AVRPART * p, AVRMEM * m);
|
||||||
|
int (*write_byte) (struct programmer_t * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char value);
|
||||||
|
int (*read_byte) (struct programmer_t * pgm, AVRPART * p, AVRMEM * m,
|
||||||
|
unsigned long addr, unsigned char * value);
|
||||||
|
int (*read_sig_bytes) (struct programmer_t * pgm, AVRPART * p, AVRMEM * m);
|
||||||
|
void (*print_parms) (struct programmer_t * pgm);
|
||||||
|
int (*set_vtarget) (struct programmer_t * pgm, double v);
|
||||||
|
int (*set_varef) (struct programmer_t * pgm, unsigned int chan, double v);
|
||||||
|
int (*set_fosc) (struct programmer_t * pgm, double v);
|
||||||
|
int (*set_sck_period) (struct programmer_t * pgm, double v);
|
||||||
|
int (*setpin) (struct programmer_t * pgm, int pin, int value);
|
||||||
|
int (*getpin) (struct programmer_t * pgm, int pin);
|
||||||
|
int (*highpulsepin) (struct programmer_t * pgm, int pin);
|
||||||
|
int (*parseexitspecs) (struct programmer_t * pgm, char *s);
|
||||||
|
int (*perform_osccal) (struct programmer_t * pgm);
|
||||||
|
int (*parseextparams) (struct programmer_t * pgm, LISTID xparams);
|
||||||
|
void (*setup) (struct programmer_t * pgm);
|
||||||
|
void (*teardown) (struct programmer_t * pgm);
|
||||||
|
char config_file[PATH_MAX]; /* config file where defined */
|
||||||
|
int lineno; /* config file line number */
|
||||||
|
void *cookie; /* for private use by the programmer */
|
||||||
|
char flag; /* for private use of the programmer */
|
||||||
|
} PROGRAMMER;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
PROGRAMMER * pgm_new(void);
|
||||||
|
|
||||||
|
void programmer_display(PROGRAMMER * pgm, const char * p);
|
||||||
|
PROGRAMMER * locate_programmer(LISTID programmers, const char * configid);
|
||||||
|
|
||||||
|
typedef void (*walk_programmers_cb)(const char *name, const char *desc,
|
||||||
|
const char *cfgname, int cfglineno,
|
||||||
|
void *cookie);
|
||||||
|
void walk_programmers(LISTID programmers, walk_programmers_cb cb, void *cookie);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,44 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef __pindefs_h__
|
||||||
|
#define __pindefs_h__
|
||||||
|
|
||||||
|
enum {
|
||||||
|
PPI_AVR_VCC=1,
|
||||||
|
PPI_AVR_BUFF,
|
||||||
|
PIN_AVR_RESET,
|
||||||
|
PIN_AVR_SCK,
|
||||||
|
PIN_AVR_MOSI,
|
||||||
|
PIN_AVR_MISO,
|
||||||
|
PIN_LED_ERR,
|
||||||
|
PIN_LED_RDY,
|
||||||
|
PIN_LED_PGM,
|
||||||
|
PIN_LED_VFY,
|
||||||
|
N_PINS
|
||||||
|
};
|
||||||
|
#define PIN_INVERSE 0x80 /* flag for inverted pin in serbb */
|
||||||
|
#define PIN_MASK 0x7f
|
||||||
|
|
||||||
|
#define LED_ON(fd,pin) ppi_setpin(fd,pin,0)
|
||||||
|
#define LED_OFF(fd,pin) ppi_setpin(fd,pin,1)
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,238 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
|
||||||
|
#if !defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#if HAVE_PARPORT
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
|
||||||
|
# include "freebsd_ppi.h"
|
||||||
|
#elif defined(__linux__)
|
||||||
|
# include "linux_ppdev.h"
|
||||||
|
#elif defined(__sun__) || defined(__sun) /* Solaris */
|
||||||
|
# include "solaris_ecpp.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "ppi.h"
|
||||||
|
|
||||||
|
enum {
|
||||||
|
PPI_READ,
|
||||||
|
PPI_WRITE,
|
||||||
|
PPI_SHADOWREAD
|
||||||
|
};
|
||||||
|
|
||||||
|
static int ppi_shadow_access(union filedescriptor *fdp, int reg,
|
||||||
|
unsigned char *v, unsigned char action)
|
||||||
|
{
|
||||||
|
static unsigned char shadow[3];
|
||||||
|
int shadow_num;
|
||||||
|
|
||||||
|
switch (reg) {
|
||||||
|
case PPIDATA:
|
||||||
|
shadow_num = 0;
|
||||||
|
break;
|
||||||
|
case PPICTRL:
|
||||||
|
shadow_num = 1;
|
||||||
|
break;
|
||||||
|
case PPISTATUS:
|
||||||
|
shadow_num = 2;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "%s: avr_set(): invalid register=%d\n",
|
||||||
|
progname, reg);
|
||||||
|
return -1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (action) {
|
||||||
|
case PPI_SHADOWREAD:
|
||||||
|
*v = shadow[shadow_num];
|
||||||
|
break;
|
||||||
|
case PPI_READ:
|
||||||
|
DO_PPI_READ(fdp->ifd, reg, v);
|
||||||
|
shadow[shadow_num]=*v;
|
||||||
|
break;
|
||||||
|
case PPI_WRITE:
|
||||||
|
shadow[shadow_num]=*v;
|
||||||
|
DO_PPI_WRITE(fdp->ifd, reg, v);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_set(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = ppi_shadow_access(fdp, reg, &v, PPI_SHADOWREAD);
|
||||||
|
v |= bit;
|
||||||
|
rc |= ppi_shadow_access(fdp, reg, &v, PPI_WRITE);
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* clear the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_clr(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = ppi_shadow_access(fdp, reg, &v, PPI_SHADOWREAD);
|
||||||
|
v &= ~bit;
|
||||||
|
rc |= ppi_shadow_access(fdp, reg, &v, PPI_WRITE);
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* get the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_get(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = ppi_shadow_access(fdp, reg, &v, PPI_READ);
|
||||||
|
v &= bit;
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return v; /* v == bit */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* toggle the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_toggle(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = ppi_shadow_access(fdp, reg, &v, PPI_SHADOWREAD);
|
||||||
|
v ^= bit;
|
||||||
|
rc |= ppi_shadow_access(fdp, reg, &v, PPI_WRITE);
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* get all bits of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_getall(union filedescriptor *fdp, int reg)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
rc = ppi_shadow_access(fdp, reg, &v, PPI_READ);
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return v; /* v == bit */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set all bits of the specified register to val.
|
||||||
|
*/
|
||||||
|
int ppi_setall(union filedescriptor *fdp, int reg, int val)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
v = val;
|
||||||
|
rc = ppi_shadow_access(fdp, reg, &v, PPI_WRITE);
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ppi_open(char * port, union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
int fd;
|
||||||
|
unsigned char v;
|
||||||
|
|
||||||
|
fd = open(port, O_RDWR);
|
||||||
|
if (fd < 0) {
|
||||||
|
fprintf(stderr, "%s: can't open device \"%s\": %s\n",
|
||||||
|
progname, port, strerror(errno));
|
||||||
|
fdp->ifd = -1;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ppi_claim (fd);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Initialize shadow registers
|
||||||
|
*/
|
||||||
|
|
||||||
|
ppi_shadow_access (fdp, PPIDATA, &v, PPI_READ);
|
||||||
|
ppi_shadow_access (fdp, PPICTRL, &v, PPI_READ);
|
||||||
|
ppi_shadow_access (fdp, PPISTATUS, &v, PPI_READ);
|
||||||
|
|
||||||
|
fdp->ifd = fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ppi_close(union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
ppi_release (fdp->ifd);
|
||||||
|
close(fdp->ifd);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* HAVE_PARPORT */
|
||||||
|
|
||||||
|
#endif /* !WIN32NATIVE */
|
|
@ -0,0 +1,60 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef ppi_h
|
||||||
|
#define ppi_h
|
||||||
|
|
||||||
|
/*
|
||||||
|
* PPI registers
|
||||||
|
*/
|
||||||
|
enum {
|
||||||
|
PPIDATA,
|
||||||
|
PPICTRL,
|
||||||
|
PPISTATUS
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int ppi_get (union filedescriptor *fdp, int reg, int bit);
|
||||||
|
|
||||||
|
int ppi_set (union filedescriptor *fdp, int reg, int bit);
|
||||||
|
|
||||||
|
int ppi_clr (union filedescriptor *fdp, int reg, int bit);
|
||||||
|
|
||||||
|
int ppi_getall (union filedescriptor *fdp, int reg);
|
||||||
|
|
||||||
|
int ppi_setall (union filedescriptor *fdp, int reg, int val);
|
||||||
|
|
||||||
|
int ppi_toggle (union filedescriptor *fdp, int reg, int bit);
|
||||||
|
|
||||||
|
void ppi_open (char * port, union filedescriptor *fdp);
|
||||||
|
|
||||||
|
void ppi_close (union filedescriptor *fdp);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,418 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003, 2004, 2006
|
||||||
|
* Eric B. Weddington <eweddington@cso.atmel.com>
|
||||||
|
* Copyright 2008, Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
This is the parallel port interface for Windows built using Cygwin.
|
||||||
|
|
||||||
|
In the ppi_* functions that access the parallel port registers,
|
||||||
|
fd = parallel port address
|
||||||
|
reg = register as defined in an enum in ppi.h. This must be converted
|
||||||
|
to a proper offset of the base address.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
#include "avrdude.h"
|
||||||
|
|
||||||
|
#if defined (WIN32NATIVE)
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <windows.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include "serial.h"
|
||||||
|
#include "ppi.h"
|
||||||
|
|
||||||
|
#define DEVICE_LPT1 "lpt1"
|
||||||
|
#define DEVICE_LPT2 "lpt2"
|
||||||
|
#define DEVICE_LPT3 "lpt3"
|
||||||
|
|
||||||
|
#define DEVICE_MAX 3
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
const char *name;
|
||||||
|
int base_address;
|
||||||
|
} winpp;
|
||||||
|
|
||||||
|
static const winpp winports[DEVICE_MAX] =
|
||||||
|
{
|
||||||
|
{DEVICE_LPT1, 0x378},
|
||||||
|
{DEVICE_LPT2, 0x278},
|
||||||
|
{DEVICE_LPT3, 0x3BC},
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* FUNCTION PROTOTYPES */
|
||||||
|
static int winnt_pp_open(void);
|
||||||
|
static unsigned short port_get(union filedescriptor *fdp, int reg);
|
||||||
|
static unsigned char reg2offset(int reg);
|
||||||
|
static unsigned char inb(unsigned short port);
|
||||||
|
static void outb(unsigned char value, unsigned short port);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* FUNCTION DEFINITIONS */
|
||||||
|
|
||||||
|
void ppi_open(char *port, union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
unsigned char i;
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
fd = winnt_pp_open();
|
||||||
|
|
||||||
|
if(fd < 0)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: can't open device \"giveio\"\n\n", progname);
|
||||||
|
fdp->ifd = -1;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Search the windows port names for a match */
|
||||||
|
fd = -1;
|
||||||
|
for(i = 0; i < DEVICE_MAX; i++)
|
||||||
|
{
|
||||||
|
if(strcmp(winports[i].name, port) == 0)
|
||||||
|
{
|
||||||
|
/* Set the file descriptor with the Windows parallel port base address. */
|
||||||
|
fd = winports[i].base_address;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(fd == -1)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Supplied port name did not match any of the pre-defined
|
||||||
|
* names. Try interpreting it as a numeric
|
||||||
|
* (hexadecimal/decimal/octal) address.
|
||||||
|
*/
|
||||||
|
char *cp;
|
||||||
|
|
||||||
|
fd = strtol(port, &cp, 0);
|
||||||
|
if(*port == '\0' || *cp != '\0')
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: port name \"%s\" is neither lpt1/2/3 nor valid number\n",
|
||||||
|
progname, port);
|
||||||
|
fd = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(fd < 0)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: can't open device \"%s\"\n\n", progname, port);
|
||||||
|
fdp->ifd = -1;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fdp->ifd = fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#define DRIVERNAME "\\\\.\\giveio"
|
||||||
|
static int winnt_pp_open(void)
|
||||||
|
{
|
||||||
|
// Only try to use giveio under Windows NT/2000/XP.
|
||||||
|
OSVERSIONINFO ver_info;
|
||||||
|
|
||||||
|
memset(&ver_info, 0, sizeof(ver_info));
|
||||||
|
|
||||||
|
ver_info.dwOSVersionInfoSize = sizeof(ver_info);
|
||||||
|
|
||||||
|
if(!GetVersionEx(&ver_info))
|
||||||
|
{
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
else if(ver_info.dwPlatformId == VER_PLATFORM_WIN32_NT)
|
||||||
|
{
|
||||||
|
HANDLE h = CreateFile(DRIVERNAME,
|
||||||
|
GENERIC_READ,
|
||||||
|
0,
|
||||||
|
NULL,
|
||||||
|
OPEN_EXISTING,
|
||||||
|
FILE_ATTRIBUTE_NORMAL,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
if(h == INVALID_HANDLE_VALUE)
|
||||||
|
{
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Close immediately. The process now has the rights it needs. */
|
||||||
|
if(h != NULL)
|
||||||
|
{
|
||||||
|
CloseHandle(h);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
void ppi_close(union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_set(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
unsigned short port;
|
||||||
|
|
||||||
|
port = port_get(fdp, reg);
|
||||||
|
v = inb(port);
|
||||||
|
v |= bit;
|
||||||
|
outb(v, port);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* clear the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_clr(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
unsigned short port;
|
||||||
|
|
||||||
|
port = port_get(fdp, reg);
|
||||||
|
v = inb(port);
|
||||||
|
v &= ~bit;
|
||||||
|
outb(v, port);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* get the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_get(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
|
||||||
|
v = inb(port_get(fdp, reg));
|
||||||
|
v &= bit;
|
||||||
|
|
||||||
|
return(v);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* toggle the indicated bit of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_toggle(union filedescriptor *fdp, int reg, int bit)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
unsigned short port;
|
||||||
|
|
||||||
|
port = port_get(fdp, reg);
|
||||||
|
|
||||||
|
v = inb(port);
|
||||||
|
v ^= bit;
|
||||||
|
outb(v, port);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* get all bits of the specified register.
|
||||||
|
*/
|
||||||
|
int ppi_getall(union filedescriptor *fdp, int reg)
|
||||||
|
{
|
||||||
|
unsigned char v;
|
||||||
|
|
||||||
|
v = inb(port_get(fdp, reg));
|
||||||
|
|
||||||
|
return((int)v);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set all bits of the specified register to val.
|
||||||
|
*/
|
||||||
|
int ppi_setall(union filedescriptor *fdp, int reg, int val)
|
||||||
|
{
|
||||||
|
outb((unsigned char)val, port_get(fdp, reg));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Calculate port address to access. */
|
||||||
|
static unsigned short port_get(union filedescriptor *fdp, int reg)
|
||||||
|
{
|
||||||
|
return((unsigned short)(fdp->ifd + reg2offset(reg)));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Convert register enum to offset of base address. */
|
||||||
|
static unsigned char reg2offset(int reg)
|
||||||
|
{
|
||||||
|
unsigned char offset = 0;
|
||||||
|
|
||||||
|
switch(reg)
|
||||||
|
{
|
||||||
|
case PPIDATA:
|
||||||
|
{
|
||||||
|
offset = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case PPISTATUS:
|
||||||
|
{
|
||||||
|
offset = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case PPICTRL:
|
||||||
|
{
|
||||||
|
offset = 2;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return(offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Read in value from port. */
|
||||||
|
static unsigned char inb(unsigned short port)
|
||||||
|
{
|
||||||
|
unsigned char t;
|
||||||
|
|
||||||
|
asm volatile ("in %1, %0"
|
||||||
|
: "=a" (t)
|
||||||
|
: "d" (port));
|
||||||
|
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Write value to port. */
|
||||||
|
static void outb(unsigned char value, unsigned short port)
|
||||||
|
{
|
||||||
|
asm volatile ("out %1, %0"
|
||||||
|
:
|
||||||
|
: "d" (port), "a" (value) );
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if !defined(HAVE_GETTIMEOFDAY)
|
||||||
|
struct timezone;
|
||||||
|
int gettimeofday(struct timeval *tv, struct timezone *unused){
|
||||||
|
// i've found only ms resolution, avrdude expects us
|
||||||
|
|
||||||
|
SYSTEMTIME st;
|
||||||
|
GetSystemTime(&st);
|
||||||
|
|
||||||
|
tv->tv_sec=(long)(st.wSecond+st.wMinute*60+st.wHour*3600);
|
||||||
|
tv->tv_usec=(long)(st.wMilliseconds*1000);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif /* HAVE_GETTIMEOFDAY */
|
||||||
|
|
||||||
|
// #define W32USLEEPDBG
|
||||||
|
|
||||||
|
#ifdef W32USLEEPDBG
|
||||||
|
|
||||||
|
# define DEBUG_QueryPerformanceCounter(arg) QueryPerformanceCounter(arg)
|
||||||
|
# define DEBUG_DisplayTimingInfo(start, stop, freq, us, has_highperf) \
|
||||||
|
do { \
|
||||||
|
unsigned long dt; \
|
||||||
|
dt = (unsigned long)((stop.QuadPart - start.QuadPart) * 1000 * 1000 \
|
||||||
|
/ freq.QuadPart); \
|
||||||
|
fprintf(stderr, \
|
||||||
|
"hpt:%i usleep usec:%lu sleep msec:%lu timed usec:%lu\n", \
|
||||||
|
has_highperf, us, ((us + 999) / 1000), dt); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
# define DEBUG_QueryPerformanceCounter(arg)
|
||||||
|
# define DEBUG_DisplayTimingInfo(start, stop, freq, us, has_highperf)
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(HAVE_USLEEP)
|
||||||
|
int usleep(unsigned int us)
|
||||||
|
{
|
||||||
|
int has_highperf;
|
||||||
|
LARGE_INTEGER freq,start,stop,loopend;
|
||||||
|
|
||||||
|
// workaround: although usleep is very precise if using
|
||||||
|
// high-performance-timers there are sometimes problems with
|
||||||
|
// verify - increasing the delay helps sometimes but not
|
||||||
|
// realiably. There must be some other problem. Maybe just
|
||||||
|
// with my test-hardware maybe in the code-base.
|
||||||
|
//// us=(unsigned long) (us*1.5);
|
||||||
|
|
||||||
|
has_highperf=QueryPerformanceFrequency(&freq);
|
||||||
|
|
||||||
|
//has_highperf=0; // debug
|
||||||
|
|
||||||
|
if (has_highperf) {
|
||||||
|
QueryPerformanceCounter(&start);
|
||||||
|
loopend.QuadPart=start.QuadPart+freq.QuadPart*us/(1000*1000);
|
||||||
|
do {
|
||||||
|
QueryPerformanceCounter(&stop);
|
||||||
|
} while (stop.QuadPart<=loopend.QuadPart);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
DEBUG_QueryPerformanceCounter(&start);
|
||||||
|
|
||||||
|
Sleep(1);
|
||||||
|
Sleep( (DWORD)((us+999)/1000) );
|
||||||
|
|
||||||
|
DEBUG_QueryPerformanceCounter(&stop);
|
||||||
|
}
|
||||||
|
|
||||||
|
DEBUG_DisplayTimingInfo(start, stop, freq, us, has_highperf);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif /* !HAVE_USLEEP */
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,363 @@
|
||||||
|
/*
|
||||||
|
* 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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#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;
|
||||||
|
}
|
|
@ -0,0 +1,47 @@
|
||||||
|
/*
|
||||||
|
* 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 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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef safemode_h
|
||||||
|
#define safemode_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* 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);
|
||||||
|
|
||||||
|
/* 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);
|
||||||
|
|
||||||
|
/* 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);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* safemode_h */
|
|
@ -0,0 +1,659 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
* Copyright (C) 2006 Christian Starkjohann
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Serial Interface emulation for USB programmer "AVR-Doper" in HID mode.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#if defined(HAVE_LIBUSB) || (defined(WIN32NATIVE) && defined(HAVE_LIBHID))
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
/* Numeric constants for 'reportType' parameters */
|
||||||
|
#define USB_HID_REPORT_TYPE_INPUT 1
|
||||||
|
#define USB_HID_REPORT_TYPE_OUTPUT 2
|
||||||
|
#define USB_HID_REPORT_TYPE_FEATURE 3
|
||||||
|
|
||||||
|
/* These are the error codes which can be returned by functions of this
|
||||||
|
* module.
|
||||||
|
*/
|
||||||
|
#define USB_ERROR_NONE 0
|
||||||
|
#define USB_ERROR_ACCESS 1
|
||||||
|
#define USB_ERROR_NOTFOUND 2
|
||||||
|
#define USB_ERROR_BUSY 16
|
||||||
|
#define USB_ERROR_IO 5
|
||||||
|
|
||||||
|
#define USB_VENDOR_ID 0x16c0
|
||||||
|
#define USB_PRODUCT_ID 0x05df
|
||||||
|
|
||||||
|
static int reportDataSizes[4] = {13, 29, 61, 125};
|
||||||
|
|
||||||
|
static unsigned char avrdoperRxBuffer[280]; /* buffer for receive data */
|
||||||
|
static int avrdoperRxLength = 0; /* amount of valid bytes in rx buffer */
|
||||||
|
static int avrdoperRxPosition = 0; /* amount of bytes already consumed in rx buffer */
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE) && defined(HAVE_LIBHID)
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
#include <setupapi.h>
|
||||||
|
|
||||||
|
#if defined(HAVE_DDK_HIDSDI_H)
|
||||||
|
# include <ddk/hidsdi.h>
|
||||||
|
#else
|
||||||
|
# include "my_ddk_hidsdi.h"
|
||||||
|
#endif
|
||||||
|
#include <ddk/hidpi.h>
|
||||||
|
|
||||||
|
#ifdef USB_DEBUG
|
||||||
|
#define DEBUG_PRINT(arg) printf arg
|
||||||
|
#else
|
||||||
|
#define DEBUG_PRINT(arg)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
static void convertUniToAscii(char *buffer)
|
||||||
|
{
|
||||||
|
unsigned short *uni = (void *)buffer;
|
||||||
|
char *ascii = buffer;
|
||||||
|
|
||||||
|
while(*uni != 0){
|
||||||
|
if(*uni >= 256){
|
||||||
|
*ascii++ = '?';
|
||||||
|
uni++;
|
||||||
|
}else{
|
||||||
|
*ascii++ = *uni++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*ascii++ = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usbOpenDevice(union filedescriptor *fdp, int vendor, char *vendorName,
|
||||||
|
int product, char *productName, int usesReportIDs)
|
||||||
|
{
|
||||||
|
GUID hidGuid; /* GUID for HID driver */
|
||||||
|
HDEVINFO deviceInfoList;
|
||||||
|
SP_DEVICE_INTERFACE_DATA deviceInfo;
|
||||||
|
SP_DEVICE_INTERFACE_DETAIL_DATA *deviceDetails = NULL;
|
||||||
|
DWORD size;
|
||||||
|
int i, openFlag = 0; /* may be FILE_FLAG_OVERLAPPED */
|
||||||
|
int errorCode = USB_ERROR_NOTFOUND;
|
||||||
|
HANDLE handle = INVALID_HANDLE_VALUE;
|
||||||
|
HIDD_ATTRIBUTES deviceAttributes;
|
||||||
|
|
||||||
|
HidD_GetHidGuid(&hidGuid);
|
||||||
|
deviceInfoList = SetupDiGetClassDevs(&hidGuid, NULL, NULL,
|
||||||
|
DIGCF_PRESENT | DIGCF_INTERFACEDEVICE);
|
||||||
|
deviceInfo.cbSize = sizeof(deviceInfo);
|
||||||
|
for(i=0;;i++){
|
||||||
|
if(handle != INVALID_HANDLE_VALUE){
|
||||||
|
CloseHandle(handle);
|
||||||
|
handle = INVALID_HANDLE_VALUE;
|
||||||
|
}
|
||||||
|
if(!SetupDiEnumDeviceInterfaces(deviceInfoList, 0, &hidGuid, i, &deviceInfo))
|
||||||
|
break; /* no more entries */
|
||||||
|
/* first do a dummy call just to determine the actual size required */
|
||||||
|
SetupDiGetDeviceInterfaceDetail(deviceInfoList, &deviceInfo, NULL, 0, &size, NULL);
|
||||||
|
if(deviceDetails != NULL)
|
||||||
|
free(deviceDetails);
|
||||||
|
deviceDetails = malloc(size);
|
||||||
|
deviceDetails->cbSize = sizeof(*deviceDetails);
|
||||||
|
/* this call is for real: */
|
||||||
|
SetupDiGetDeviceInterfaceDetail(deviceInfoList, &deviceInfo, deviceDetails,
|
||||||
|
size, &size, NULL);
|
||||||
|
DEBUG_PRINT(("checking HID path \"%s\"\n", deviceDetails->DevicePath));
|
||||||
|
/* attempt opening for R/W -- we don't care about devices which can't be accessed */
|
||||||
|
handle = CreateFile(deviceDetails->DevicePath, GENERIC_READ|GENERIC_WRITE,
|
||||||
|
FILE_SHARE_READ|FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
|
||||||
|
openFlag, NULL);
|
||||||
|
if(handle == INVALID_HANDLE_VALUE){
|
||||||
|
DEBUG_PRINT(("opening failed: %d\n", (int)GetLastError()));
|
||||||
|
/* errorCode = USB_ERROR_ACCESS; opening will always fail for mouse -- ignore */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
deviceAttributes.Size = sizeof(deviceAttributes);
|
||||||
|
HidD_GetAttributes(handle, &deviceAttributes);
|
||||||
|
DEBUG_PRINT(("device attributes: vid=%d pid=%d\n",
|
||||||
|
deviceAttributes.VendorID, deviceAttributes.ProductID));
|
||||||
|
if(deviceAttributes.VendorID != vendor || deviceAttributes.ProductID != product)
|
||||||
|
continue; /* ignore this device */
|
||||||
|
errorCode = USB_ERROR_NOTFOUND;
|
||||||
|
if(vendorName != NULL && productName != NULL){
|
||||||
|
char buffer[512];
|
||||||
|
if(!HidD_GetManufacturerString(handle, buffer, sizeof(buffer))){
|
||||||
|
DEBUG_PRINT(("error obtaining vendor name\n"));
|
||||||
|
errorCode = USB_ERROR_IO;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
convertUniToAscii(buffer);
|
||||||
|
DEBUG_PRINT(("vendorName = \"%s\"\n", buffer));
|
||||||
|
if(strcmp(vendorName, buffer) != 0)
|
||||||
|
continue;
|
||||||
|
if(!HidD_GetProductString(handle, buffer, sizeof(buffer))){
|
||||||
|
DEBUG_PRINT(("error obtaining product name\n"));
|
||||||
|
errorCode = USB_ERROR_IO;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
convertUniToAscii(buffer);
|
||||||
|
DEBUG_PRINT(("productName = \"%s\"\n", buffer));
|
||||||
|
if(strcmp(productName, buffer) != 0)
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
break; /* we have found the device we are looking for! */
|
||||||
|
}
|
||||||
|
SetupDiDestroyDeviceInfoList(deviceInfoList);
|
||||||
|
if(deviceDetails != NULL)
|
||||||
|
free(deviceDetails);
|
||||||
|
if(handle != INVALID_HANDLE_VALUE){
|
||||||
|
fdp->pfd = (void *)handle;
|
||||||
|
errorCode = 0;
|
||||||
|
}
|
||||||
|
return errorCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
static void usbCloseDevice(union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
CloseHandle((HANDLE)fdp->pfd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
static int usbSetReport(union filedescriptor *fdp, int reportType, char *buffer, int len)
|
||||||
|
{
|
||||||
|
HANDLE handle = (HANDLE)fdp->pfd;
|
||||||
|
BOOLEAN rval = 0;
|
||||||
|
DWORD bytesWritten;
|
||||||
|
|
||||||
|
switch(reportType){
|
||||||
|
case USB_HID_REPORT_TYPE_INPUT:
|
||||||
|
break;
|
||||||
|
case USB_HID_REPORT_TYPE_OUTPUT:
|
||||||
|
rval = WriteFile(handle, buffer, len, &bytesWritten, NULL);
|
||||||
|
break;
|
||||||
|
case USB_HID_REPORT_TYPE_FEATURE:
|
||||||
|
rval = HidD_SetFeature(handle, buffer, len);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return rval == 0 ? USB_ERROR_IO : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
static int usbGetReport(union filedescriptor *fdp, int reportType, int reportNumber,
|
||||||
|
char *buffer, int *len)
|
||||||
|
{
|
||||||
|
HANDLE handle = (HANDLE)fdp->pfd;
|
||||||
|
BOOLEAN rval = 0;
|
||||||
|
DWORD bytesRead;
|
||||||
|
|
||||||
|
switch(reportType){
|
||||||
|
case USB_HID_REPORT_TYPE_INPUT:
|
||||||
|
buffer[0] = reportNumber;
|
||||||
|
rval = ReadFile(handle, buffer, *len, &bytesRead, NULL);
|
||||||
|
if(rval)
|
||||||
|
*len = bytesRead;
|
||||||
|
break;
|
||||||
|
case USB_HID_REPORT_TYPE_OUTPUT:
|
||||||
|
break;
|
||||||
|
case USB_HID_REPORT_TYPE_FEATURE:
|
||||||
|
buffer[0] = reportNumber;
|
||||||
|
rval = HidD_GetFeature(handle, buffer, *len);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return rval == 0 ? USB_ERROR_IO : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
#else /* !(WIN32NATIVE && HAVE_LIBHID) */
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
#include <usb.h>
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
#define USBRQ_HID_GET_REPORT 0x01
|
||||||
|
#define USBRQ_HID_SET_REPORT 0x09
|
||||||
|
|
||||||
|
static int usesReportIDs;
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int usbGetStringAscii(usb_dev_handle *dev, int index, int langid, char *buf, int buflen)
|
||||||
|
{
|
||||||
|
char buffer[256];
|
||||||
|
int rval, i;
|
||||||
|
|
||||||
|
if((rval = usb_control_msg(dev, USB_ENDPOINT_IN, USB_REQ_GET_DESCRIPTOR,
|
||||||
|
(USB_DT_STRING << 8) + index, langid, buffer,
|
||||||
|
sizeof(buffer), 1000)) < 0)
|
||||||
|
return rval;
|
||||||
|
if(buffer[1] != USB_DT_STRING)
|
||||||
|
return 0;
|
||||||
|
if((unsigned char)buffer[0] < rval)
|
||||||
|
rval = (unsigned char)buffer[0];
|
||||||
|
rval /= 2;
|
||||||
|
/* lossy conversion to ISO Latin1 */
|
||||||
|
for(i=1;i<rval;i++){
|
||||||
|
if(i > buflen) /* destination buffer overflow */
|
||||||
|
break;
|
||||||
|
buf[i-1] = buffer[2 * i];
|
||||||
|
if(buffer[2 * i + 1] != 0) /* outside of ISO Latin1 range */
|
||||||
|
buf[i-1] = '?';
|
||||||
|
}
|
||||||
|
buf[i-1] = 0;
|
||||||
|
return i-1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usbOpenDevice(union filedescriptor *fdp, int vendor, char *vendorName,
|
||||||
|
int product, char *productName, int doReportIDs)
|
||||||
|
{
|
||||||
|
struct usb_bus *bus;
|
||||||
|
struct usb_device *dev;
|
||||||
|
usb_dev_handle *handle = NULL;
|
||||||
|
int errorCode = USB_ERROR_NOTFOUND;
|
||||||
|
static int didUsbInit = 0;
|
||||||
|
|
||||||
|
if(!didUsbInit){
|
||||||
|
usb_init();
|
||||||
|
didUsbInit = 1;
|
||||||
|
}
|
||||||
|
usb_find_busses();
|
||||||
|
usb_find_devices();
|
||||||
|
for(bus=usb_get_busses(); bus; bus=bus->next){
|
||||||
|
for(dev=bus->devices; dev; dev=dev->next){
|
||||||
|
if(dev->descriptor.idVendor == vendor && dev->descriptor.idProduct == product){
|
||||||
|
char string[256];
|
||||||
|
int len;
|
||||||
|
handle = usb_open(dev); /* we need to open the device in order to query strings */
|
||||||
|
if(!handle){
|
||||||
|
errorCode = USB_ERROR_ACCESS;
|
||||||
|
fprintf(stderr, "Warning: cannot open USB device: %s\n",
|
||||||
|
usb_strerror());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if(vendorName == NULL && productName == NULL){ /* name does not matter */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* now check whether the names match: */
|
||||||
|
len = usbGetStringAscii(handle, dev->descriptor.iManufacturer,
|
||||||
|
0x0409, string, sizeof(string));
|
||||||
|
if(len < 0){
|
||||||
|
errorCode = USB_ERROR_IO;
|
||||||
|
fprintf(stderr,
|
||||||
|
"Warning: cannot query manufacturer for device: %s\n",
|
||||||
|
usb_strerror());
|
||||||
|
}else{
|
||||||
|
errorCode = USB_ERROR_NOTFOUND;
|
||||||
|
/* fprintf(stderr, "seen device from vendor ->%s<-\n", string); */
|
||||||
|
if(strcmp(string, vendorName) == 0){
|
||||||
|
len = usbGetStringAscii(handle, dev->descriptor.iProduct,
|
||||||
|
0x0409, string, sizeof(string));
|
||||||
|
if(len < 0){
|
||||||
|
errorCode = USB_ERROR_IO;
|
||||||
|
fprintf(stderr,
|
||||||
|
"Warning: cannot query product for device: %s\n",
|
||||||
|
usb_strerror());
|
||||||
|
}else{
|
||||||
|
errorCode = USB_ERROR_NOTFOUND;
|
||||||
|
/* fprintf(stderr, "seen product ->%s<-\n", string); */
|
||||||
|
if(strcmp(string, productName) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
usb_close(handle);
|
||||||
|
handle = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(handle)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if(handle != NULL){
|
||||||
|
int rval, retries = 3;
|
||||||
|
if(usb_set_configuration(handle, 1)){
|
||||||
|
fprintf(stderr, "Warning: could not set configuration: %s\n",
|
||||||
|
usb_strerror());
|
||||||
|
}
|
||||||
|
/* now try to claim the interface and detach the kernel HID driver on
|
||||||
|
* linux and other operating systems which support the call.
|
||||||
|
*/
|
||||||
|
while((rval = usb_claim_interface(handle, 0)) != 0 && retries-- > 0){
|
||||||
|
#ifdef LIBUSB_HAS_DETACH_KERNEL_DRIVER_NP
|
||||||
|
if(usb_detach_kernel_driver_np(handle, 0) < 0){
|
||||||
|
fprintf(stderr, "Warning: could not detach kernel HID driver: %s\n",
|
||||||
|
usb_strerror());
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
if(rval != 0)
|
||||||
|
fprintf(stderr, "Warning: could not claim interface\n");
|
||||||
|
/* Continue anyway, even if we could not claim the interface. Control transfers
|
||||||
|
* should still work.
|
||||||
|
*/
|
||||||
|
errorCode = 0;
|
||||||
|
fdp->pfd = (void *)handle;
|
||||||
|
usesReportIDs = doReportIDs;
|
||||||
|
}
|
||||||
|
return errorCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static void usbCloseDevice(union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
usb_close((usb_dev_handle *)fdp->pfd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int usbSetReport(union filedescriptor *fdp, int reportType, char *buffer, int len)
|
||||||
|
{
|
||||||
|
int bytesSent;
|
||||||
|
|
||||||
|
if(!usesReportIDs){
|
||||||
|
buffer++; /* skip dummy report ID */
|
||||||
|
len--;
|
||||||
|
}
|
||||||
|
bytesSent = usb_control_msg((usb_dev_handle *)fdp->pfd, USB_TYPE_CLASS |
|
||||||
|
USB_RECIP_INTERFACE | USB_ENDPOINT_OUT, USBRQ_HID_SET_REPORT,
|
||||||
|
reportType << 8 | buffer[0], 0, buffer, len, 5000);
|
||||||
|
if(bytesSent != len){
|
||||||
|
if(bytesSent < 0)
|
||||||
|
fprintf(stderr, "Error sending message: %s\n", usb_strerror());
|
||||||
|
return USB_ERROR_IO;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int usbGetReport(union filedescriptor *fdp, int reportType, int reportNumber,
|
||||||
|
char *buffer, int *len)
|
||||||
|
{
|
||||||
|
int bytesReceived, maxLen = *len;
|
||||||
|
|
||||||
|
if(!usesReportIDs){
|
||||||
|
buffer++; /* make room for dummy report ID */
|
||||||
|
maxLen--;
|
||||||
|
}
|
||||||
|
bytesReceived = usb_control_msg((usb_dev_handle *)fdp->pfd, USB_TYPE_CLASS |
|
||||||
|
USB_RECIP_INTERFACE | USB_ENDPOINT_IN, USBRQ_HID_GET_REPORT,
|
||||||
|
reportType << 8 | reportNumber, 0, buffer, maxLen, 5000);
|
||||||
|
if(bytesReceived < 0){
|
||||||
|
fprintf(stderr, "Error sending message: %s\n", usb_strerror());
|
||||||
|
return USB_ERROR_IO;
|
||||||
|
}
|
||||||
|
*len = bytesReceived;
|
||||||
|
if(!usesReportIDs){
|
||||||
|
buffer[-1] = reportNumber; /* add dummy report ID */
|
||||||
|
len++;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* WIN32NATIVE */
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
/* ------------------------------------------------------------------------ */
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static void dumpBlock(char *prefix, unsigned char *buf, int len)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if(len <= 8){ /* more compact format for short blocks */
|
||||||
|
fprintf(stderr, "%s: %d bytes: ", prefix, len);
|
||||||
|
for(i = 0; i < len; i++){
|
||||||
|
fprintf(stderr, "%02x ", buf[i]);
|
||||||
|
}
|
||||||
|
fprintf(stderr, " \"");
|
||||||
|
for(i = 0; i < len; i++){
|
||||||
|
if(buf[i] >= 0x20 && buf[i] < 0x7f){
|
||||||
|
fputc(buf[i], stderr);
|
||||||
|
}else{
|
||||||
|
fputc('.', stderr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\"\n");
|
||||||
|
}else{
|
||||||
|
fprintf(stderr, "%s: %d bytes:\n", prefix, len);
|
||||||
|
while(len > 0){
|
||||||
|
for(i = 0; i < 16; i++){
|
||||||
|
if(i < len){
|
||||||
|
fprintf(stderr, "%02x ", buf[i]);
|
||||||
|
}else{
|
||||||
|
fprintf(stderr, " ");
|
||||||
|
}
|
||||||
|
if(i == 7)
|
||||||
|
fputc(' ', stderr);
|
||||||
|
}
|
||||||
|
fprintf(stderr, " \"");
|
||||||
|
for(i = 0; i < 16; i++){
|
||||||
|
if(i < len){
|
||||||
|
if(buf[i] >= 0x20 && buf[i] < 0x7f){
|
||||||
|
fputc(buf[i], stderr);
|
||||||
|
}else{
|
||||||
|
fputc('.', stderr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\"\n");
|
||||||
|
buf += 16;
|
||||||
|
len -= 16;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static char *usbErrorText(int usbErrno)
|
||||||
|
{
|
||||||
|
static char buffer[32];
|
||||||
|
|
||||||
|
switch(usbErrno){
|
||||||
|
case USB_ERROR_NONE: return "Success.";
|
||||||
|
case USB_ERROR_ACCESS: return "Access denied.";
|
||||||
|
case USB_ERROR_NOTFOUND:return "Device not found.";
|
||||||
|
case USB_ERROR_BUSY: return "Device is busy.";
|
||||||
|
case USB_ERROR_IO: return "I/O Error.";
|
||||||
|
default:
|
||||||
|
sprintf(buffer, "Unknown error %d.", usbErrno);
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int avrdoper_open(char *port, long baud, union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
int rval;
|
||||||
|
char *vname = "obdev.at";
|
||||||
|
char *devname = "AVR-Doper";
|
||||||
|
|
||||||
|
rval = usbOpenDevice(fdp, USB_VENDOR_ID, vname, USB_PRODUCT_ID, devname, 1);
|
||||||
|
if(rval != 0){
|
||||||
|
fprintf(stderr, "%s: avrdoper_open(): %s\n", progname, usbErrorText(rval));
|
||||||
|
exit(1);
|
||||||
|
//return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static void avrdoper_close(union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
usbCloseDevice(fdp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int chooseDataSize(int len)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for(i = 0; i < sizeof(reportDataSizes)/sizeof(reportDataSizes[0]); i++){
|
||||||
|
if(reportDataSizes[i] >= len)
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
return i - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int avrdoper_send(union filedescriptor *fdp, unsigned char *buf, size_t buflen)
|
||||||
|
{
|
||||||
|
if(verbose > 3)
|
||||||
|
dumpBlock("Send", buf, buflen);
|
||||||
|
while(buflen > 0){
|
||||||
|
unsigned char buffer[256];
|
||||||
|
int rval, lenIndex = chooseDataSize(buflen);
|
||||||
|
int thisLen = buflen > reportDataSizes[lenIndex] ?
|
||||||
|
reportDataSizes[lenIndex] : buflen;
|
||||||
|
buffer[0] = lenIndex + 1; /* report ID */
|
||||||
|
buffer[1] = thisLen;
|
||||||
|
memcpy(buffer + 2, buf, thisLen);
|
||||||
|
if(verbose > 3)
|
||||||
|
fprintf(stderr, "Sending %d bytes data chunk\n", thisLen);
|
||||||
|
rval = usbSetReport(fdp, USB_HID_REPORT_TYPE_FEATURE, (char *)buffer,
|
||||||
|
reportDataSizes[lenIndex] + 2);
|
||||||
|
if(rval != 0){
|
||||||
|
fprintf(stderr, "%s: avrdoper_send(): %s\n", progname, usbErrorText(rval));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
buflen -= thisLen;
|
||||||
|
buf += thisLen;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static void avrdoperFillBuffer(union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
int bytesPending = reportDataSizes[1]; /* guess how much data is buffered in device */
|
||||||
|
|
||||||
|
avrdoperRxPosition = avrdoperRxLength = 0;
|
||||||
|
while(bytesPending > 0){
|
||||||
|
int len, usbErr, lenIndex = chooseDataSize(bytesPending);
|
||||||
|
unsigned char buffer[128];
|
||||||
|
len = sizeof(avrdoperRxBuffer) - avrdoperRxLength; /* bytes remaining */
|
||||||
|
if(reportDataSizes[lenIndex] + 2 > len) /* requested data would not fit into buffer */
|
||||||
|
break;
|
||||||
|
len = reportDataSizes[lenIndex] + 2;
|
||||||
|
usbErr = usbGetReport(fdp, USB_HID_REPORT_TYPE_FEATURE, lenIndex + 1,
|
||||||
|
(char *)buffer, &len);
|
||||||
|
if(usbErr != 0){
|
||||||
|
fprintf(stderr, "%s: avrdoperFillBuffer(): %s\n", progname, usbErrorText(usbErr));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
if(verbose > 3)
|
||||||
|
fprintf(stderr, "Received %d bytes data chunk of total %d\n", len - 2, buffer[1]);
|
||||||
|
len -= 2; /* compensate for report ID and length byte */
|
||||||
|
bytesPending = buffer[1] - len; /* amount still buffered */
|
||||||
|
if(len > buffer[1]) /* cut away padding */
|
||||||
|
len = buffer[1];
|
||||||
|
if(avrdoperRxLength + len > sizeof(avrdoperRxBuffer)){
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: avrdoperFillBuffer(): internal error: buffer overflow\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
memcpy(avrdoperRxBuffer + avrdoperRxLength, buffer + 2, len);
|
||||||
|
avrdoperRxLength += len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int avrdoper_recv(union filedescriptor *fdp, unsigned char *buf, size_t buflen)
|
||||||
|
{
|
||||||
|
unsigned char *p = buf;
|
||||||
|
int remaining = buflen;
|
||||||
|
|
||||||
|
while(remaining > 0){
|
||||||
|
int len, available = avrdoperRxLength - avrdoperRxPosition;
|
||||||
|
if(available <= 0){ /* buffer is empty */
|
||||||
|
avrdoperFillBuffer(fdp);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
len = remaining < available ? remaining : available;
|
||||||
|
memcpy(p, avrdoperRxBuffer + avrdoperRxPosition, len);
|
||||||
|
p += len;
|
||||||
|
remaining -= len;
|
||||||
|
avrdoperRxPosition += len;
|
||||||
|
}
|
||||||
|
if(verbose > 3)
|
||||||
|
dumpBlock("Receive", buf, buflen);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int avrdoper_drain(union filedescriptor *fdp, int display)
|
||||||
|
{
|
||||||
|
do{
|
||||||
|
avrdoperFillBuffer(fdp);
|
||||||
|
}while(avrdoperRxLength > 0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static int avrdoper_set_dtr_rts(union filedescriptor *fdp, int is_on)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: AVR-Doper doesn't support DTR/RTS setting\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
struct serial_device avrdoper_serdev =
|
||||||
|
{
|
||||||
|
.open = avrdoper_open,
|
||||||
|
.close = avrdoper_close,
|
||||||
|
.send = avrdoper_send,
|
||||||
|
.recv = avrdoper_recv,
|
||||||
|
.drain = avrdoper_drain,
|
||||||
|
.set_dtr_rts = avrdoper_set_dtr_rts,
|
||||||
|
.flags = SERDEV_FL_NONE,
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* defined(HAVE_LIBUSB) || (defined(WIN32NATIVE) && defined(HAVE_LIBHID)) */
|
|
@ -0,0 +1,503 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Posix serial interface for avrdude.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netdb.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <termios.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
long serial_recv_timeout = 5000; /* ms */
|
||||||
|
|
||||||
|
struct baud_mapping {
|
||||||
|
long baud;
|
||||||
|
speed_t speed;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* There are a lot more baud rates we could handle, but what's the point? */
|
||||||
|
|
||||||
|
static struct baud_mapping baud_lookup_table [] = {
|
||||||
|
{ 1200, B1200 },
|
||||||
|
{ 2400, B2400 },
|
||||||
|
{ 4800, B4800 },
|
||||||
|
{ 9600, B9600 },
|
||||||
|
{ 19200, B19200 },
|
||||||
|
{ 38400, B38400 },
|
||||||
|
{ 57600, B57600 },
|
||||||
|
{ 115200, B115200 },
|
||||||
|
{ 230400, B230400 },
|
||||||
|
{ 0, 0 } /* Terminator. */
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct termios original_termios;
|
||||||
|
static int saved_original_termios;
|
||||||
|
|
||||||
|
static speed_t serial_baud_lookup(long baud)
|
||||||
|
{
|
||||||
|
struct baud_mapping *map = baud_lookup_table;
|
||||||
|
|
||||||
|
while (map->baud) {
|
||||||
|
if (map->baud == baud)
|
||||||
|
return map->speed;
|
||||||
|
map++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If a non-standard BAUD rate is used, issue
|
||||||
|
* a warning (if we are verbose) and return the raw rate
|
||||||
|
*/
|
||||||
|
if (verbose > 0)
|
||||||
|
fprintf(stderr, "%s: serial_baud_lookup(): Using non-standard baud rate: %ld",
|
||||||
|
progname, baud);
|
||||||
|
|
||||||
|
return baud;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ser_setspeed(union filedescriptor *fd, long baud)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
struct termios termios;
|
||||||
|
speed_t speed = serial_baud_lookup (baud);
|
||||||
|
|
||||||
|
if (!isatty(fd->ifd))
|
||||||
|
return -ENOTTY;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* initialize terminal modes
|
||||||
|
*/
|
||||||
|
rc = tcgetattr(fd->ifd, &termios);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: ser_setspeed(): tcgetattr() failed",
|
||||||
|
progname);
|
||||||
|
return -errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* copy termios for ser_close if we haven't already
|
||||||
|
*/
|
||||||
|
if (! saved_original_termios++) {
|
||||||
|
original_termios = termios;
|
||||||
|
}
|
||||||
|
|
||||||
|
termios.c_iflag = IGNBRK;
|
||||||
|
termios.c_oflag = 0;
|
||||||
|
termios.c_lflag = 0;
|
||||||
|
termios.c_cflag = (CS8 | CREAD | CLOCAL);
|
||||||
|
termios.c_cc[VMIN] = 1;
|
||||||
|
termios.c_cc[VTIME] = 0;
|
||||||
|
|
||||||
|
cfsetospeed(&termios, speed);
|
||||||
|
cfsetispeed(&termios, speed);
|
||||||
|
|
||||||
|
rc = tcsetattr(fd->ifd, TCSANOW, &termios);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: ser_setspeed(): tcsetattr() failed\n",
|
||||||
|
progname);
|
||||||
|
return -errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Everything is now set up for a local line without modem control
|
||||||
|
* or flow control, so clear O_NONBLOCK again.
|
||||||
|
*/
|
||||||
|
rc = fcntl(fd->ifd, F_GETFL, 0);
|
||||||
|
if (rc != -1)
|
||||||
|
fcntl(fd->ifd, F_SETFL, rc & ~O_NONBLOCK);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Given a port description of the form <host>:<port>, open a TCP
|
||||||
|
* connection to the specified destination, which is assumed to be a
|
||||||
|
* terminal/console server with serial parameters configured
|
||||||
|
* appropriately (e. g. 115200-8-N-1 for a STK500.)
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
net_open(const char *port, union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
char *hstr, *pstr, *end;
|
||||||
|
unsigned int pnum;
|
||||||
|
int fd;
|
||||||
|
struct sockaddr_in sockaddr;
|
||||||
|
struct hostent *hp;
|
||||||
|
|
||||||
|
if ((hstr = strdup(port)) == NULL) {
|
||||||
|
fprintf(stderr, "%s: net_open(): Out of memory!\n",
|
||||||
|
progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (((pstr = strchr(hstr, ':')) == NULL) || (pstr == hstr)) {
|
||||||
|
fprintf(stderr, "%s: net_open(): Mangled host:port string \"%s\"\n",
|
||||||
|
progname, hstr);
|
||||||
|
free(hstr);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Terminate the host section of the description.
|
||||||
|
*/
|
||||||
|
*pstr++ = '\0';
|
||||||
|
|
||||||
|
pnum = strtoul(pstr, &end, 10);
|
||||||
|
|
||||||
|
if ((*pstr == '\0') || (*end != '\0') || (pnum == 0) || (pnum > 65535)) {
|
||||||
|
fprintf(stderr, "%s: net_open(): Bad port number \"%s\"\n",
|
||||||
|
progname, pstr);
|
||||||
|
free(hstr);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((hp = gethostbyname(hstr)) == NULL) {
|
||||||
|
fprintf(stderr, "%s: net_open(): unknown host \"%s\"\n",
|
||||||
|
progname, hstr);
|
||||||
|
free(hstr);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(hstr);
|
||||||
|
|
||||||
|
if ((fd = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
|
||||||
|
fprintf(stderr, "%s: net_open(): Cannot open socket: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(&sockaddr, 0, sizeof(struct sockaddr_in));
|
||||||
|
sockaddr.sin_family = AF_INET;
|
||||||
|
sockaddr.sin_port = htons(pnum);
|
||||||
|
memcpy(&(sockaddr.sin_addr.s_addr), hp->h_addr, sizeof(struct in_addr));
|
||||||
|
|
||||||
|
if (connect(fd, (struct sockaddr *)&sockaddr, sizeof(sockaddr))) {
|
||||||
|
fprintf(stderr, "%s: net_open(): Connect failed: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fdp->ifd = fd;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_set_dtr_rts(union filedescriptor *fdp, int is_on)
|
||||||
|
{
|
||||||
|
unsigned int ctl;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
r = ioctl(fdp->ifd, TIOCMGET, &ctl);
|
||||||
|
if (r < 0) {
|
||||||
|
perror("ioctl(\"TIOCMGET\")");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_on) {
|
||||||
|
/* Set DTR and RTS */
|
||||||
|
ctl |= (TIOCM_DTR | TIOCM_RTS);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
/* Clear DTR and RTS */
|
||||||
|
ctl &= ~(TIOCM_DTR | TIOCM_RTS);
|
||||||
|
}
|
||||||
|
|
||||||
|
r = ioctl(fdp->ifd, TIOCMSET, &ctl);
|
||||||
|
if (r < 0) {
|
||||||
|
perror("ioctl(\"TIOCMSET\")");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ser_open(char * port, long baud, union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the port is of the form "net:<host>:<port>", then
|
||||||
|
* handle it as a TCP connection to a terminal server.
|
||||||
|
*/
|
||||||
|
if (strncmp(port, "net:", strlen("net:")) == 0) {
|
||||||
|
return net_open(port + strlen("net:"), fdp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* open the serial port
|
||||||
|
*/
|
||||||
|
fd = open(port, O_RDWR | O_NOCTTY | O_NONBLOCK);
|
||||||
|
if (fd < 0) {
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't open device \"%s\": %s\n",
|
||||||
|
progname, port, strerror(errno));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fdp->ifd = fd;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set serial line attributes
|
||||||
|
*/
|
||||||
|
rc = ser_setspeed(fdp, baud);
|
||||||
|
if (rc) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_open(): can't set attributes for device \"%s\": %s\n",
|
||||||
|
progname, port, strerror(-rc));
|
||||||
|
close(fd);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void ser_close(union filedescriptor *fd)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* restore original termios settings from ser_open
|
||||||
|
*/
|
||||||
|
if (saved_original_termios) {
|
||||||
|
int rc = tcsetattr(fd->ifd, TCSANOW | TCSADRAIN, &original_termios);
|
||||||
|
if (rc) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_close(): can't reset attributes for device: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
}
|
||||||
|
saved_original_termios = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
close(fd->ifd);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_send(union filedescriptor *fd, unsigned char * buf, size_t buflen)
|
||||||
|
{
|
||||||
|
struct timeval timeout, to2;
|
||||||
|
int rc;
|
||||||
|
unsigned char * p = buf;
|
||||||
|
size_t len = buflen;
|
||||||
|
|
||||||
|
if (!len)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (verbose > 3)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Send: ", progname);
|
||||||
|
|
||||||
|
while (buflen) {
|
||||||
|
unsigned char c = *buf;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
|
||||||
|
buf++;
|
||||||
|
buflen--;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
timeout.tv_sec = 0;
|
||||||
|
timeout.tv_usec = 500000;
|
||||||
|
to2 = timeout;
|
||||||
|
|
||||||
|
while (len) {
|
||||||
|
rc = write(fd->ifd, p, (len > 1024) ? 1024 : len);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: ser_send(): write error: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
p += rc;
|
||||||
|
len -= rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_recv(union filedescriptor *fd, unsigned char * buf, size_t buflen)
|
||||||
|
{
|
||||||
|
struct timeval timeout, to2;
|
||||||
|
fd_set rfds;
|
||||||
|
int nfds;
|
||||||
|
int rc;
|
||||||
|
unsigned char * p = buf;
|
||||||
|
size_t len = 0;
|
||||||
|
|
||||||
|
timeout.tv_sec = serial_recv_timeout / 1000L;
|
||||||
|
timeout.tv_usec = (serial_recv_timeout % 1000L) * 1000;
|
||||||
|
to2 = timeout;
|
||||||
|
|
||||||
|
while (len < buflen) {
|
||||||
|
reselect:
|
||||||
|
FD_ZERO(&rfds);
|
||||||
|
FD_SET(fd->ifd, &rfds);
|
||||||
|
|
||||||
|
nfds = select(fd->ifd + 1, &rfds, NULL, NULL, &to2);
|
||||||
|
if (nfds == 0) {
|
||||||
|
if (verbose > 1)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_recv(): programmer is not responding\n",
|
||||||
|
progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else if (nfds == -1) {
|
||||||
|
if (errno == EINTR || errno == EAGAIN) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_recv(): programmer is not responding,reselecting\n",
|
||||||
|
progname);
|
||||||
|
goto reselect;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, "%s: ser_recv(): select(): %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = read(fd->ifd, p, (buflen - len > 1024) ? 1024 : buflen - len);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: ser_recv(): read error: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
p += rc;
|
||||||
|
len += rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
p = buf;
|
||||||
|
|
||||||
|
if (verbose > 3)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Recv: ", progname);
|
||||||
|
|
||||||
|
while (len) {
|
||||||
|
unsigned char c = *p;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
|
||||||
|
p++;
|
||||||
|
len--;
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_drain(union filedescriptor *fd, int display)
|
||||||
|
{
|
||||||
|
struct timeval timeout;
|
||||||
|
fd_set rfds;
|
||||||
|
int nfds;
|
||||||
|
int rc;
|
||||||
|
unsigned char buf;
|
||||||
|
|
||||||
|
timeout.tv_sec = 0;
|
||||||
|
timeout.tv_usec = 250000;
|
||||||
|
|
||||||
|
if (display) {
|
||||||
|
fprintf(stderr, "drain>");
|
||||||
|
}
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
FD_ZERO(&rfds);
|
||||||
|
FD_SET(fd->ifd, &rfds);
|
||||||
|
|
||||||
|
reselect:
|
||||||
|
nfds = select(fd->ifd + 1, &rfds, NULL, NULL, &timeout);
|
||||||
|
if (nfds == 0) {
|
||||||
|
if (display) {
|
||||||
|
fprintf(stderr, "<drain\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else if (nfds == -1) {
|
||||||
|
if (errno == EINTR) {
|
||||||
|
goto reselect;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, "%s: ser_drain(): select(): %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = read(fd->ifd, &buf, 1);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: ser_drain(): read error: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
if (display) {
|
||||||
|
fprintf(stderr, "%02x ", buf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct serial_device serial_serdev =
|
||||||
|
{
|
||||||
|
.open = ser_open,
|
||||||
|
.setspeed = ser_setspeed,
|
||||||
|
.close = ser_close,
|
||||||
|
.send = ser_send,
|
||||||
|
.recv = ser_recv,
|
||||||
|
.drain = ser_drain,
|
||||||
|
.set_dtr_rts = ser_set_dtr_rts,
|
||||||
|
.flags = SERDEV_FL_CANSETSPEED,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct serial_device *serdev = &serial_serdev;
|
||||||
|
|
||||||
|
#endif /* WIN32NATIVE */
|
|
@ -0,0 +1,408 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003, 2004 Martin J. Thomas <mthomas@rhrk.uni-kl.de>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Native Win32 serial interface for avrdude.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <ctype.h> /* for isprint */
|
||||||
|
|
||||||
|
#include "serial.h"
|
||||||
|
|
||||||
|
long serial_recv_timeout = 5000; /* ms */
|
||||||
|
|
||||||
|
#define W32SERBUFSIZE 1024
|
||||||
|
|
||||||
|
struct baud_mapping {
|
||||||
|
long baud;
|
||||||
|
DWORD speed;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* HANDLE hComPort=INVALID_HANDLE_VALUE; */
|
||||||
|
|
||||||
|
static struct baud_mapping baud_lookup_table [] = {
|
||||||
|
{ 1200, CBR_1200 },
|
||||||
|
{ 2400, CBR_2400 },
|
||||||
|
{ 4800, CBR_4800 },
|
||||||
|
{ 9600, CBR_9600 },
|
||||||
|
{ 19200, CBR_19200 },
|
||||||
|
{ 38400, CBR_38400 },
|
||||||
|
{ 57600, CBR_57600 },
|
||||||
|
{ 115200, CBR_115200 },
|
||||||
|
{ 0, 0 } /* Terminator. */
|
||||||
|
};
|
||||||
|
|
||||||
|
static DWORD serial_baud_lookup(long baud)
|
||||||
|
{
|
||||||
|
struct baud_mapping *map = baud_lookup_table;
|
||||||
|
|
||||||
|
while (map->baud) {
|
||||||
|
if (map->baud == baud)
|
||||||
|
return map->speed;
|
||||||
|
map++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If a non-standard BAUD rate is used, issue
|
||||||
|
* a warning (if we are verbose) and return the raw rate
|
||||||
|
*/
|
||||||
|
if (verbose > 0)
|
||||||
|
fprintf(stderr, "%s: serial_baud_lookup(): Using non-standard baud rate: %ld",
|
||||||
|
progname, baud);
|
||||||
|
|
||||||
|
return baud;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static BOOL serial_w32SetTimeOut(HANDLE hComPort, DWORD timeout) // in ms
|
||||||
|
{
|
||||||
|
COMMTIMEOUTS ctmo;
|
||||||
|
ZeroMemory (&ctmo, sizeof(COMMTIMEOUTS));
|
||||||
|
ctmo.ReadIntervalTimeout = timeout;
|
||||||
|
ctmo.ReadTotalTimeoutMultiplier = timeout;
|
||||||
|
ctmo.ReadTotalTimeoutConstant = timeout;
|
||||||
|
|
||||||
|
return SetCommTimeouts(hComPort, &ctmo);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ser_setspeed(union filedescriptor *fd, long baud)
|
||||||
|
{
|
||||||
|
DCB dcb;
|
||||||
|
HANDLE hComPort = (HANDLE)fd->pfd;
|
||||||
|
|
||||||
|
ZeroMemory (&dcb, sizeof(DCB));
|
||||||
|
dcb.DCBlength = sizeof(DCB);
|
||||||
|
dcb.BaudRate = serial_baud_lookup (baud);
|
||||||
|
dcb.fBinary = 1;
|
||||||
|
dcb.fDtrControl = DTR_CONTROL_DISABLE;
|
||||||
|
dcb.fRtsControl = RTS_CONTROL_DISABLE;
|
||||||
|
dcb.ByteSize = 8;
|
||||||
|
dcb.Parity = NOPARITY;
|
||||||
|
dcb.StopBits = ONESTOPBIT;
|
||||||
|
|
||||||
|
if (!SetCommState(hComPort, &dcb))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_open(char * port, long baud, union filedescriptor *fdp)
|
||||||
|
{
|
||||||
|
LPVOID lpMsgBuf;
|
||||||
|
HANDLE hComPort=INVALID_HANDLE_VALUE;
|
||||||
|
char *newname = 0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the port is of the form "net:<host>:<port>", then
|
||||||
|
* handle it as a TCP connection to a terminal server.
|
||||||
|
*
|
||||||
|
* This is curently not implemented for Win32.
|
||||||
|
*/
|
||||||
|
if (strncmp(port, "net:", strlen("net:")) == 0) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_open(): network connects are currently not"
|
||||||
|
"implemented for Win32 environments\n",
|
||||||
|
progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strncasecmp(port, "com", strlen("com")) == 0) {
|
||||||
|
|
||||||
|
// prepend "\\\\.\\" to name, required for port # >= 10
|
||||||
|
newname = malloc(strlen("\\\\.\\") + strlen(port) + 1);
|
||||||
|
|
||||||
|
if (newname == 0) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_open(): out of memory\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
strcpy(newname, "\\\\.\\");
|
||||||
|
strcat(newname, port);
|
||||||
|
|
||||||
|
port = newname;
|
||||||
|
}
|
||||||
|
|
||||||
|
hComPort = CreateFile(port, GENERIC_READ | GENERIC_WRITE, 0, NULL,
|
||||||
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
||||||
|
|
||||||
|
if (hComPort == INVALID_HANDLE_VALUE) {
|
||||||
|
FormatMessage(
|
||||||
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL,
|
||||||
|
GetLastError(),
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0,
|
||||||
|
NULL);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't open device \"%s\": %s\n",
|
||||||
|
progname, port, (char*)lpMsgBuf);
|
||||||
|
LocalFree( lpMsgBuf );
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!SetupComm(hComPort, W32SERBUFSIZE, W32SERBUFSIZE))
|
||||||
|
{
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't set buffers for \"%s\"\n",
|
||||||
|
progname, port);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fdp->pfd = (void *)hComPort;
|
||||||
|
if (ser_setspeed(fdp, baud) != 0)
|
||||||
|
{
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't set com-state for \"%s\"\n",
|
||||||
|
progname, port);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!serial_w32SetTimeOut(hComPort,0))
|
||||||
|
{
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't set initial timeout for \"%s\"\n",
|
||||||
|
progname, port);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newname != 0) {
|
||||||
|
free(newname);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void ser_close(union filedescriptor *fd)
|
||||||
|
{
|
||||||
|
HANDLE hComPort=(HANDLE)fd->pfd;
|
||||||
|
if (hComPort != INVALID_HANDLE_VALUE)
|
||||||
|
CloseHandle (hComPort);
|
||||||
|
|
||||||
|
hComPort = INVALID_HANDLE_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ser_set_dtr_rts(union filedescriptor *fd, int is_on)
|
||||||
|
{
|
||||||
|
HANDLE hComPort=(HANDLE)fd->pfd;
|
||||||
|
|
||||||
|
if (is_on) {
|
||||||
|
EscapeCommFunction(hComPort, SETDTR);
|
||||||
|
EscapeCommFunction(hComPort, SETRTS);
|
||||||
|
} else {
|
||||||
|
EscapeCommFunction(hComPort, CLRDTR);
|
||||||
|
EscapeCommFunction(hComPort, CLRRTS);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_send(union filedescriptor *fd, unsigned char * buf, size_t buflen)
|
||||||
|
{
|
||||||
|
size_t len = buflen;
|
||||||
|
unsigned char c='\0';
|
||||||
|
DWORD written;
|
||||||
|
unsigned char * b = buf;
|
||||||
|
|
||||||
|
HANDLE hComPort=(HANDLE)fd->pfd;
|
||||||
|
|
||||||
|
if (hComPort == INVALID_HANDLE_VALUE) {
|
||||||
|
fprintf(stderr, "%s: ser_send(): port not open\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!len)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (verbose > 3)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Send: ", progname);
|
||||||
|
|
||||||
|
while (len) {
|
||||||
|
c = *b;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
b++;
|
||||||
|
len--;
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
serial_w32SetTimeOut(hComPort,500);
|
||||||
|
|
||||||
|
if (!WriteFile (hComPort, buf, buflen, &written, NULL)) {
|
||||||
|
fprintf(stderr, "%s: ser_send(): write error: %s\n",
|
||||||
|
progname, "sorry no info avail"); // TODO
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (written != buflen) {
|
||||||
|
fprintf(stderr, "%s: ser_send(): size/send mismatch\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_recv(union filedescriptor *fd, unsigned char * buf, size_t buflen)
|
||||||
|
{
|
||||||
|
unsigned char c;
|
||||||
|
unsigned char * p = buf;
|
||||||
|
DWORD read;
|
||||||
|
|
||||||
|
HANDLE hComPort=(HANDLE)fd->pfd;
|
||||||
|
|
||||||
|
if (hComPort == INVALID_HANDLE_VALUE) {
|
||||||
|
fprintf(stderr, "%s: ser_read(): port not open\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
serial_w32SetTimeOut(hComPort, serial_recv_timeout);
|
||||||
|
|
||||||
|
if (!ReadFile(hComPort, buf, buflen, &read, NULL)) {
|
||||||
|
LPVOID lpMsgBuf;
|
||||||
|
FormatMessage(
|
||||||
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL,
|
||||||
|
GetLastError(),
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0,
|
||||||
|
NULL );
|
||||||
|
fprintf(stderr, "%s: ser_recv(): read error: %s\n",
|
||||||
|
progname, (char*)lpMsgBuf);
|
||||||
|
LocalFree( lpMsgBuf );
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
p = buf;
|
||||||
|
|
||||||
|
if (verbose > 3)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Recv: ", progname);
|
||||||
|
|
||||||
|
while (read) {
|
||||||
|
c = *p;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
|
||||||
|
p++;
|
||||||
|
read--;
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int ser_drain(union filedescriptor *fd, int display)
|
||||||
|
{
|
||||||
|
// int rc;
|
||||||
|
unsigned char buf[10];
|
||||||
|
BOOL readres;
|
||||||
|
DWORD read;
|
||||||
|
|
||||||
|
HANDLE hComPort=(HANDLE)fd->pfd;
|
||||||
|
|
||||||
|
if (hComPort == INVALID_HANDLE_VALUE) {
|
||||||
|
fprintf(stderr, "%s: ser_drain(): port not open\n",
|
||||||
|
progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
serial_w32SetTimeOut(hComPort,250);
|
||||||
|
|
||||||
|
if (display) {
|
||||||
|
fprintf(stderr, "drain>");
|
||||||
|
}
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
readres=ReadFile(hComPort, buf, 1, &read, NULL);
|
||||||
|
if (!readres) {
|
||||||
|
LPVOID lpMsgBuf;
|
||||||
|
FormatMessage(
|
||||||
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL,
|
||||||
|
GetLastError(),
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0,
|
||||||
|
NULL );
|
||||||
|
fprintf(stderr, "%s: ser_drain(): read error: %s\n",
|
||||||
|
progname, (char*)lpMsgBuf);
|
||||||
|
LocalFree( lpMsgBuf );
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (read) { // data avail
|
||||||
|
if (display) fprintf(stderr, "%02x ", buf[0]);
|
||||||
|
}
|
||||||
|
else { // no more data
|
||||||
|
if (display) fprintf(stderr, "<drain\n");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} // while
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct serial_device serial_serdev =
|
||||||
|
{
|
||||||
|
.open = ser_open,
|
||||||
|
.setspeed = ser_setspeed,
|
||||||
|
.close = ser_close,
|
||||||
|
.send = ser_send,
|
||||||
|
.recv = ser_recv,
|
||||||
|
.drain = ser_drain,
|
||||||
|
.set_dtr_rts = ser_set_dtr_rts,
|
||||||
|
.flags = SERDEV_FL_CANSETSPEED,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct serial_device *serdev = &serial_serdev;
|
||||||
|
|
||||||
|
#endif /* WIN32NATIVE */
|
|
@ -0,0 +1,37 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000, 2001, 2002, 2003 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2005 Michael Holzt <kju-avr@fqdn.org>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef serbb_h
|
||||||
|
#define serbb_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void serbb_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,313 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000, 2001, 2002, 2003 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2005 Michael Holzt <kju-avr@fqdn.org>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Posix serial bitbanging interface for avrdude.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
#include <termios.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "bitbang.h"
|
||||||
|
|
||||||
|
#undef DEBUG
|
||||||
|
|
||||||
|
static struct termios oldmode;
|
||||||
|
|
||||||
|
/*
|
||||||
|
serial port/pin mapping
|
||||||
|
|
||||||
|
1 cd <-
|
||||||
|
2 (rxd) <-
|
||||||
|
3 txd ->
|
||||||
|
4 dtr ->
|
||||||
|
5 GND
|
||||||
|
6 dsr <-
|
||||||
|
7 rts ->
|
||||||
|
8 cts <-
|
||||||
|
9 ri <-
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define DB9PINS 9
|
||||||
|
|
||||||
|
static int serregbits[DB9PINS + 1] =
|
||||||
|
{ 0, TIOCM_CD, 0, 0, TIOCM_DTR, 0, TIOCM_DSR, TIOCM_RTS, TIOCM_CTS, TIOCM_RI };
|
||||||
|
|
||||||
|
#ifdef DEBUG
|
||||||
|
static char *serpins[DB9PINS + 1] =
|
||||||
|
{ "NONE", "CD", "RXD", "TXD", "DTR", "GND", "DSR", "RTS", "CTS", "RI" };
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static int serbb_setpin(PROGRAMMER * pgm, int pin, int value)
|
||||||
|
{
|
||||||
|
unsigned int ctl;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
if (pin & PIN_INVERSE)
|
||||||
|
{
|
||||||
|
value = !value;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( pin < 1 || pin > DB9PINS )
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
#ifdef DEBUG
|
||||||
|
printf("%s to %d\n",serpins[pin],value);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
switch ( pin )
|
||||||
|
{
|
||||||
|
case 3: /* txd */
|
||||||
|
r = ioctl(pgm->fd.ifd, value ? TIOCSBRK : TIOCCBRK, 0);
|
||||||
|
if (r < 0) {
|
||||||
|
perror("ioctl(\"TIOCxBRK\")");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4: /* dtr */
|
||||||
|
case 7: /* rts */
|
||||||
|
r = ioctl(pgm->fd.ifd, TIOCMGET, &ctl);
|
||||||
|
if (r < 0) {
|
||||||
|
perror("ioctl(\"TIOCMGET\")");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if ( value )
|
||||||
|
ctl |= serregbits[pin];
|
||||||
|
else
|
||||||
|
ctl &= ~(serregbits[pin]);
|
||||||
|
r = ioctl(pgm->fd.ifd, TIOCMSET, &ctl);
|
||||||
|
if (r < 0) {
|
||||||
|
perror("ioctl(\"TIOCMSET\")");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: /* impossible */
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int serbb_getpin(PROGRAMMER * pgm, int pin)
|
||||||
|
{
|
||||||
|
unsigned int ctl;
|
||||||
|
unsigned char invert;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
if (pin & PIN_INVERSE)
|
||||||
|
{
|
||||||
|
invert = 1;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
} else
|
||||||
|
invert = 0;
|
||||||
|
|
||||||
|
if ( pin < 1 || pin > DB9PINS )
|
||||||
|
return(-1);
|
||||||
|
|
||||||
|
switch ( pin )
|
||||||
|
{
|
||||||
|
case 2: /* rxd, currently not implemented, FIXME */
|
||||||
|
return(-1);
|
||||||
|
|
||||||
|
case 1: /* cd */
|
||||||
|
case 6: /* dsr */
|
||||||
|
case 8: /* cts */
|
||||||
|
case 9: /* ri */
|
||||||
|
r = ioctl(pgm->fd.ifd, TIOCMGET, &ctl);
|
||||||
|
if (r < 0) {
|
||||||
|
perror("ioctl(\"TIOCMGET\")");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if ( !invert )
|
||||||
|
{
|
||||||
|
#ifdef DEBUG
|
||||||
|
printf("%s is %d\n",serpins[pin],(ctl & serregbits[pin]) ? 1 : 0 );
|
||||||
|
#endif
|
||||||
|
return ( (ctl & serregbits[pin]) ? 1 : 0 );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
#ifdef DEBUG
|
||||||
|
printf("%s is %d (~)\n",serpins[pin],(ctl & serregbits[pin]) ? 0 : 1 );
|
||||||
|
#endif
|
||||||
|
return (( ctl & serregbits[pin]) ? 0 : 1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
default: /* impossible */
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int serbb_highpulsepin(PROGRAMMER * pgm, int pin)
|
||||||
|
{
|
||||||
|
if ( (pin & PIN_MASK) < 1 || (pin & PIN_MASK) > DB9PINS )
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
serbb_setpin(pgm, pin, 1);
|
||||||
|
serbb_setpin(pgm, pin, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static void serbb_display(PROGRAMMER *pgm, const char *p)
|
||||||
|
{
|
||||||
|
/* MAYBE */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_enable(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_disable(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_powerup(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_powerdown(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static int serbb_open(PROGRAMMER *pgm, char *port)
|
||||||
|
{
|
||||||
|
struct termios mode;
|
||||||
|
int flags;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
bitbang_check_prerequisites(pgm);
|
||||||
|
|
||||||
|
/* adapted from uisp code */
|
||||||
|
|
||||||
|
pgm->fd.ifd = open(port, O_RDWR | O_NOCTTY | O_NONBLOCK);
|
||||||
|
|
||||||
|
if (pgm->fd.ifd < 0) {
|
||||||
|
perror(port);
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
r = tcgetattr(pgm->fd.ifd, &mode);
|
||||||
|
if (r < 0) {
|
||||||
|
fprintf(stderr, "%s: ", port);
|
||||||
|
perror("tcgetattr");
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
oldmode = mode;
|
||||||
|
|
||||||
|
mode.c_iflag = IGNBRK | IGNPAR;
|
||||||
|
mode.c_oflag = 0;
|
||||||
|
mode.c_cflag = CLOCAL | CREAD | CS8 | B9600;
|
||||||
|
mode.c_cc [VMIN] = 1;
|
||||||
|
mode.c_cc [VTIME] = 0;
|
||||||
|
|
||||||
|
r = tcsetattr(pgm->fd.ifd, TCSANOW, &mode);
|
||||||
|
if (r < 0) {
|
||||||
|
fprintf(stderr, "%s: ", port);
|
||||||
|
perror("tcsetattr");
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Clear O_NONBLOCK flag. */
|
||||||
|
flags = fcntl(pgm->fd.ifd, F_GETFL, 0);
|
||||||
|
if (flags == -1)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Can not get flags: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
flags &= ~O_NONBLOCK;
|
||||||
|
if (fcntl(pgm->fd.ifd, F_SETFL, flags) == -1)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Can not clear nonblock flag: %s\n",
|
||||||
|
progname, strerror(errno));
|
||||||
|
return(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_close(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
if (pgm->fd.ifd != -1)
|
||||||
|
{
|
||||||
|
(void)tcsetattr(pgm->fd.ifd, TCSANOW, &oldmode);
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_RESET], 1);
|
||||||
|
close(pgm->fd.ifd);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void serbb_initpgm(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "SERBB");
|
||||||
|
|
||||||
|
pgm->rdy_led = bitbang_rdy_led;
|
||||||
|
pgm->err_led = bitbang_err_led;
|
||||||
|
pgm->pgm_led = bitbang_pgm_led;
|
||||||
|
pgm->vfy_led = bitbang_vfy_led;
|
||||||
|
pgm->initialize = bitbang_initialize;
|
||||||
|
pgm->display = serbb_display;
|
||||||
|
pgm->enable = serbb_enable;
|
||||||
|
pgm->disable = serbb_disable;
|
||||||
|
pgm->powerup = serbb_powerup;
|
||||||
|
pgm->powerdown = serbb_powerdown;
|
||||||
|
pgm->program_enable = bitbang_program_enable;
|
||||||
|
pgm->chip_erase = bitbang_chip_erase;
|
||||||
|
pgm->cmd = bitbang_cmd;
|
||||||
|
pgm->cmd_tpi = bitbang_cmd_tpi;
|
||||||
|
pgm->open = serbb_open;
|
||||||
|
pgm->close = serbb_close;
|
||||||
|
pgm->setpin = serbb_setpin;
|
||||||
|
pgm->getpin = serbb_getpin;
|
||||||
|
pgm->highpulsepin = serbb_highpulsepin;
|
||||||
|
pgm->read_byte = avr_read_byte_default;
|
||||||
|
pgm->write_byte = avr_write_byte_default;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* WIN32NATIVE */
|
|
@ -0,0 +1,374 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003, 2004 Martin J. Thomas <mthomas@rhrk.uni-kl.de>
|
||||||
|
* Copyright (C) 2005 Michael Holzt <kju-avr@fqdn.org>
|
||||||
|
* Copyright (C) 2005, 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Win32 serial bitbanging interface for avrdude.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "bitbang.h"
|
||||||
|
|
||||||
|
/* cached status lines */
|
||||||
|
static int dtr, rts, txd;
|
||||||
|
|
||||||
|
#define W32SERBUFSIZE 1024
|
||||||
|
|
||||||
|
/*
|
||||||
|
serial port/pin mapping
|
||||||
|
|
||||||
|
1 cd <-
|
||||||
|
2 (rxd) <-
|
||||||
|
3 txd ->
|
||||||
|
4 dtr ->
|
||||||
|
5 GND
|
||||||
|
6 dsr <-
|
||||||
|
7 rts ->
|
||||||
|
8 cts <-
|
||||||
|
9 ri <-
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define DB9PINS 9
|
||||||
|
|
||||||
|
static int serbb_setpin(PROGRAMMER * pgm, int pin, int value)
|
||||||
|
{
|
||||||
|
HANDLE hComPort = (HANDLE)pgm->fd.pfd;
|
||||||
|
LPVOID lpMsgBuf;
|
||||||
|
DWORD dwFunc;
|
||||||
|
const char *name;
|
||||||
|
|
||||||
|
if (pin & PIN_INVERSE)
|
||||||
|
{
|
||||||
|
value = !value;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pin < 1 || pin > DB9PINS)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
switch (pin)
|
||||||
|
{
|
||||||
|
case 3: /* txd */
|
||||||
|
dwFunc = value? SETBREAK: CLRBREAK;
|
||||||
|
name = value? "SETBREAK": "CLRBREAK";
|
||||||
|
txd = value;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4: /* dtr */
|
||||||
|
dwFunc = value? SETDTR: CLRDTR;
|
||||||
|
name = value? "SETDTR": "CLRDTR";
|
||||||
|
dtr = value;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 7: /* rts */
|
||||||
|
dwFunc = value? SETRTS: CLRRTS;
|
||||||
|
name = value? "SETRTS": "CLRRTS";
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_setpin(): unknown pin %d\n",
|
||||||
|
progname, pin + 1);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (verbose > 4)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_setpin(): EscapeCommFunction(%s)\n",
|
||||||
|
progname, name);
|
||||||
|
if (!EscapeCommFunction(hComPort, dwFunc))
|
||||||
|
{
|
||||||
|
FormatMessage(
|
||||||
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL,
|
||||||
|
GetLastError(),
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0,
|
||||||
|
NULL);
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_setpin(): SetCommState() failed: %s\n",
|
||||||
|
progname, (char *)lpMsgBuf);
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
LocalFree(lpMsgBuf);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pgm->ispdelay > 1)
|
||||||
|
bitbang_delay(pgm->ispdelay);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int serbb_getpin(PROGRAMMER * pgm, int pin)
|
||||||
|
{
|
||||||
|
HANDLE hComPort = (HANDLE)pgm->fd.pfd;
|
||||||
|
LPVOID lpMsgBuf;
|
||||||
|
int invert, rv;
|
||||||
|
const char *name;
|
||||||
|
DWORD modemstate;
|
||||||
|
|
||||||
|
if (pin & PIN_INVERSE)
|
||||||
|
{
|
||||||
|
invert = 1;
|
||||||
|
pin &= PIN_MASK;
|
||||||
|
} else
|
||||||
|
invert = 0;
|
||||||
|
|
||||||
|
if (pin < 1 || pin > DB9PINS)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (pin == 1 /* cd */ || pin == 6 /* dsr */ || pin == 8 /* cts */)
|
||||||
|
{
|
||||||
|
if (!GetCommModemStatus(hComPort, &modemstate))
|
||||||
|
{
|
||||||
|
FormatMessage(
|
||||||
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL,
|
||||||
|
GetLastError(),
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0,
|
||||||
|
NULL);
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_setpin(): GetCommModemStatus() failed: %s\n",
|
||||||
|
progname, (char *)lpMsgBuf);
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
LocalFree(lpMsgBuf);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
if (verbose > 4)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_getpin(): GetCommState() => 0x%lx\n",
|
||||||
|
progname, modemstate);
|
||||||
|
switch (pin)
|
||||||
|
{
|
||||||
|
case 1:
|
||||||
|
modemstate &= MS_RLSD_ON;
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
modemstate &= MS_DSR_ON;
|
||||||
|
break;
|
||||||
|
case 8:
|
||||||
|
modemstate &= MS_CTS_ON;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
rv = modemstate != 0;
|
||||||
|
if (invert)
|
||||||
|
rv = !rv;
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (pin)
|
||||||
|
{
|
||||||
|
case 3: /* txd */
|
||||||
|
rv = txd;
|
||||||
|
name = "TXD";
|
||||||
|
break;
|
||||||
|
case 4: /* dtr */
|
||||||
|
rv = dtr;
|
||||||
|
name = "DTR";
|
||||||
|
break;
|
||||||
|
case 7: /* rts */
|
||||||
|
rv = rts;
|
||||||
|
name = "RTS";
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_getpin(): unknown pin %d\n",
|
||||||
|
progname, pin + 1);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (verbose > 4)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: serbb_getpin(): return cached state for %s\n",
|
||||||
|
progname, name);
|
||||||
|
if (invert)
|
||||||
|
rv = !rv;
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int serbb_highpulsepin(PROGRAMMER * pgm, int pin)
|
||||||
|
{
|
||||||
|
if ( (pin & PIN_MASK) < 1 || (pin & PIN_MASK) > DB9PINS )
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
serbb_setpin(pgm, pin, 1);
|
||||||
|
serbb_setpin(pgm, pin, 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void serbb_display(PROGRAMMER *pgm, const char *p)
|
||||||
|
{
|
||||||
|
/* MAYBE */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_enable(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_disable(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_powerup(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_powerdown(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
/* nothing */
|
||||||
|
}
|
||||||
|
|
||||||
|
static int serbb_open(PROGRAMMER *pgm, char *port)
|
||||||
|
{
|
||||||
|
DCB dcb;
|
||||||
|
LPVOID lpMsgBuf;
|
||||||
|
HANDLE hComPort = INVALID_HANDLE_VALUE;
|
||||||
|
|
||||||
|
bitbang_check_prerequisites(pgm);
|
||||||
|
|
||||||
|
hComPort = CreateFile(port, GENERIC_READ | GENERIC_WRITE, 0, NULL,
|
||||||
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
||||||
|
|
||||||
|
if (hComPort == INVALID_HANDLE_VALUE) {
|
||||||
|
FormatMessage(
|
||||||
|
FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
||||||
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
||||||
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
||||||
|
NULL,
|
||||||
|
GetLastError(),
|
||||||
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
||||||
|
(LPTSTR) &lpMsgBuf,
|
||||||
|
0,
|
||||||
|
NULL);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't open device \"%s\": %s\n",
|
||||||
|
progname, port, (char*)lpMsgBuf);
|
||||||
|
LocalFree(lpMsgBuf);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!SetupComm(hComPort, W32SERBUFSIZE, W32SERBUFSIZE))
|
||||||
|
{
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't set buffers for \"%s\"\n",
|
||||||
|
progname, port);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ZeroMemory(&dcb, sizeof(DCB));
|
||||||
|
dcb.DCBlength = sizeof(DCB);
|
||||||
|
dcb.BaudRate = CBR_9600;
|
||||||
|
dcb.fBinary = 1;
|
||||||
|
dcb.fDtrControl = DTR_CONTROL_DISABLE;
|
||||||
|
dcb.fRtsControl = RTS_CONTROL_DISABLE;
|
||||||
|
dcb.ByteSize = 8;
|
||||||
|
dcb.Parity = NOPARITY;
|
||||||
|
dcb.StopBits = ONESTOPBIT;
|
||||||
|
|
||||||
|
if (!SetCommState(hComPort, &dcb))
|
||||||
|
{
|
||||||
|
CloseHandle(hComPort);
|
||||||
|
fprintf(stderr, "%s: ser_open(): can't set com-state for \"%s\"\n",
|
||||||
|
progname, port);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (verbose > 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_open(): opened comm port \"%s\", handle 0x%x\n",
|
||||||
|
progname, port, (int)hComPort);
|
||||||
|
|
||||||
|
pgm->fd.pfd = (void *)hComPort;
|
||||||
|
|
||||||
|
dtr = rts = txd = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void serbb_close(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
HANDLE hComPort=(HANDLE)pgm->fd.pfd;
|
||||||
|
if (hComPort != INVALID_HANDLE_VALUE)
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_RESET], 1);
|
||||||
|
CloseHandle (hComPort);
|
||||||
|
}
|
||||||
|
if (verbose > 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: ser_close(): closed comm port handle 0x%x\n",
|
||||||
|
progname, (int)hComPort);
|
||||||
|
|
||||||
|
hComPort = INVALID_HANDLE_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
void serbb_initpgm(PROGRAMMER *pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "SERBB");
|
||||||
|
|
||||||
|
pgm->rdy_led = bitbang_rdy_led;
|
||||||
|
pgm->err_led = bitbang_err_led;
|
||||||
|
pgm->pgm_led = bitbang_pgm_led;
|
||||||
|
pgm->vfy_led = bitbang_vfy_led;
|
||||||
|
pgm->initialize = bitbang_initialize;
|
||||||
|
pgm->display = serbb_display;
|
||||||
|
pgm->enable = serbb_enable;
|
||||||
|
pgm->disable = serbb_disable;
|
||||||
|
pgm->powerup = serbb_powerup;
|
||||||
|
pgm->powerdown = serbb_powerdown;
|
||||||
|
pgm->program_enable = bitbang_program_enable;
|
||||||
|
pgm->chip_erase = bitbang_chip_erase;
|
||||||
|
pgm->cmd = bitbang_cmd;
|
||||||
|
pgm->cmd_tpi = bitbang_cmd_tpi;
|
||||||
|
pgm->open = serbb_open;
|
||||||
|
pgm->close = serbb_close;
|
||||||
|
pgm->setpin = serbb_setpin;
|
||||||
|
pgm->getpin = serbb_getpin;
|
||||||
|
pgm->highpulsepin = serbb_highpulsepin;
|
||||||
|
pgm->read_byte = avr_read_byte_default;
|
||||||
|
pgm->write_byte = avr_write_byte_default;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* WIN32NATIVE */
|
|
@ -0,0 +1,77 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2003-2004 Theodore A. Roth <troth@openavr.org>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/* This is the API for the generic serial interface. The implementations are
|
||||||
|
actually provided by the target dependant files:
|
||||||
|
|
||||||
|
ser_posix.c : posix serial interface.
|
||||||
|
ser_win32.c : native win32 serial interface.
|
||||||
|
|
||||||
|
The target file will be selected at configure time. */
|
||||||
|
|
||||||
|
#ifndef serial_h
|
||||||
|
#define serial_h
|
||||||
|
|
||||||
|
extern long serial_recv_timeout;
|
||||||
|
union filedescriptor
|
||||||
|
{
|
||||||
|
int ifd;
|
||||||
|
void *pfd;
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *handle;
|
||||||
|
int ep;
|
||||||
|
} usb;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct serial_device
|
||||||
|
{
|
||||||
|
// open should return -1 on error, other values on success
|
||||||
|
int (*open)(char * port, long baud, union filedescriptor *fd);
|
||||||
|
int (*setspeed)(union filedescriptor *fd, long baud);
|
||||||
|
void (*close)(union filedescriptor *fd);
|
||||||
|
|
||||||
|
int (*send)(union filedescriptor *fd, unsigned char * buf, size_t buflen);
|
||||||
|
int (*recv)(union filedescriptor *fd, unsigned char * buf, size_t buflen);
|
||||||
|
int (*drain)(union filedescriptor *fd, int display);
|
||||||
|
|
||||||
|
int (*set_dtr_rts)(union filedescriptor *fd, int is_on);
|
||||||
|
|
||||||
|
int flags;
|
||||||
|
#define SERDEV_FL_NONE 0x0000 /* no flags */
|
||||||
|
#define SERDEV_FL_CANSETSPEED 0x0001 /* device can change speed */
|
||||||
|
};
|
||||||
|
|
||||||
|
extern struct serial_device *serdev;
|
||||||
|
extern struct serial_device serial_serdev;
|
||||||
|
extern struct serial_device usb_serdev;
|
||||||
|
extern struct serial_device usb_serdev_frame;
|
||||||
|
extern struct serial_device avrdoper_serdev;
|
||||||
|
|
||||||
|
#define serial_open (serdev->open)
|
||||||
|
#define serial_setspeed (serdev->setspeed)
|
||||||
|
#define serial_close (serdev->close)
|
||||||
|
#define serial_send (serdev->send)
|
||||||
|
#define serial_recv (serdev->recv)
|
||||||
|
#define serial_drain (serdev->drain)
|
||||||
|
#define serial_set_dtr_rts (serdev->set_dtr_rts)
|
||||||
|
|
||||||
|
#endif /* serial_h */
|
|
@ -0,0 +1,51 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2005 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef solaris_ecpp_h
|
||||||
|
#define solaris_ecpp_h
|
||||||
|
|
||||||
|
#include <sys/ecppio.h>
|
||||||
|
|
||||||
|
#define ppi_claim(fd) \
|
||||||
|
do { \
|
||||||
|
struct ecpp_transfer_parms p; \
|
||||||
|
(void)ioctl(fd, ECPPIOC_GETPARMS, &p); \
|
||||||
|
p.mode = ECPP_DIAG_MODE; \
|
||||||
|
(void)ioctl(fd, ECPPIOC_SETPARMS, &p); \
|
||||||
|
} while(0);
|
||||||
|
|
||||||
|
#define ppi_release(fd)
|
||||||
|
|
||||||
|
#define DO_PPI_READ(fd, reg, valp) \
|
||||||
|
do { struct ecpp_regs r; \
|
||||||
|
if ((reg) == PPIDATA) { (void)ioctl(fd, ECPPIOC_GETDATA, valp); } \
|
||||||
|
else { (void)ioctl(fd, ECPPIOC_GETREGS, &r); \
|
||||||
|
*(valp) = ((reg) == PPICTRL)? r.dcr: r.dsr; } \
|
||||||
|
} while(0)
|
||||||
|
#define DO_PPI_WRITE(fd, reg, valp) \
|
||||||
|
do { struct ecpp_regs r; \
|
||||||
|
if ((reg) == PPIDATA) { (void)ioctl(fd, ECPPIOC_SETDATA, valp); } \
|
||||||
|
else { if ((reg) == PPICTRL) r.dcr = *(valp); else r.dsr = *(valp); \
|
||||||
|
(void)ioctl(fd, ECPPIOC_SETREGS, &r); } \
|
||||||
|
} while(0)
|
||||||
|
|
||||||
|
|
||||||
|
#endif /* solaris_ecpp_h */
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,41 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2002-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef stk500_h
|
||||||
|
#define stk500_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void stk500_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
/* used by arduino.c to avoid duplicate code */
|
||||||
|
int stk500_getsync(PROGRAMMER * pgm);
|
||||||
|
int stk500_drain(PROGRAMMER * pgm, int display);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,103 @@
|
||||||
|
//**** ATMEL AVR - A P P L I C A T I O N N O T E ************************
|
||||||
|
//*
|
||||||
|
//* Title: AVR061 - STK500 Communication Protocol
|
||||||
|
//* Filename: command.h
|
||||||
|
//* Version: 1.0
|
||||||
|
//* Last updated: 09.09.2002
|
||||||
|
//*
|
||||||
|
//* Support E-mail: avr@atmel.com
|
||||||
|
//*
|
||||||
|
//**************************************************************************
|
||||||
|
|
||||||
|
// *****************[ STK Message constants ]***************************
|
||||||
|
|
||||||
|
#define STK_SIGN_ON_MESSAGE "AVR STK" // Sign on string for Cmnd_STK_GET_SIGN_ON
|
||||||
|
|
||||||
|
// *****************[ STK Response constants ]***************************
|
||||||
|
|
||||||
|
#define Resp_STK_OK 0x10 // ' '
|
||||||
|
#define Resp_STK_FAILED 0x11 // ' '
|
||||||
|
#define Resp_STK_UNKNOWN 0x12 // ' '
|
||||||
|
#define Resp_STK_NODEVICE 0x13 // ' '
|
||||||
|
#define Resp_STK_INSYNC 0x14 // ' '
|
||||||
|
#define Resp_STK_NOSYNC 0x15 // ' '
|
||||||
|
|
||||||
|
#define Resp_ADC_CHANNEL_ERROR 0x16 // ' '
|
||||||
|
#define Resp_ADC_MEASURE_OK 0x17 // ' '
|
||||||
|
#define Resp_PWM_CHANNEL_ERROR 0x18 // ' '
|
||||||
|
#define Resp_PWM_ADJUST_OK 0x19 // ' '
|
||||||
|
|
||||||
|
// *****************[ STK Special constants ]***************************
|
||||||
|
|
||||||
|
#define Sync_CRC_EOP 0x20 // 'SPACE'
|
||||||
|
|
||||||
|
// *****************[ STK Command constants ]***************************
|
||||||
|
|
||||||
|
#define Cmnd_STK_GET_SYNC 0x30 // ' '
|
||||||
|
#define Cmnd_STK_GET_SIGN_ON 0x31 // ' '
|
||||||
|
|
||||||
|
#define Cmnd_STK_SET_PARAMETER 0x40 // ' '
|
||||||
|
#define Cmnd_STK_GET_PARAMETER 0x41 // ' '
|
||||||
|
#define Cmnd_STK_SET_DEVICE 0x42 // ' '
|
||||||
|
#define Cmnd_STK_SET_DEVICE_EXT 0x45 // ' '
|
||||||
|
|
||||||
|
#define Cmnd_STK_ENTER_PROGMODE 0x50 // ' '
|
||||||
|
#define Cmnd_STK_LEAVE_PROGMODE 0x51 // ' '
|
||||||
|
#define Cmnd_STK_CHIP_ERASE 0x52 // ' '
|
||||||
|
#define Cmnd_STK_CHECK_AUTOINC 0x53 // ' '
|
||||||
|
#define Cmnd_STK_LOAD_ADDRESS 0x55 // ' '
|
||||||
|
#define Cmnd_STK_UNIVERSAL 0x56 // ' '
|
||||||
|
#define Cmnd_STK_UNIVERSAL_MULTI 0x57 // ' '
|
||||||
|
|
||||||
|
#define Cmnd_STK_PROG_FLASH 0x60 // ' '
|
||||||
|
#define Cmnd_STK_PROG_DATA 0x61 // ' '
|
||||||
|
#define Cmnd_STK_PROG_FUSE 0x62 // ' '
|
||||||
|
#define Cmnd_STK_PROG_LOCK 0x63 // ' '
|
||||||
|
#define Cmnd_STK_PROG_PAGE 0x64 // ' '
|
||||||
|
#define Cmnd_STK_PROG_FUSE_EXT 0x65 // ' '
|
||||||
|
|
||||||
|
#define Cmnd_STK_READ_FLASH 0x70 // ' '
|
||||||
|
#define Cmnd_STK_READ_DATA 0x71 // ' '
|
||||||
|
#define Cmnd_STK_READ_FUSE 0x72 // ' '
|
||||||
|
#define Cmnd_STK_READ_LOCK 0x73 // ' '
|
||||||
|
#define Cmnd_STK_READ_PAGE 0x74 // ' '
|
||||||
|
#define Cmnd_STK_READ_SIGN 0x75 // ' '
|
||||||
|
#define Cmnd_STK_READ_OSCCAL 0x76 // ' '
|
||||||
|
#define Cmnd_STK_READ_FUSE_EXT 0x77 // ' '
|
||||||
|
#define Cmnd_STK_READ_OSCCAL_EXT 0x78 // ' '
|
||||||
|
|
||||||
|
// *****************[ STK Parameter constants ]***************************
|
||||||
|
|
||||||
|
#define Parm_STK_HW_VER 0x80 // ' ' - R
|
||||||
|
#define Parm_STK_SW_MAJOR 0x81 // ' ' - R
|
||||||
|
#define Parm_STK_SW_MINOR 0x82 // ' ' - R
|
||||||
|
#define Parm_STK_LEDS 0x83 // ' ' - R/W
|
||||||
|
#define Parm_STK_VTARGET 0x84 // ' ' - R/W
|
||||||
|
#define Parm_STK_VADJUST 0x85 // ' ' - R/W
|
||||||
|
#define Parm_STK_OSC_PSCALE 0x86 // ' ' - R/W
|
||||||
|
#define Parm_STK_OSC_CMATCH 0x87 // ' ' - R/W
|
||||||
|
#define Parm_STK_RESET_DURATION 0x88 // ' ' - R/W
|
||||||
|
#define Parm_STK_SCK_DURATION 0x89 // ' ' - R/W
|
||||||
|
|
||||||
|
#define Parm_STK_BUFSIZEL 0x90 // ' ' - R/W, Range {0..255}
|
||||||
|
#define Parm_STK_BUFSIZEH 0x91 // ' ' - R/W, Range {0..255}
|
||||||
|
#define Parm_STK_DEVICE 0x92 // ' ' - R/W, Range {0..255}
|
||||||
|
#define Parm_STK_PROGMODE 0x93 // ' ' - 'P' or 'S'
|
||||||
|
#define Parm_STK_PARAMODE 0x94 // ' ' - TRUE or FALSE
|
||||||
|
#define Parm_STK_POLLING 0x95 // ' ' - TRUE or FALSE
|
||||||
|
#define Parm_STK_SELFTIMED 0x96 // ' ' - TRUE or FALSE
|
||||||
|
#define Param_STK500_TOPCARD_DETECT 0x98 // ' ' - Detect top-card attached
|
||||||
|
|
||||||
|
// *****************[ STK status bit definitions ]***************************
|
||||||
|
|
||||||
|
#define Stat_STK_INSYNC 0x01 // INSYNC status bit, '1' - INSYNC
|
||||||
|
#define Stat_STK_PROGMODE 0x02 // Programming mode, '1' - PROGMODE
|
||||||
|
#define Stat_STK_STANDALONE 0x04 // Standalone mode, '1' - SM mode
|
||||||
|
#define Stat_STK_RESET 0x08 // RESET button, '1' - Pushed
|
||||||
|
#define Stat_STK_PROGRAM 0x10 // Program button, ' 1' - Pushed
|
||||||
|
#define Stat_STK_LEDG 0x20 // Green LED status, '1' - Lit
|
||||||
|
#define Stat_STK_LEDR 0x40 // Red LED status, '1' - Lit
|
||||||
|
#define Stat_STK_LEDBLINK 0x80 // LED blink ON/OFF, '1' - Blink
|
||||||
|
|
||||||
|
|
||||||
|
// *****************************[ End Of COMMAND.H ]**************************
|
|
@ -0,0 +1,91 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* avrdude interface for Atmel STK500 programmer
|
||||||
|
*
|
||||||
|
* This is a wrapper around the STK500[v1] and STK500v2 programmers.
|
||||||
|
* Try to select the programmer type that actually responds, and
|
||||||
|
* divert to the actual programmer implementation if successful.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "stk500.h"
|
||||||
|
#include "stk500v2.h"
|
||||||
|
|
||||||
|
static int stk500generic_open(PROGRAMMER * pgm, char * port)
|
||||||
|
{
|
||||||
|
stk500_initpgm(pgm);
|
||||||
|
if (pgm->open(pgm, port) >= 0)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: successfully opened stk500v1 device -- please use -c stk500v1\n",
|
||||||
|
progname);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->close(pgm);
|
||||||
|
|
||||||
|
stk500v2_initpgm(pgm);
|
||||||
|
if (pgm->open(pgm, port) >= 0)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: successfully opened stk500v2 device -- please use -c stk500v2\n",
|
||||||
|
progname);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: cannot open either stk500v1 or stk500v2 programmer\n",
|
||||||
|
progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void stk500generic_setup(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Only STK500v2 needs setup/teardown.
|
||||||
|
*/
|
||||||
|
stk500v2_initpgm(pgm);
|
||||||
|
pgm->setup(pgm);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void stk500generic_teardown(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
stk500v2_initpgm(pgm);
|
||||||
|
pgm->teardown(pgm);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void stk500generic_initpgm(PROGRAMMER * pgm)
|
||||||
|
{
|
||||||
|
strcpy(pgm->type, "STK500GENERIC");
|
||||||
|
|
||||||
|
pgm->open = stk500generic_open;
|
||||||
|
pgm->setup = stk500generic_setup;
|
||||||
|
pgm->teardown = stk500generic_teardown;
|
||||||
|
}
|
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef stk500generic_h__
|
||||||
|
#define stk500generic_h__
|
||||||
|
|
||||||
|
void stk500generic_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2002-2005 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2006 Joerg Wunsch <j@uriah.heep.sax.de>
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef stk500v2_h
|
||||||
|
#define stk500v2_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void stk500v2_initpgm (PROGRAMMER * pgm);
|
||||||
|
void stk500hvsp_initpgm (PROGRAMMER * pgm);
|
||||||
|
void stk500pp_initpgm (PROGRAMMER * pgm);
|
||||||
|
void stk500v2_jtagmkII_initpgm(PROGRAMMER * pgm);
|
||||||
|
void stk500v2_dragon_hvsp_initpgm(PROGRAMMER * pgm);
|
||||||
|
void stk500v2_dragon_isp_initpgm(PROGRAMMER * pgm);
|
||||||
|
void stk500v2_dragon_pp_initpgm(PROGRAMMER * pgm);
|
||||||
|
void stk600_initpgm (PROGRAMMER * pgm);
|
||||||
|
void stk600hvsp_initpgm (PROGRAMMER * pgm);
|
||||||
|
void stk600pp_initpgm (PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
void stk500v2_setup(PROGRAMMER * pgm);
|
||||||
|
void stk500v2_teardown(PROGRAMMER * pgm);
|
||||||
|
int stk500v2_drain(PROGRAMMER * pgm, int display);
|
||||||
|
int stk500v2_getsync(PROGRAMMER * pgm);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,322 @@
|
||||||
|
//**** ATMEL AVR - A P P L I C A T I O N N O T E ************************
|
||||||
|
//*
|
||||||
|
//* Title: AVR068 - STK500 Communication Protocol
|
||||||
|
//* Filename: command.h
|
||||||
|
//* Version: 1.0
|
||||||
|
//* Last updated: 10.01.2005
|
||||||
|
//*
|
||||||
|
//* Support E-mail: avr@atmel.com
|
||||||
|
//*
|
||||||
|
//**************************************************************************
|
||||||
|
|
||||||
|
#include "pgm.h"
|
||||||
|
|
||||||
|
// *****************[ STK message constants ]***************************
|
||||||
|
|
||||||
|
#define MESSAGE_START 0x1B //= ESC = 27 decimal
|
||||||
|
#define TOKEN 0x0E
|
||||||
|
|
||||||
|
// *****************[ STK general command constants ]**************************
|
||||||
|
|
||||||
|
#define CMD_SIGN_ON 0x01
|
||||||
|
#define CMD_SET_PARAMETER 0x02
|
||||||
|
#define CMD_GET_PARAMETER 0x03
|
||||||
|
#define CMD_SET_DEVICE_PARAMETERS 0x04
|
||||||
|
#define CMD_OSCCAL 0x05
|
||||||
|
#define CMD_LOAD_ADDRESS 0x06
|
||||||
|
#define CMD_FIRMWARE_UPGRADE 0x07
|
||||||
|
#define CMD_CHECK_TARGET_CONNECTION 0x0D
|
||||||
|
#define CMD_LOAD_RC_ID_TABLE 0x0E
|
||||||
|
#define CMD_LOAD_EC_ID_TABLE 0x0F
|
||||||
|
|
||||||
|
|
||||||
|
// *****************[ STK ISP command constants ]******************************
|
||||||
|
|
||||||
|
#define CMD_ENTER_PROGMODE_ISP 0x10
|
||||||
|
#define CMD_LEAVE_PROGMODE_ISP 0x11
|
||||||
|
#define CMD_CHIP_ERASE_ISP 0x12
|
||||||
|
#define CMD_PROGRAM_FLASH_ISP 0x13
|
||||||
|
#define CMD_READ_FLASH_ISP 0x14
|
||||||
|
#define CMD_PROGRAM_EEPROM_ISP 0x15
|
||||||
|
#define CMD_READ_EEPROM_ISP 0x16
|
||||||
|
#define CMD_PROGRAM_FUSE_ISP 0x17
|
||||||
|
#define CMD_READ_FUSE_ISP 0x18
|
||||||
|
#define CMD_PROGRAM_LOCK_ISP 0x19
|
||||||
|
#define CMD_READ_LOCK_ISP 0x1A
|
||||||
|
#define CMD_READ_SIGNATURE_ISP 0x1B
|
||||||
|
#define CMD_READ_OSCCAL_ISP 0x1C
|
||||||
|
#define CMD_SPI_MULTI 0x1D
|
||||||
|
|
||||||
|
// *****************[ STK PP command constants ]*******************************
|
||||||
|
|
||||||
|
#define CMD_ENTER_PROGMODE_PP 0x20
|
||||||
|
#define CMD_LEAVE_PROGMODE_PP 0x21
|
||||||
|
#define CMD_CHIP_ERASE_PP 0x22
|
||||||
|
#define CMD_PROGRAM_FLASH_PP 0x23
|
||||||
|
#define CMD_READ_FLASH_PP 0x24
|
||||||
|
#define CMD_PROGRAM_EEPROM_PP 0x25
|
||||||
|
#define CMD_READ_EEPROM_PP 0x26
|
||||||
|
#define CMD_PROGRAM_FUSE_PP 0x27
|
||||||
|
#define CMD_READ_FUSE_PP 0x28
|
||||||
|
#define CMD_PROGRAM_LOCK_PP 0x29
|
||||||
|
#define CMD_READ_LOCK_PP 0x2A
|
||||||
|
#define CMD_READ_SIGNATURE_PP 0x2B
|
||||||
|
#define CMD_READ_OSCCAL_PP 0x2C
|
||||||
|
|
||||||
|
#define CMD_SET_CONTROL_STACK 0x2D
|
||||||
|
|
||||||
|
// *****************[ STK HVSP command constants ]*****************************
|
||||||
|
|
||||||
|
#define CMD_ENTER_PROGMODE_HVSP 0x30
|
||||||
|
#define CMD_LEAVE_PROGMODE_HVSP 0x31
|
||||||
|
#define CMD_CHIP_ERASE_HVSP 0x32
|
||||||
|
#define CMD_PROGRAM_FLASH_HVSP 0x33
|
||||||
|
#define CMD_READ_FLASH_HVSP 0x34
|
||||||
|
#define CMD_PROGRAM_EEPROM_HVSP 0x35
|
||||||
|
#define CMD_READ_EEPROM_HVSP 0x36
|
||||||
|
#define CMD_PROGRAM_FUSE_HVSP 0x37
|
||||||
|
#define CMD_READ_FUSE_HVSP 0x38
|
||||||
|
#define CMD_PROGRAM_LOCK_HVSP 0x39
|
||||||
|
#define CMD_READ_LOCK_HVSP 0x3A
|
||||||
|
#define CMD_READ_SIGNATURE_HVSP 0x3B
|
||||||
|
#define CMD_READ_OSCCAL_HVSP 0x3C
|
||||||
|
// These two are redefined since 0x30/0x31 collide
|
||||||
|
// with the STK600 bootloader.
|
||||||
|
#define CMD_ENTER_PROGMODE_HVSP_STK600 0x3D
|
||||||
|
#define CMD_LEAVE_PROGMODE_HVSP_STK600 0x3E
|
||||||
|
|
||||||
|
// *** XPROG command constants ***
|
||||||
|
|
||||||
|
#define CMD_XPROG 0x50
|
||||||
|
#define CMD_XPROG_SETMODE 0x51
|
||||||
|
|
||||||
|
|
||||||
|
// *** AVR32 JTAG Programming command ***
|
||||||
|
|
||||||
|
#define CMD_JTAG_AVR32 0x80
|
||||||
|
#define CMD_ENTER_PROGMODE_JTAG_AVR32 0x81
|
||||||
|
#define CMD_LEAVE_PROGMODE_JTAG_AVR32 0x82
|
||||||
|
|
||||||
|
|
||||||
|
// *** AVR JTAG Programming command ***
|
||||||
|
|
||||||
|
#define CMD_JTAG_AVR 0x90
|
||||||
|
|
||||||
|
// *****************[ STK test command constants ]***************************
|
||||||
|
|
||||||
|
#define CMD_ENTER_TESTMODE 0x60
|
||||||
|
#define CMD_LEAVE_TESTMODE 0x61
|
||||||
|
#define CMD_CHIP_WRITE 0x62
|
||||||
|
#define CMD_PROGRAM_FLASH_PARTIAL 0x63
|
||||||
|
#define CMD_PROGRAM_EEPROM_PARTIAL 0x64
|
||||||
|
#define CMD_PROGRAM_SIGNATURE_ROW 0x65
|
||||||
|
#define CMD_READ_FLASH_MARGIN 0x66
|
||||||
|
#define CMD_READ_EEPROM_MARGIN 0x67
|
||||||
|
#define CMD_READ_SIGNATURE_ROW_MARGIN 0x68
|
||||||
|
#define CMD_PROGRAM_TEST_FUSE 0x69
|
||||||
|
#define CMD_READ_TEST_FUSE 0x6A
|
||||||
|
#define CMD_PROGRAM_HIDDEN_FUSE_LOW 0x6B
|
||||||
|
#define CMD_READ_HIDDEN_FUSE_LOW 0x6C
|
||||||
|
#define CMD_PROGRAM_HIDDEN_FUSE_HIGH 0x6D
|
||||||
|
#define CMD_READ_HIDDEN_FUSE_HIGH 0x6E
|
||||||
|
#define CMD_PROGRAM_HIDDEN_FUSE_EXT 0x6F
|
||||||
|
#define CMD_READ_HIDDEN_FUSE_EXT 0x70
|
||||||
|
|
||||||
|
// *****************[ STK status constants ]***************************
|
||||||
|
|
||||||
|
// Success
|
||||||
|
#define STATUS_CMD_OK 0x00
|
||||||
|
|
||||||
|
// Warnings
|
||||||
|
#define STATUS_CMD_TOUT 0x80
|
||||||
|
#define STATUS_RDY_BSY_TOUT 0x81
|
||||||
|
#define STATUS_SET_PARAM_MISSING 0x82
|
||||||
|
|
||||||
|
// Errors
|
||||||
|
#define STATUS_CMD_FAILED 0xC0
|
||||||
|
#define STATUS_CKSUM_ERROR 0xC1
|
||||||
|
#define STATUS_CMD_UNKNOWN 0xC9
|
||||||
|
#define STATUS_CMD_ILLEGAL_PARAMETER 0xCA
|
||||||
|
|
||||||
|
// Status
|
||||||
|
#define STATUS_ISP_READY 0x00
|
||||||
|
#define STATUS_CONN_FAIL_MOSI 0x01
|
||||||
|
#define STATUS_CONN_FAIL_RST 0x02
|
||||||
|
#define STATUS_CONN_FAIL_SCK 0x04
|
||||||
|
#define STATUS_TGT_NOT_DETECTED 0x10
|
||||||
|
#define STATUS_TGT_REVERSE_INSERTED 0x20
|
||||||
|
|
||||||
|
// hw_status
|
||||||
|
// Bits in status variable
|
||||||
|
// Bit 0-3: Slave MCU
|
||||||
|
// Bit 4-7: Master MCU
|
||||||
|
|
||||||
|
#define STATUS_AREF_ERROR 0
|
||||||
|
// Set to '1' if AREF is short circuited
|
||||||
|
|
||||||
|
#define STATUS_VTG_ERROR 4
|
||||||
|
// Set to '1' if VTG is short circuited
|
||||||
|
|
||||||
|
#define STATUS_RC_CARD_ERROR 5
|
||||||
|
// Set to '1' if board id changes when board is powered
|
||||||
|
|
||||||
|
#define STATUS_PROGMODE 6
|
||||||
|
// Set to '1' if board is in programming mode
|
||||||
|
|
||||||
|
#define STATUS_POWER_SURGE 7
|
||||||
|
// Set to '1' if board draws excessive current
|
||||||
|
|
||||||
|
// *****************[ STK parameter constants ]***************************
|
||||||
|
#define PARAM_BUILD_NUMBER_LOW 0x80 /* ??? */
|
||||||
|
#define PARAM_BUILD_NUMBER_HIGH 0x81 /* ??? */
|
||||||
|
#define PARAM_HW_VER 0x90
|
||||||
|
#define PARAM_SW_MAJOR 0x91
|
||||||
|
#define PARAM_SW_MINOR 0x92
|
||||||
|
#define PARAM_VTARGET 0x94
|
||||||
|
#define PARAM_VADJUST 0x95 /* STK500 only */
|
||||||
|
#define PARAM_OSC_PSCALE 0x96 /* STK500 only */
|
||||||
|
#define PARAM_OSC_CMATCH 0x97 /* STK500 only */
|
||||||
|
#define PARAM_SCK_DURATION 0x98 /* STK500 only */
|
||||||
|
#define PARAM_TOPCARD_DETECT 0x9A /* STK500 only */
|
||||||
|
#define PARAM_STATUS 0x9C /* STK500 only */
|
||||||
|
#define PARAM_DATA 0x9D /* STK500 only */
|
||||||
|
#define PARAM_RESET_POLARITY 0x9E /* STK500 only, and STK600 FW
|
||||||
|
* version <= 2.0.3 */
|
||||||
|
#define PARAM_CONTROLLER_INIT 0x9F
|
||||||
|
|
||||||
|
/* STK600 parameters */
|
||||||
|
#define PARAM_STATUS_TGT_CONN 0xA1
|
||||||
|
#define PARAM_DISCHARGEDELAY 0xA4
|
||||||
|
#define PARAM_SOCKETCARD_ID 0xA5
|
||||||
|
#define PARAM_ROUTINGCARD_ID 0xA6
|
||||||
|
#define PARAM_EXPCARD_ID 0xA7
|
||||||
|
#define PARAM_SW_MAJOR_SLAVE1 0xA8
|
||||||
|
#define PARAM_SW_MINOR_SLAVE1 0xA9
|
||||||
|
#define PARAM_SW_MAJOR_SLAVE2 0xAA
|
||||||
|
#define PARAM_SW_MINOR_SLAVE2 0xAB
|
||||||
|
#define PARAM_BOARD_ID_STATUS 0xAD
|
||||||
|
#define PARAM_RESET 0xB4
|
||||||
|
|
||||||
|
#define PARAM_JTAG_ALLOW_FULL_PAGE_STREAM 0x50
|
||||||
|
#define PARAM_JTAG_EEPROM_PAGE_SIZE 0x52
|
||||||
|
#define PARAM_JTAG_DAISY_BITS_BEFORE 0x53
|
||||||
|
#define PARAM_JTAG_DAISY_BITS_AFTER 0x54
|
||||||
|
#define PARAM_JTAG_DAISY_UNITS_BEFORE 0x55
|
||||||
|
#define PARAM_JTAG_DAISY_UNITS_AFTER 0x56
|
||||||
|
|
||||||
|
// *** Parameter constants for 2 byte values ***
|
||||||
|
#define PARAM2_SCK_DURATION 0xC0
|
||||||
|
#define PARAM2_CLOCK_CONF 0xC1
|
||||||
|
#define PARAM2_AREF0 0xC2
|
||||||
|
#define PARAM2_AREF1 0xC3
|
||||||
|
|
||||||
|
#define PARAM2_JTAG_FLASH_SIZE_H 0xC5
|
||||||
|
#define PARAM2_JTAG_FLASH_SIZE_L 0xC6
|
||||||
|
#define PARAM2_JTAG_FLASH_PAGE_SIZE 0xC7
|
||||||
|
#define PARAM2_RC_ID_TABLE_REV 0xC8
|
||||||
|
#define PARAM2_EC_ID_TABLE_REV 0xC9
|
||||||
|
|
||||||
|
/* STK600 XPROG section */
|
||||||
|
// XPROG modes
|
||||||
|
#define XPRG_MODE_PDI 0
|
||||||
|
#define XPRG_MODE_JTAG 1
|
||||||
|
#define XPRG_MODE_TPI 2
|
||||||
|
|
||||||
|
// XPROG commands
|
||||||
|
#define XPRG_CMD_ENTER_PROGMODE 0x01
|
||||||
|
#define XPRG_CMD_LEAVE_PROGMODE 0x02
|
||||||
|
#define XPRG_CMD_ERASE 0x03
|
||||||
|
#define XPRG_CMD_WRITE_MEM 0x04
|
||||||
|
#define XPRG_CMD_READ_MEM 0x05
|
||||||
|
#define XPRG_CMD_CRC 0x06
|
||||||
|
#define XPRG_CMD_SET_PARAM 0x07
|
||||||
|
|
||||||
|
// Memory types
|
||||||
|
#define XPRG_MEM_TYPE_APPL 1
|
||||||
|
#define XPRG_MEM_TYPE_BOOT 2
|
||||||
|
#define XPRG_MEM_TYPE_EEPROM 3
|
||||||
|
#define XPRG_MEM_TYPE_FUSE 4
|
||||||
|
#define XPRG_MEM_TYPE_LOCKBITS 5
|
||||||
|
#define XPRG_MEM_TYPE_USERSIG 6
|
||||||
|
#define XPRG_MEM_TYPE_FACTORY_CALIBRATION 7
|
||||||
|
|
||||||
|
// Erase types
|
||||||
|
#define XPRG_ERASE_CHIP 1
|
||||||
|
#define XPRG_ERASE_APP 2
|
||||||
|
#define XPRG_ERASE_BOOT 3
|
||||||
|
#define XPRG_ERASE_EEPROM 4
|
||||||
|
#define XPRG_ERASE_APP_PAGE 5
|
||||||
|
#define XPRG_ERASE_BOOT_PAGE 6
|
||||||
|
#define XPRG_ERASE_EEPROM_PAGE 7
|
||||||
|
#define XPRG_ERASE_USERSIG 8
|
||||||
|
#define XPRG_ERASE_CONFIG 9 // TPI only, prepare fuse write
|
||||||
|
|
||||||
|
// Write mode flags
|
||||||
|
#define XPRG_MEM_WRITE_ERASE 0
|
||||||
|
#define XPRG_MEM_WRITE_WRITE 1
|
||||||
|
|
||||||
|
// CRC types
|
||||||
|
#define XPRG_CRC_APP 1
|
||||||
|
#define XPRG_CRC_BOOT 2
|
||||||
|
#define XPRG_CRC_FLASH 3
|
||||||
|
|
||||||
|
// Error codes
|
||||||
|
#define XPRG_ERR_OK 0
|
||||||
|
#define XPRG_ERR_FAILED 1
|
||||||
|
#define XPRG_ERR_COLLISION 2
|
||||||
|
#define XPRG_ERR_TIMEOUT 3
|
||||||
|
|
||||||
|
// XPROG parameters of different sizes
|
||||||
|
// 4-byte address
|
||||||
|
#define XPRG_PARAM_NVMBASE 0x01
|
||||||
|
// 2-byte page size
|
||||||
|
#define XPRG_PARAM_EEPPAGESIZE 0x02
|
||||||
|
// 1-byte, undocumented TPI param
|
||||||
|
#define XPRG_PARAM_TPI_3 0x03
|
||||||
|
// 1-byte, undocumented TPI param
|
||||||
|
#define XPRG_PARAM_TPI_4 0x04
|
||||||
|
|
||||||
|
// *****************[ STK answer constants ]***************************
|
||||||
|
|
||||||
|
#define ANSWER_CKSUM_ERROR 0xB0
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Private data for this programmer.
|
||||||
|
*/
|
||||||
|
struct pdata
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* See stk500pp_read_byte() for an explanation of the flash and
|
||||||
|
* EEPROM page caches.
|
||||||
|
*/
|
||||||
|
unsigned char *flash_pagecache;
|
||||||
|
unsigned long flash_pageaddr;
|
||||||
|
unsigned int flash_pagesize;
|
||||||
|
|
||||||
|
unsigned char *eeprom_pagecache;
|
||||||
|
unsigned long eeprom_pageaddr;
|
||||||
|
unsigned int eeprom_pagesize;
|
||||||
|
|
||||||
|
unsigned char command_sequence;
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
PGMTYPE_UNKNOWN,
|
||||||
|
PGMTYPE_STK500,
|
||||||
|
PGMTYPE_AVRISP,
|
||||||
|
PGMTYPE_AVRISP_MKII,
|
||||||
|
PGMTYPE_JTAGICE_MKII,
|
||||||
|
PGMTYPE_STK600,
|
||||||
|
}
|
||||||
|
pgmtype;
|
||||||
|
|
||||||
|
AVRPART *lastpart;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Chained pdata for the JTAG ICE mkII backend. This is used when
|
||||||
|
* calling the backend functions for ISP/HVSP/PP programming
|
||||||
|
* functionality of the JTAG ICE mkII and AVR Dragon.
|
||||||
|
*/
|
||||||
|
void *chained_pdata;
|
||||||
|
};
|
||||||
|
|
|
@ -0,0 +1,940 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <limits.h>
|
||||||
|
|
||||||
|
#if defined(HAVE_LIBREADLINE)
|
||||||
|
#if !defined(WIN32NATIVE)
|
||||||
|
# include <readline/readline.h>
|
||||||
|
# include <readline/history.h>
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "config.h"
|
||||||
|
#include "lists.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
#include "pindefs.h"
|
||||||
|
#include "ppi.h"
|
||||||
|
|
||||||
|
struct command {
|
||||||
|
char * name;
|
||||||
|
int (*func)(PROGRAMMER * pgm, struct avrpart * p, int argc, char *argv[]);
|
||||||
|
char * desc;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_dump (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_write (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_erase (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_sig (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_part (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_help (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_quit (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_send (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_parms (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_vtarg (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_varef (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_fosc (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_sck (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_spi (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
static int cmd_pgm (PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char *argv[]);
|
||||||
|
|
||||||
|
struct command cmd[] = {
|
||||||
|
{ "dump", cmd_dump, "dump memory : %s <memtype> <addr> <N-Bytes>" },
|
||||||
|
{ "read", cmd_dump, "alias for dump" },
|
||||||
|
{ "write", cmd_write, "write memory : %s <memtype> <addr> <b1> <b2> ... <bN>" },
|
||||||
|
{ "erase", cmd_erase, "perform a chip erase" },
|
||||||
|
{ "sig", cmd_sig, "display device signature bytes" },
|
||||||
|
{ "part", cmd_part, "display the current part information" },
|
||||||
|
{ "send", cmd_send, "send a raw command : %s <b1> <b2> <b3> <b4>" },
|
||||||
|
{ "parms", cmd_parms, "display adjustable parameters (STK500 only)" },
|
||||||
|
{ "vtarg", cmd_vtarg, "set <V[target]> (STK500 only)" },
|
||||||
|
{ "varef", cmd_varef, "set <V[aref]> (STK500 only)" },
|
||||||
|
{ "fosc", cmd_fosc, "set <oscillator frequency> (STK500 only)" },
|
||||||
|
{ "sck", cmd_sck, "set <SCK period> (STK500 only)" },
|
||||||
|
{ "spi", cmd_spi, "enter direct SPI mode" },
|
||||||
|
{ "pgm", cmd_pgm, "return to programming mode" },
|
||||||
|
{ "help", cmd_help, "help" },
|
||||||
|
{ "?", cmd_help, "help" },
|
||||||
|
{ "quit", cmd_quit, "quit" }
|
||||||
|
};
|
||||||
|
|
||||||
|
#define NCMDS (sizeof(cmd)/sizeof(struct command))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static int spi_mode = 0;
|
||||||
|
|
||||||
|
static int nexttok(char * buf, char ** tok, char ** next)
|
||||||
|
{
|
||||||
|
char * q, * n;
|
||||||
|
|
||||||
|
q = buf;
|
||||||
|
while (isspace((int)*q))
|
||||||
|
q++;
|
||||||
|
|
||||||
|
/* isolate first token */
|
||||||
|
n = q+1;
|
||||||
|
while (*n && !isspace((int)*n))
|
||||||
|
n++;
|
||||||
|
|
||||||
|
if (*n) {
|
||||||
|
*n = 0;
|
||||||
|
n++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* find start of next token */
|
||||||
|
while (isspace((int)*n))
|
||||||
|
n++;
|
||||||
|
|
||||||
|
*tok = q;
|
||||||
|
*next = n;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int hexdump_line(char * buffer, unsigned char * p, int n, int pad)
|
||||||
|
{
|
||||||
|
char * hexdata = "0123456789abcdef";
|
||||||
|
char * b;
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
b = buffer;
|
||||||
|
|
||||||
|
j = 0;
|
||||||
|
for (i=0; i<n; i++) {
|
||||||
|
if (i && ((i % 8) == 0))
|
||||||
|
b[j++] = ' ';
|
||||||
|
b[j++] = hexdata[(p[i] & 0xf0) >> 4];
|
||||||
|
b[j++] = hexdata[(p[i] & 0x0f)];
|
||||||
|
if (i < 15)
|
||||||
|
b[j++] = ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=j; i<pad; i++)
|
||||||
|
b[i] = ' ';
|
||||||
|
|
||||||
|
b[i] = 0;
|
||||||
|
|
||||||
|
for (i=0; i<pad; i++) {
|
||||||
|
if (!((b[i] == '0') || (b[i] == ' ')))
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int chardump_line(char * buffer, unsigned char * p, int n, int pad)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
char b [ 128 ];
|
||||||
|
|
||||||
|
for (i=0; i<n; i++) {
|
||||||
|
memcpy(b, p, n);
|
||||||
|
buffer[i] = '.';
|
||||||
|
if (isalpha((int)(b[i])) || isdigit((int)(b[i])) || ispunct((int)(b[i])))
|
||||||
|
buffer[i] = b[i];
|
||||||
|
else if (isspace((int)(b[i])))
|
||||||
|
buffer[i] = ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=n; i<pad; i++)
|
||||||
|
buffer[i] = ' ';
|
||||||
|
|
||||||
|
buffer[i] = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int hexdump_buf(FILE * f, int startaddr, unsigned char * buf, int len)
|
||||||
|
{
|
||||||
|
int addr;
|
||||||
|
int i, n;
|
||||||
|
unsigned char * p;
|
||||||
|
char dst1[80];
|
||||||
|
char dst2[80];
|
||||||
|
|
||||||
|
addr = startaddr;
|
||||||
|
i = 0;
|
||||||
|
p = (unsigned char *)buf;
|
||||||
|
while (len) {
|
||||||
|
n = 16;
|
||||||
|
if (n > len)
|
||||||
|
n = len;
|
||||||
|
hexdump_line(dst1, p, n, 48);
|
||||||
|
chardump_line(dst2, p, n, 16);
|
||||||
|
fprintf(stdout, "%04x %s |%s|\n", addr, dst1, dst2);
|
||||||
|
len -= n;
|
||||||
|
addr += n;
|
||||||
|
p += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_dump(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
static char prevmem[128] = {0};
|
||||||
|
char * e;
|
||||||
|
unsigned char * buf;
|
||||||
|
int maxsize;
|
||||||
|
unsigned long i;
|
||||||
|
static unsigned long addr=0;
|
||||||
|
static int len=64;
|
||||||
|
AVRMEM * mem;
|
||||||
|
char * memtype = NULL;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
if (!((argc == 2) || (argc == 4))) {
|
||||||
|
fprintf(stderr, "Usage: dump <memtype> [<addr> <len>]\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
memtype = argv[1];
|
||||||
|
|
||||||
|
if (strncmp(prevmem, memtype, strlen(memtype)) != 0) {
|
||||||
|
addr = 0;
|
||||||
|
len = 64;
|
||||||
|
strncpy(prevmem, memtype, sizeof(prevmem)-1);
|
||||||
|
prevmem[sizeof(prevmem)-1] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
mem = avr_locate_mem(p, memtype);
|
||||||
|
if (mem == NULL) {
|
||||||
|
fprintf(stderr, "\"%s\" memory type not defined for part \"%s\"\n",
|
||||||
|
memtype, p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (argc == 4) {
|
||||||
|
addr = strtoul(argv[2], &e, 0);
|
||||||
|
if (*e || (e == argv[2])) {
|
||||||
|
fprintf(stderr, "%s (dump): can't parse address \"%s\"\n",
|
||||||
|
progname, argv[2]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
len = strtol(argv[3], &e, 0);
|
||||||
|
if (*e || (e == argv[3])) {
|
||||||
|
fprintf(stderr, "%s (dump): can't parse length \"%s\"\n",
|
||||||
|
progname, argv[3]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
maxsize = mem->size;
|
||||||
|
|
||||||
|
if (addr >= maxsize) {
|
||||||
|
if (argc == 2) {
|
||||||
|
/* wrap around */
|
||||||
|
addr = 0;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (dump): address 0x%05lx is out of range for %s memory\n",
|
||||||
|
progname, addr, mem->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* trim len if nessary to not read past the end of memory */
|
||||||
|
if ((addr + len) > maxsize)
|
||||||
|
len = maxsize - addr;
|
||||||
|
|
||||||
|
buf = malloc(len);
|
||||||
|
if (buf == NULL) {
|
||||||
|
fprintf(stderr, "%s (dump): out of memory\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=0; i<len; i++) {
|
||||||
|
rc = pgm->read_byte(pgm, p, mem, addr+i, &buf[i]);
|
||||||
|
if (rc != 0) {
|
||||||
|
fprintf(stderr, "error reading %s address 0x%05lx of part %s\n",
|
||||||
|
mem->desc, addr+i, p->desc);
|
||||||
|
if (rc == -1)
|
||||||
|
fprintf(stderr, "read operation not supported on memory type \"%s\"\n",
|
||||||
|
mem->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
hexdump_buf(stdout, addr, buf, len);
|
||||||
|
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
|
||||||
|
free(buf);
|
||||||
|
|
||||||
|
addr = addr + len;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_write(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
char * e;
|
||||||
|
int len, maxsize;
|
||||||
|
char * memtype;
|
||||||
|
unsigned long addr, i;
|
||||||
|
unsigned char * buf;
|
||||||
|
unsigned char b;
|
||||||
|
int rc;
|
||||||
|
int werror;
|
||||||
|
AVRMEM * mem;
|
||||||
|
|
||||||
|
if (argc < 4) {
|
||||||
|
fprintf(stderr, "Usage: write <memtype> <addr> <byte1> "
|
||||||
|
"<byte2> ... byteN>\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
memtype = argv[1];
|
||||||
|
|
||||||
|
mem = avr_locate_mem(p, memtype);
|
||||||
|
if (mem == NULL) {
|
||||||
|
fprintf(stderr, "\"%s\" memory type not defined for part \"%s\"\n",
|
||||||
|
memtype, p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
maxsize = mem->size;
|
||||||
|
|
||||||
|
addr = strtoul(argv[2], &e, 0);
|
||||||
|
if (*e || (e == argv[2])) {
|
||||||
|
fprintf(stderr, "%s (write): can't parse address \"%s\"\n",
|
||||||
|
progname, argv[2]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (addr > maxsize) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (write): address 0x%05lx is out of range for %s memory\n",
|
||||||
|
progname, addr, memtype);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* number of bytes to write at the specified address */
|
||||||
|
len = argc - 3;
|
||||||
|
|
||||||
|
if ((addr + len) > maxsize) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (write): selected address and # bytes exceed "
|
||||||
|
"range for %s memory\n",
|
||||||
|
progname, memtype);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf = malloc(len);
|
||||||
|
if (buf == NULL) {
|
||||||
|
fprintf(stderr, "%s (write): out of memory\n", progname);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i=3; i<argc; i++) {
|
||||||
|
buf[i-3] = strtoul(argv[i], &e, 0);
|
||||||
|
if (*e || (e == argv[i])) {
|
||||||
|
fprintf(stderr, "%s (write): can't parse byte \"%s\"\n",
|
||||||
|
progname, argv[i]);
|
||||||
|
free(buf);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->err_led(pgm, OFF);
|
||||||
|
for (werror=0, i=0; i<len; i++) {
|
||||||
|
|
||||||
|
rc = avr_write_byte(pgm, p, mem, addr+i, buf[i]);
|
||||||
|
if (rc) {
|
||||||
|
fprintf(stderr, "%s (write): error writing 0x%02x at 0x%05lx, rc=%d\n",
|
||||||
|
progname, buf[i], addr+i, rc);
|
||||||
|
if (rc == -1)
|
||||||
|
fprintf(stderr,
|
||||||
|
"write operation not supported on memory type \"%s\"\n",
|
||||||
|
mem->desc);
|
||||||
|
werror = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = pgm->read_byte(pgm, p, mem, addr+i, &b);
|
||||||
|
if (b != buf[i]) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (write): error writing 0x%02x at 0x%05lx cell=0x%02x\n",
|
||||||
|
progname, buf[i], addr+i, b);
|
||||||
|
werror = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (werror) {
|
||||||
|
pgm->err_led(pgm, ON);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
free(buf);
|
||||||
|
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_send(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
unsigned char cmd[4], res[4];
|
||||||
|
char * e;
|
||||||
|
int i;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
if (pgm->cmd == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"The %s programmer does not support direct ISP commands.\n",
|
||||||
|
pgm->type);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (spi_mode && (pgm->spi == NULL)) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"The %s programmer does not support direct SPI transfers.\n",
|
||||||
|
pgm->type);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if ((argc > 5) || ((argc < 5) && (!spi_mode))) {
|
||||||
|
fprintf(stderr, spi_mode?
|
||||||
|
"Usage: send <byte1> [<byte2> [<byte3> [<byte4>]]]\n":
|
||||||
|
"Usage: send <byte1> <byte2> <byte3> <byte4>\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* number of bytes to write at the specified address */
|
||||||
|
len = argc - 1;
|
||||||
|
|
||||||
|
/* load command bytes */
|
||||||
|
for (i=1; i<argc; i++) {
|
||||||
|
cmd[i-1] = strtoul(argv[i], &e, 0);
|
||||||
|
if (*e || (e == argv[i])) {
|
||||||
|
fprintf(stderr, "%s (send): can't parse byte \"%s\"\n",
|
||||||
|
progname, argv[i]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->err_led(pgm, OFF);
|
||||||
|
|
||||||
|
if (spi_mode)
|
||||||
|
pgm->spi(pgm, cmd, res, argc-1);
|
||||||
|
else
|
||||||
|
pgm->cmd(pgm, cmd, res);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* display results
|
||||||
|
*/
|
||||||
|
fprintf(stderr, "results:");
|
||||||
|
for (i=0; i<len; i++)
|
||||||
|
fprintf(stderr, " %02x", res[i]);
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_erase(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: erasing chip\n", progname);
|
||||||
|
pgm->chip_erase(pgm, p);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_part(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
avr_display(stdout, p, "", 0);
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_sig(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int rc;
|
||||||
|
AVRMEM * m;
|
||||||
|
|
||||||
|
rc = avr_signature(pgm, p);
|
||||||
|
if (rc != 0) {
|
||||||
|
fprintf(stderr, "error reading signature data, rc=%d\n",
|
||||||
|
rc);
|
||||||
|
}
|
||||||
|
|
||||||
|
m = avr_locate_mem(p, "signature");
|
||||||
|
if (m == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"signature data not defined for device \"%s\"\n",
|
||||||
|
p->desc);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stdout, "Device signature = 0x");
|
||||||
|
for (i=0; i<m->size; i++)
|
||||||
|
fprintf(stdout, "%02x", m->buf[i]);
|
||||||
|
fprintf(stdout, "\n\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_quit(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_parms(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
if (pgm->print_parms == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (parms): the %s programmer does not support "
|
||||||
|
"adjustable parameters\n",
|
||||||
|
progname, pgm->type);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
pgm->print_parms(pgm);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_vtarg(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
double v;
|
||||||
|
char *endp;
|
||||||
|
|
||||||
|
if (argc != 2) {
|
||||||
|
fprintf(stderr, "Usage: vtarg <value>\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
v = strtod(argv[1], &endp);
|
||||||
|
if (endp == argv[1]) {
|
||||||
|
fprintf(stderr, "%s (vtarg): can't parse voltage \"%s\"\n",
|
||||||
|
progname, argv[1]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (pgm->set_vtarget == NULL) {
|
||||||
|
fprintf(stderr, "%s (vtarg): the %s programmer cannot set V[target]\n",
|
||||||
|
progname, pgm->type);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
if ((rc = pgm->set_vtarget(pgm, v)) != 0) {
|
||||||
|
fprintf(stderr, "%s (vtarg): failed to set V[target] (rc = %d)\n",
|
||||||
|
progname, rc);
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_fosc(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
double v;
|
||||||
|
char *endp;
|
||||||
|
|
||||||
|
if (argc != 2) {
|
||||||
|
fprintf(stderr, "Usage: fosc <value>[M|k] | off\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
v = strtod(argv[1], &endp);
|
||||||
|
if (endp == argv[1]) {
|
||||||
|
if (strcmp(argv[1], "off") == 0)
|
||||||
|
v = 0.0;
|
||||||
|
else {
|
||||||
|
fprintf(stderr, "%s (fosc): can't parse frequency \"%s\"\n",
|
||||||
|
progname, argv[1]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (*endp == 'm' || *endp == 'M')
|
||||||
|
v *= 1e6;
|
||||||
|
else if (*endp == 'k' || *endp == 'K')
|
||||||
|
v *= 1e3;
|
||||||
|
if (pgm->set_fosc == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (fosc): the %s programmer cannot set oscillator frequency\n",
|
||||||
|
progname, pgm->type);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
if ((rc = pgm->set_fosc(pgm, v)) != 0) {
|
||||||
|
fprintf(stderr, "%s (fosc): failed to set oscillator_frequency (rc = %d)\n",
|
||||||
|
progname, rc);
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_sck(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
double v;
|
||||||
|
char *endp;
|
||||||
|
|
||||||
|
if (argc != 2) {
|
||||||
|
fprintf(stderr, "Usage: sck <value>\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
v = strtod(argv[1], &endp);
|
||||||
|
if (endp == argv[1]) {
|
||||||
|
fprintf(stderr, "%s (sck): can't parse period \"%s\"\n",
|
||||||
|
progname, argv[1]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
v *= 1e-6; /* Convert from microseconds to seconds. */
|
||||||
|
if (pgm->set_sck_period == NULL) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s (sck): the %s programmer cannot set SCK period\n",
|
||||||
|
progname, pgm->type);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
if ((rc = pgm->set_sck_period(pgm, v)) != 0) {
|
||||||
|
fprintf(stderr, "%s (sck): failed to set SCK period (rc = %d)\n",
|
||||||
|
progname, rc);
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_varef(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
unsigned int chan;
|
||||||
|
double v;
|
||||||
|
char *endp;
|
||||||
|
|
||||||
|
if (argc != 2 && argc != 3) {
|
||||||
|
fprintf(stderr, "Usage: varef [channel] <value>\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (argc == 2) {
|
||||||
|
chan = 0;
|
||||||
|
v = strtod(argv[1], &endp);
|
||||||
|
if (endp == argv[1]) {
|
||||||
|
fprintf(stderr, "%s (varef): can't parse voltage \"%s\"\n",
|
||||||
|
progname, argv[1]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
chan = strtoul(argv[1], &endp, 10);
|
||||||
|
if (endp == argv[1]) {
|
||||||
|
fprintf(stderr, "%s (varef): can't parse channel \"%s\"\n",
|
||||||
|
progname, argv[1]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
v = strtod(argv[2], &endp);
|
||||||
|
if (endp == argv[2]) {
|
||||||
|
fprintf(stderr, "%s (varef): can't parse voltage \"%s\"\n",
|
||||||
|
progname, argv[2]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (pgm->set_varef == NULL) {
|
||||||
|
fprintf(stderr, "%s (varef): the %s programmer cannot set V[aref]\n",
|
||||||
|
progname, pgm->type);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
if ((rc = pgm->set_varef(pgm, chan, v)) != 0) {
|
||||||
|
fprintf(stderr, "%s (varef): failed to set V[aref] (rc = %d)\n",
|
||||||
|
progname, rc);
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int cmd_help(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
fprintf(stdout, "Valid commands:\n\n");
|
||||||
|
for (i=0; i<NCMDS; i++) {
|
||||||
|
fprintf(stdout, " %-6s : ", cmd[i].name);
|
||||||
|
fprintf(stdout, cmd[i].desc, cmd[i].name);
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
}
|
||||||
|
fprintf(stdout,
|
||||||
|
"\nUse the 'part' command to display valid memory types for use with the\n"
|
||||||
|
"'dump' and 'write' commands.\n\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cmd_spi(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_RESET], 1);
|
||||||
|
spi_mode = 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cmd_pgm(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
pgm->setpin(pgm, pgm->pinno[PIN_AVR_RESET], 0);
|
||||||
|
spi_mode = 0;
|
||||||
|
pgm->initialize(pgm, p);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int tokenize(char * s, char *** argv)
|
||||||
|
{
|
||||||
|
int i, n, l, nargs, offset;
|
||||||
|
int len, slen;
|
||||||
|
char * buf;
|
||||||
|
int bufsize;
|
||||||
|
char ** bufv;
|
||||||
|
char * q, * r;
|
||||||
|
char * nbuf;
|
||||||
|
char ** av;
|
||||||
|
|
||||||
|
slen = strlen(s);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* initialize allow for 20 arguments, use realloc to grow this if
|
||||||
|
* necessary
|
||||||
|
*/
|
||||||
|
nargs = 20;
|
||||||
|
bufsize = slen + 20;
|
||||||
|
buf = malloc(bufsize);
|
||||||
|
bufv = (char **) malloc(nargs*sizeof(char *));
|
||||||
|
for (i=0; i<nargs; i++) {
|
||||||
|
bufv[i] = NULL;
|
||||||
|
}
|
||||||
|
buf[0] = 0;
|
||||||
|
|
||||||
|
n = 0;
|
||||||
|
l = 0;
|
||||||
|
nbuf = buf;
|
||||||
|
r = s;
|
||||||
|
while (*r) {
|
||||||
|
nexttok(r, &q, &r);
|
||||||
|
strcpy(nbuf, q);
|
||||||
|
bufv[n] = nbuf;
|
||||||
|
len = strlen(q);
|
||||||
|
l += len + 1;
|
||||||
|
nbuf += len + 1;
|
||||||
|
nbuf[0] = 0;
|
||||||
|
n++;
|
||||||
|
if ((n % 20) == 0) {
|
||||||
|
/* realloc space for another 20 args */
|
||||||
|
bufsize += 20;
|
||||||
|
nargs += 20;
|
||||||
|
buf = realloc(buf, bufsize);
|
||||||
|
bufv = realloc(bufv, nargs*sizeof(char *));
|
||||||
|
nbuf = &buf[l];
|
||||||
|
for (i=n; i<nargs; i++)
|
||||||
|
bufv[i] = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We have parsed all the args, n == argc, bufv contains an array of
|
||||||
|
* pointers to each arg, and buf points to one memory block that
|
||||||
|
* contains all the args, back to back, seperated by a nul
|
||||||
|
* terminator. Consilidate bufv and buf into one big memory block
|
||||||
|
* so that the code that calls us, will have an easy job of freeing
|
||||||
|
* this memory.
|
||||||
|
*/
|
||||||
|
av = (char **) malloc(slen + n + (n+1)*sizeof(char *));
|
||||||
|
q = (char *)&av[n+1];
|
||||||
|
memcpy(q, buf, l);
|
||||||
|
for (i=0; i<n; i++) {
|
||||||
|
offset = bufv[i] - buf;
|
||||||
|
av[i] = q + offset;
|
||||||
|
}
|
||||||
|
av[i] = NULL;
|
||||||
|
|
||||||
|
free(buf);
|
||||||
|
free(bufv);
|
||||||
|
|
||||||
|
*argv = av;
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int do_cmd(PROGRAMMER * pgm, struct avrpart * p,
|
||||||
|
int argc, char * argv[])
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int hold;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
len = strlen(argv[0]);
|
||||||
|
hold = -1;
|
||||||
|
for (i=0; i<NCMDS; i++) {
|
||||||
|
if (strcasecmp(argv[0], cmd[i].name) == 0) {
|
||||||
|
return cmd[i].func(pgm, p, argc, argv);
|
||||||
|
}
|
||||||
|
else if (strncasecmp(argv[0], cmd[i].name, len)==0) {
|
||||||
|
if (hold != -1) {
|
||||||
|
fprintf(stderr, "%s: command \"%s\" is ambiguous\n",
|
||||||
|
progname, argv[0]);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
hold = i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hold != -1)
|
||||||
|
return cmd[hold].func(pgm, p, argc, argv);
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: invalid command \"%s\"\n",
|
||||||
|
progname, argv[0]);
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
char * terminal_get_input(const char *prompt)
|
||||||
|
{
|
||||||
|
#if defined(HAVE_LIBREADLINE) && !defined(WIN32NATIVE)
|
||||||
|
char *input;
|
||||||
|
input = readline(prompt);
|
||||||
|
if ((input != NULL) && (strlen(input) >= 1))
|
||||||
|
add_history(input);
|
||||||
|
|
||||||
|
return input;
|
||||||
|
#else
|
||||||
|
char input[256];
|
||||||
|
printf("%s", prompt);
|
||||||
|
if (fgets(input, sizeof(input), stdin))
|
||||||
|
{
|
||||||
|
/* FIXME: readline strips the '\n', should this too? */
|
||||||
|
return strdup(input);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return NULL;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int terminal_mode(PROGRAMMER * pgm, struct avrpart * p)
|
||||||
|
{
|
||||||
|
char * cmdbuf;
|
||||||
|
int i;
|
||||||
|
char * q;
|
||||||
|
int rc;
|
||||||
|
int argc;
|
||||||
|
char ** argv;
|
||||||
|
|
||||||
|
rc = 0;
|
||||||
|
while ((cmdbuf = terminal_get_input("avrdude> ")) != NULL) {
|
||||||
|
/*
|
||||||
|
* find the start of the command, skipping any white space
|
||||||
|
*/
|
||||||
|
q = cmdbuf;
|
||||||
|
while (*q && isspace((int)*q))
|
||||||
|
q++;
|
||||||
|
|
||||||
|
/* skip blank lines and comments */
|
||||||
|
if (!*q || (*q == '#'))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* tokenize command line */
|
||||||
|
argc = tokenize(q, &argv);
|
||||||
|
|
||||||
|
fprintf(stdout, ">>> ");
|
||||||
|
for (i=0; i<argc; i++)
|
||||||
|
fprintf(stdout, "%s ", argv[i]);
|
||||||
|
fprintf(stdout, "\n");
|
||||||
|
|
||||||
|
/* run the command */
|
||||||
|
rc = do_cmd(pgm, p, argc, argv);
|
||||||
|
free(argv);
|
||||||
|
if (rc > 0) {
|
||||||
|
rc = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
free(cmdbuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2004 Brian S. Dean <bsd@bsdhome.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef term_h
|
||||||
|
#define term_h
|
||||||
|
|
||||||
|
#include "avr.h"
|
||||||
|
#include "pgm.h"
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int terminal_mode(PROGRAMMER * pgm, struct avrpart * p);
|
||||||
|
char * terminal_get_input(const char *prompt);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,156 @@
|
||||||
|
<?xml version="1.0" encoding='ISO-8859-1' ?>
|
||||||
|
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
|
||||||
|
<!--
|
||||||
|
* Copyright (c) 2006 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
-->
|
||||||
|
<!--
|
||||||
|
* Extract the debugWire parameters
|
||||||
|
* from the XML, and format it the way src/devdescr.cc needs it.
|
||||||
|
*
|
||||||
|
* Run this file together with the respective AVR's XML file through
|
||||||
|
* an XSLT processor (xsltproc, saxon), and capture the output for
|
||||||
|
* inclusion into avrdude.conf.in.
|
||||||
|
-->
|
||||||
|
<xsl:output method="text"/>
|
||||||
|
<xsl:template match="/">
|
||||||
|
<!-- Extract everything we need out of the XML. -->
|
||||||
|
<xsl:variable name="devname_orig"
|
||||||
|
select="/AVRPART/ADMIN/PART_NAME" />
|
||||||
|
<xsl:variable name="devname"
|
||||||
|
select="translate(/AVRPART/ADMIN/PART_NAME,
|
||||||
|
'abcdefghijklmnopqrstuvwxyz',
|
||||||
|
'ABCDEFGHIJKLMNOPQRSTUVWXYZ')" />
|
||||||
|
<xsl:variable name="devname_lower"
|
||||||
|
select="translate(/AVRPART/ADMIN/PART_NAME,
|
||||||
|
'ABCDEFGHIJKLMNOPQRSTUVWXYZ',
|
||||||
|
'abcdefghijklmnopqrstuvwxyz')" />
|
||||||
|
<xsl:variable name="ucEepromInst"
|
||||||
|
select="//AVRPART/ICE_SETTINGS/JTAGICEmkII/ucEepromInst" />
|
||||||
|
<xsl:variable name="ucFlashInst"
|
||||||
|
select="//AVRPART/ICE_SETTINGS/JTAGICEmkII/ucFlashInst" />
|
||||||
|
|
||||||
|
<!-- If there's a JTAGICEmkII node indicating debugWire, emit the entry. -->
|
||||||
|
<xsl:if test='//AVRPART/ICE_SETTINGS/JTAGICEmkII/Interface="DebugWire"'>
|
||||||
|
|
||||||
|
<!-- start of new entry -->
|
||||||
|
<xsl:text>#------------------------------------------------------------
</xsl:text>
|
||||||
|
<xsl:text># </xsl:text>
|
||||||
|
<xsl:value-of select="$devname_orig" />
|
||||||
|
<xsl:text>
</xsl:text>
|
||||||
|
<xsl:text>#------------------------------------------------------------
</xsl:text>
|
||||||
|
<xsl:text>part
 desc = "</xsl:text>
|
||||||
|
<xsl:value-of select="$devname_orig" />
|
||||||
|
<xsl:text>";
 has_debugwire = yes;
</xsl:text>
|
||||||
|
|
||||||
|
<xsl:text> flash_instr = </xsl:text>
|
||||||
|
<xsl:call-template name="format-hex">
|
||||||
|
<xsl:with-param name="arg" select="$ucFlashInst" />
|
||||||
|
<xsl:with-param name="count" select="0" />
|
||||||
|
</xsl:call-template>
|
||||||
|
<xsl:text>;
</xsl:text>
|
||||||
|
|
||||||
|
<xsl:text> eeprom_instr = </xsl:text>
|
||||||
|
<xsl:call-template name="format-hex">
|
||||||
|
<xsl:with-param name="arg" select="$ucEepromInst" />
|
||||||
|
<xsl:with-param name="count" select="0" />
|
||||||
|
</xsl:call-template>
|
||||||
|
<xsl:text>;
</xsl:text>
|
||||||
|
|
||||||
|
</xsl:if> <!-- JTAGICEmkII uses debugWire -->
|
||||||
|
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
<xsl:template name="toupper">
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
<!-- return argument $arg if non-empty, 0 otherwise -->
|
||||||
|
<xsl:template name="maybezero">
|
||||||
|
<xsl:param name="arg" />
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="string-length($arg) = 0"><xsl:text>0</xsl:text></xsl:when>
|
||||||
|
<xsl:otherwise><xsl:value-of select="$arg" /></xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
</xsl:template> <!-- maybezero -->
|
||||||
|
|
||||||
|
<!-- convert $XX hex number in $arg (if any) into 0xXX; -->
|
||||||
|
<!-- return 0 if $arg is empty -->
|
||||||
|
<xsl:template name="dollar-to-0x">
|
||||||
|
<xsl:param name="arg" />
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="string-length($arg) = 0">
|
||||||
|
<xsl:text>0</xsl:text>
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:when test="substring($arg, 1, 1) = '$'">
|
||||||
|
<xsl:text>0x</xsl:text>
|
||||||
|
<xsl:value-of select="substring($arg, 2)" />
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<xsl:value-of select="$arg" />
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
</xsl:template> <!-- dollar-to-0x -->
|
||||||
|
|
||||||
|
<!-- Format a string of 0xXX numbers: start a new line -->
|
||||||
|
<!-- after each 8 hex numbers -->
|
||||||
|
<!-- call with parameter $count = 0, calls itself -->
|
||||||
|
<!-- recursively then until everything has been done -->
|
||||||
|
<xsl:template name="format-hex">
|
||||||
|
<xsl:param name="arg" />
|
||||||
|
<xsl:param name="count" />
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="string-length($arg) <= 4">
|
||||||
|
<!-- Last element, print it, and leave template. -->
|
||||||
|
<xsl:value-of select="$arg" />
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<!--
|
||||||
|
* More arguments follow, print first value,
|
||||||
|
* followed by a comma, decide whether a space
|
||||||
|
* or a newline needs to be emitted, and recurse
|
||||||
|
* with the remaining part of $arg.
|
||||||
|
-->
|
||||||
|
<xsl:value-of select="substring($arg, 1, 4)" />
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="$count mod 8 = 7">
|
||||||
|
<xsl:text>,
	 </xsl:text>
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<xsl:text>, </xsl:text>
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
<xsl:variable name="newarg">
|
||||||
|
<!-- see whether there is a space after comma -->
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="substring($arg, 6, 1) = ' '">
|
||||||
|
<xsl:value-of select="substring($arg, 7)" />
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<xsl:value-of select="substring($arg, 6)" />
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
</xsl:variable>
|
||||||
|
<xsl:call-template name="format-hex">
|
||||||
|
<xsl:with-param name="arg" select="$newarg" />
|
||||||
|
<xsl:with-param name="count" select="$count + 1" />
|
||||||
|
</xsl:call-template>
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
</xsl:stylesheet>
|
|
@ -0,0 +1,255 @@
|
||||||
|
<?xml version="1.0" encoding='ISO-8859-1' ?>
|
||||||
|
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
|
||||||
|
<!--
|
||||||
|
* Copyright (c) 2006 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
-->
|
||||||
|
<!--
|
||||||
|
* Extract high-voltage (parallel and serial) programming parameters
|
||||||
|
* out of the Atmel XML files, and convert them into avrdude.conf
|
||||||
|
* snippets.
|
||||||
|
*
|
||||||
|
* Run this file together with the respective AVR's XML file through
|
||||||
|
* an XSLT processor (xsltproc, saxon), and capture the output for
|
||||||
|
* inclusion into avrdude.conf.in.
|
||||||
|
-->
|
||||||
|
<xsl:output method="text"/>
|
||||||
|
<xsl:template match="/">
|
||||||
|
<xsl:for-each select="//*">
|
||||||
|
<xsl:if test='name() = "STK500_2"'>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
* High-voltage parallel programming parameters.
|
||||||
|
-->
|
||||||
|
<xsl:for-each
|
||||||
|
select="*[starts-with(translate(name(),
|
||||||
|
'abcdefghijklmnopqrstuvwxyz',
|
||||||
|
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'),
|
||||||
|
'PP')]">
|
||||||
|
<xsl:if test="self::node()[name() = 'PPControlStack']"
|
||||||
|
> pp_controlstack =
|
||||||
|
<xsl:call-template name="format_cstack">
|
||||||
|
<xsl:with-param name="stack" select="." />
|
||||||
|
<xsl:with-param name="count" select="0" />
|
||||||
|
</xsl:call-template>;
|
||||||
|
</xsl:if> <!-- PPControlStack -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'PpEnterProgMode']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'stabDelay']"
|
||||||
|
> hventerstabdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'progModeDelay']"
|
||||||
|
> progmodedelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'latchCycles']"
|
||||||
|
> latchcycles = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'toggleVtg']"
|
||||||
|
> togglevtg = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'powerOffDelay']"
|
||||||
|
> poweroffdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'resetDelayMs']"
|
||||||
|
> resetdelayms = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'resetDelayUs']"
|
||||||
|
> resetdelayus = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- PpEnterProgMode -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'PpLeaveProgMode']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'stabDelay']"
|
||||||
|
> hvleavestabdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- PpLeaveProgMode -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'PpChipErase']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'pulseWidth']"
|
||||||
|
> chiperasepulsewidth = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'pollTimeout']"
|
||||||
|
> chiperasepolltimeout = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- PpChipErase -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'PpProgramFuse']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'pulseWidth']"
|
||||||
|
> programfusepulsewidth = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'pollTimeout']"
|
||||||
|
> programfusepolltimeout = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- PpProgramFuse -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'PpProgramLock']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'pulseWidth']"
|
||||||
|
> programlockpulsewidth = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'pollTimeout']"
|
||||||
|
> programlockpolltimeout = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- PpProgramLock -->
|
||||||
|
|
||||||
|
</xsl:for-each> <!-- PP parameters -->
|
||||||
|
|
||||||
|
<!--
|
||||||
|
* High-voltage serial programming parameters.
|
||||||
|
-->
|
||||||
|
<xsl:for-each
|
||||||
|
select="*[starts-with(translate(name(),
|
||||||
|
'abcdefghijklmnopqrstuvwxyz',
|
||||||
|
'ABCDEFGHIJKLMNOPQRSTUVWXYZ'),
|
||||||
|
'HVSP')]">
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'HvspControlStack']"
|
||||||
|
> hvsp_controlstack =
|
||||||
|
<xsl:call-template name="format_cstack">
|
||||||
|
<xsl:with-param name="stack" select="." />
|
||||||
|
<xsl:with-param name="count" select="0" />
|
||||||
|
</xsl:call-template>;
|
||||||
|
</xsl:if> <!-- HvspControlStack -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'HvspEnterProgMode']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'stabDelay']"
|
||||||
|
> hventerstabdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'cmdexeDelay']"
|
||||||
|
> hvspcmdexedelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'synchCycles']"
|
||||||
|
> synchcycles = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'latchCycles']"
|
||||||
|
> latchcycles = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'toggleVtg']"
|
||||||
|
> togglevtg = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'powoffDelay']"
|
||||||
|
> poweroffdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'resetDelay1']"
|
||||||
|
> resetdelayms = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'resetDelay2']"
|
||||||
|
> resetdelayus = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- HvspEnterProgMode -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'HvspLeaveProgMode']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'stabDelay']"
|
||||||
|
> hvleavestabdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'resetDelay']"
|
||||||
|
> resetdelay = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- HvspLeaveProgMode -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'HvspChipErase']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'pollTimeout']"
|
||||||
|
> chiperasepolltimeout = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
<xsl:if test="self::node()[name() = 'eraseTime']"
|
||||||
|
> chiperasetime = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- HvspChipErase -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'HvspProgramFuse']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'pollTimeout']"
|
||||||
|
> programfusepolltimeout = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- HvspProgramFuse -->
|
||||||
|
|
||||||
|
<xsl:if test="self::node()[name() = 'HvspProgramLock']">
|
||||||
|
<xsl:for-each select="*">
|
||||||
|
<xsl:if test="self::node()[name() = 'pollTimeout']"
|
||||||
|
> programlockpolltimeout = <xsl:value-of select="." />;
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each>
|
||||||
|
</xsl:if> <!-- HvspProgramLock -->
|
||||||
|
|
||||||
|
</xsl:for-each> <!-- HVSP parameters -->
|
||||||
|
|
||||||
|
</xsl:if> <!-- STK500_2 parameters -->
|
||||||
|
</xsl:for-each> <!-- All nodes -->
|
||||||
|
|
||||||
|
</xsl:template>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
* Format the control stack argument: replace space-separated
|
||||||
|
* list by a list separated with commas, followed by either
|
||||||
|
* a space or a newline, dependend on the current argument
|
||||||
|
* count.
|
||||||
|
* This template calls itself recursively, until the entire
|
||||||
|
* argument $stack has been processed.
|
||||||
|
-->
|
||||||
|
<xsl:template name="format_cstack">
|
||||||
|
<xsl:param name="stack" />
|
||||||
|
<xsl:param name="count" />
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="string-length($stack) <= 4">
|
||||||
|
<!-- Last element, print it, and leave template. -->
|
||||||
|
<xsl:value-of select="$stack" />
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<!--
|
||||||
|
* More arguments follow, print first value,
|
||||||
|
* followed by a comma, decide whether a space
|
||||||
|
* or a newline needs to be emitted, and recurse
|
||||||
|
* with the remaining part of $stack.
|
||||||
|
-->
|
||||||
|
<xsl:value-of select="substring($stack, 1, 4)" />
|
||||||
|
<xsl:choose>
|
||||||
|
<xsl:when test="$count mod 8 = 7">
|
||||||
|
<!-- comma, newline, 8 spaces indentation -->
|
||||||
|
<xsl:text>,
|
||||||
|
</xsl:text>
|
||||||
|
</xsl:when>
|
||||||
|
<xsl:otherwise>
|
||||||
|
<!-- comma, space -->
|
||||||
|
<xsl:text>, </xsl:text>
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
<xsl:call-template name="format_cstack">
|
||||||
|
<xsl:with-param name="stack" select="substring($stack, 6)"
|
||||||
|
/>
|
||||||
|
<xsl:with-param name="count" select="$count + 1" />
|
||||||
|
</xsl:call-template>
|
||||||
|
</xsl:otherwise>
|
||||||
|
</xsl:choose>
|
||||||
|
</xsl:template>
|
||||||
|
</xsl:stylesheet>
|
|
@ -0,0 +1,63 @@
|
||||||
|
<?xml version="1.0" encoding='ISO-8859-1' ?>
|
||||||
|
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
|
||||||
|
<!--
|
||||||
|
* Copyright (c) 2008 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
-->
|
||||||
|
<!--
|
||||||
|
* Extract STK600 routing and socket card information out of
|
||||||
|
* targetboards.xml.
|
||||||
|
*
|
||||||
|
* Run this like:
|
||||||
|
*
|
||||||
|
* xsltproc -param what "'RC'" \
|
||||||
|
* tools/get-stk600-cards.xsl targetboard.xml | sort -u
|
||||||
|
*
|
||||||
|
* xsltproc -param what "'SC'" \
|
||||||
|
* tools/get-stk600-cards.xsl targetboard.xml | sort -u
|
||||||
|
*
|
||||||
|
* and copy&paste the results into the respective tables.
|
||||||
|
-->
|
||||||
|
<xsl:output method="text"/>
|
||||||
|
<xsl:template match="/">
|
||||||
|
<xsl:if test="$what = 'RC'">
|
||||||
|
<xsl:for-each select="/STK600/ROUTING/CARD">
|
||||||
|
<xsl:if test="RC_NAME != ''">
|
||||||
|
<xsl:text> { </xsl:text>
|
||||||
|
<xsl:value-of select="RC_ID" />
|
||||||
|
<xsl:text>, "</xsl:text>
|
||||||
|
<xsl:value-of select="RC_NAME" />
|
||||||
|
<xsl:text>" },
</xsl:text>
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each> <!-- All cards -->
|
||||||
|
</xsl:if> <!-- Routing cards -->
|
||||||
|
|
||||||
|
<xsl:if test="$what = 'SC'">
|
||||||
|
<xsl:for-each select="/STK600/ROUTING/CARD">
|
||||||
|
<xsl:if test="SC_NAME != ''">
|
||||||
|
<xsl:text> { </xsl:text>
|
||||||
|
<xsl:value-of select="SC_ID" />
|
||||||
|
<xsl:text>, "</xsl:text>
|
||||||
|
<xsl:value-of select="SC_NAME" />
|
||||||
|
<xsl:text>" },
</xsl:text>
|
||||||
|
</xsl:if>
|
||||||
|
</xsl:for-each> <!-- All cards -->
|
||||||
|
</xsl:if> <!-- Socket cards -->
|
||||||
|
|
||||||
|
</xsl:template>
|
||||||
|
</xsl:stylesheet>
|
|
@ -0,0 +1,52 @@
|
||||||
|
<?xml version="1.0" encoding='ISO-8859-1' ?>
|
||||||
|
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
|
||||||
|
<!--
|
||||||
|
* Copyright (c) 2008 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
-->
|
||||||
|
<!--
|
||||||
|
* Extract STK600 device support out of
|
||||||
|
* targetboards.xml.
|
||||||
|
*
|
||||||
|
* Run this like:
|
||||||
|
*
|
||||||
|
* xsltproc \
|
||||||
|
* tools/get-stk600-devices.xsl targetboard.xml
|
||||||
|
*
|
||||||
|
* and copy&paste the results into the respective table.
|
||||||
|
-->
|
||||||
|
<xsl:output method="text"/>
|
||||||
|
<xsl:template match="/">
|
||||||
|
<xsl:text>@multitable @columnfractions .15 .15 .6
</xsl:text>
|
||||||
|
<xsl:text>Routing card @tab Socket card @tab Devices
</xsl:text>
|
||||||
|
<xsl:for-each select="/STK600/ROUTING/CARD">
|
||||||
|
<xsl:text>@item @code{</xsl:text>
|
||||||
|
<xsl:value-of select="RC_NAME" />
|
||||||
|
<xsl:text>} @tab @code{</xsl:text>
|
||||||
|
<xsl:value-of select="SC_NAME" />
|
||||||
|
<xsl:text>} @tab</xsl:text>
|
||||||
|
<xsl:for-each select="TARGET">
|
||||||
|
<xsl:text> </xsl:text>
|
||||||
|
<xsl:value-of select="NAME" />
|
||||||
|
</xsl:for-each>
|
||||||
|
<xsl:text>
</xsl:text>
|
||||||
|
</xsl:for-each> <!-- All cards -->
|
||||||
|
<xsl:text>@end multitable
</xsl:text>
|
||||||
|
|
||||||
|
</xsl:template>
|
||||||
|
</xsl:stylesheet>
|
|
@ -0,0 +1,72 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2011 Darell Tan <darell.tan@gmail.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, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef tpi_h
|
||||||
|
#define tpi_h
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static const unsigned char tpi_skey[] = { 0x12, 0x89, 0xAB, 0x45, 0xCD, 0xD8, 0x88, 0xFF };
|
||||||
|
|
||||||
|
/* registers */
|
||||||
|
#define TPI_REG_TPIIR 0x0F
|
||||||
|
#define TPI_REG_TPIPCR 0x02
|
||||||
|
#define TPI_REG_TPISR 0x00
|
||||||
|
|
||||||
|
#define TPI_REG_TPISR_NVMEN (1 << 1)
|
||||||
|
|
||||||
|
/* TPI commands */
|
||||||
|
#define TPI_CMD_SLD 0x20
|
||||||
|
#define TPI_CMD_SLD_PI 0x24
|
||||||
|
#define TPI_CMD_SIN 0x10
|
||||||
|
#define TPI_CMD_SOUT 0x90
|
||||||
|
#define TPI_CMD_SSTCS 0xC0
|
||||||
|
#define TPI_CMD_SST 0x60
|
||||||
|
#define TPI_CMD_SST_PI 0x64
|
||||||
|
|
||||||
|
#define TPI_CMD_SLDCS 0x80
|
||||||
|
#define TPI_CMD_SSTPR 0x68
|
||||||
|
#define TPI_CMD_SKEY 0xE0
|
||||||
|
|
||||||
|
/* for TPI_CMD_SIN & TPI_CMD_SOUT */
|
||||||
|
#define TPI_SIO_ADDR(x) ((x & 0x30) << 1 | (x & 0x0F))
|
||||||
|
|
||||||
|
/* ATtiny4/5/9/10 I/O registers */
|
||||||
|
#define TPI_IOREG_NVMCSR 0x32
|
||||||
|
#define TPI_IOREG_NVMCMD 0x33
|
||||||
|
|
||||||
|
/* bit for NVMCSR */
|
||||||
|
#define TPI_IOREG_NVMCSR_NVMBSY (1 << 7)
|
||||||
|
|
||||||
|
/* NVM commands */
|
||||||
|
#define TPI_NVMCMD_NO_OPERATION 0x00
|
||||||
|
#define TPI_NVMCMD_CHIP_ERASE 0x10
|
||||||
|
#define TPI_NVMCMD_SECTION_ERASE 0x14
|
||||||
|
#define TPI_NVMCMD_WORD_WRITE 0x1D
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -0,0 +1,372 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2005 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2007 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "avr.h"
|
||||||
|
#include "config.h"
|
||||||
|
#include "confwin.h"
|
||||||
|
#include "fileio.h"
|
||||||
|
#include "update.h"
|
||||||
|
|
||||||
|
UPDATE * parse_op(char * s)
|
||||||
|
{
|
||||||
|
char buf[1024];
|
||||||
|
char * p, * cp, c;
|
||||||
|
UPDATE * upd;
|
||||||
|
int i;
|
||||||
|
size_t fnlen;
|
||||||
|
|
||||||
|
upd = (UPDATE *)malloc(sizeof(UPDATE));
|
||||||
|
if (upd == NULL) {
|
||||||
|
fprintf(stderr, "%s: out of memory\n", progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
i = 0;
|
||||||
|
p = s;
|
||||||
|
while ((i < (sizeof(buf)-1) && *p && (*p != ':')))
|
||||||
|
buf[i++] = *p++;
|
||||||
|
buf[i] = 0;
|
||||||
|
|
||||||
|
if (*p != ':') {
|
||||||
|
upd->memtype = (char *)malloc(strlen("flash")+1);
|
||||||
|
if (upd->memtype == NULL) {
|
||||||
|
outofmem:
|
||||||
|
fprintf(stderr, "%s: out of memory\n", progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
strcpy(upd->memtype, "flash");
|
||||||
|
upd->op = DEVICE_WRITE;
|
||||||
|
upd->filename = (char *)malloc(strlen(buf) + 1);
|
||||||
|
if (upd->filename == NULL)
|
||||||
|
goto outofmem;
|
||||||
|
strcpy(upd->filename, buf);
|
||||||
|
upd->format = FMT_AUTO;
|
||||||
|
return upd;
|
||||||
|
}
|
||||||
|
|
||||||
|
upd->memtype = (char *)malloc(strlen(buf)+1);
|
||||||
|
if (upd->memtype == NULL) {
|
||||||
|
fprintf(stderr, "%s: out of memory\n", progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
strcpy(upd->memtype, buf);
|
||||||
|
|
||||||
|
p++;
|
||||||
|
if (*p == 'r') {
|
||||||
|
upd->op = DEVICE_READ;
|
||||||
|
}
|
||||||
|
else if (*p == 'w') {
|
||||||
|
upd->op = DEVICE_WRITE;
|
||||||
|
}
|
||||||
|
else if (*p == 'v') {
|
||||||
|
upd->op = DEVICE_VERIFY;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, "%s: invalid I/O mode '%c' in update specification\n",
|
||||||
|
progname, *p);
|
||||||
|
fprintf(stderr,
|
||||||
|
" allowed values are:\n"
|
||||||
|
" r = read device\n"
|
||||||
|
" w = write device\n"
|
||||||
|
" v = verify device\n");
|
||||||
|
free(upd->memtype);
|
||||||
|
free(upd);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
p++;
|
||||||
|
|
||||||
|
if (*p != ':') {
|
||||||
|
fprintf(stderr, "%s: invalid update specification\n", progname);
|
||||||
|
free(upd->memtype);
|
||||||
|
free(upd);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
p++;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Now, parse the filename component. Instead of looking for the
|
||||||
|
* leftmost possible colon delimiter, we look for the rightmost one.
|
||||||
|
* If we found one, we do have a trailing :format specifier, and
|
||||||
|
* process it. Otherwise, the remainder of the string is our file
|
||||||
|
* name component. That way, the file name itself is allowed to
|
||||||
|
* contain a colon itself (e. g. C:/some/file.hex), except the
|
||||||
|
* optional format specifier becomes mandatory then.
|
||||||
|
*/
|
||||||
|
cp = p;
|
||||||
|
p = strrchr(cp, ':');
|
||||||
|
if (p == NULL) {
|
||||||
|
upd->format = FMT_AUTO;
|
||||||
|
fnlen = strlen(cp);
|
||||||
|
upd->filename = (char *)malloc(fnlen + 1);
|
||||||
|
} else {
|
||||||
|
fnlen = p - cp;
|
||||||
|
upd->filename = (char *)malloc(fnlen +1);
|
||||||
|
c = *++p;
|
||||||
|
if (c && p[1])
|
||||||
|
/* More than one char - force failure below. */
|
||||||
|
c = '?';
|
||||||
|
switch (c) {
|
||||||
|
case 'a': upd->format = FMT_AUTO; break;
|
||||||
|
case 's': upd->format = FMT_SREC; break;
|
||||||
|
case 'i': upd->format = FMT_IHEX; break;
|
||||||
|
case 'r': upd->format = FMT_RBIN; break;
|
||||||
|
case 'm': upd->format = FMT_IMM; break;
|
||||||
|
case 'b': upd->format = FMT_BIN; break;
|
||||||
|
case 'd': upd->format = FMT_DEC; break;
|
||||||
|
case 'h': upd->format = FMT_HEX; break;
|
||||||
|
case 'o': upd->format = FMT_OCT; break;
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "%s: invalid file format '%s' in update specifier\n",
|
||||||
|
progname, p);
|
||||||
|
free(upd->memtype);
|
||||||
|
free(upd);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (upd->filename == NULL) {
|
||||||
|
fprintf(stderr, "%s: out of memory\n", progname);
|
||||||
|
free(upd->memtype);
|
||||||
|
free(upd);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
memcpy(upd->filename, cp, fnlen);
|
||||||
|
upd->filename[fnlen] = 0;
|
||||||
|
|
||||||
|
return upd;
|
||||||
|
}
|
||||||
|
|
||||||
|
UPDATE * dup_update(UPDATE * upd)
|
||||||
|
{
|
||||||
|
UPDATE * u;
|
||||||
|
|
||||||
|
u = (UPDATE *)malloc(sizeof(UPDATE));
|
||||||
|
if (u == NULL) {
|
||||||
|
fprintf(stderr, "%s: out of memory\n", progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(u, upd, sizeof(UPDATE));
|
||||||
|
|
||||||
|
u->memtype = strdup(upd->memtype);
|
||||||
|
u->filename = strdup(upd->filename);
|
||||||
|
|
||||||
|
return u;
|
||||||
|
}
|
||||||
|
|
||||||
|
UPDATE * new_update(int op, char * memtype, int filefmt, char * filename)
|
||||||
|
{
|
||||||
|
UPDATE * u;
|
||||||
|
|
||||||
|
u = (UPDATE *)malloc(sizeof(UPDATE));
|
||||||
|
if (u == NULL) {
|
||||||
|
fprintf(stderr, "%s: out of memory\n", progname);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
u->memtype = strdup(memtype);
|
||||||
|
u->filename = strdup(filename);
|
||||||
|
u->op = op;
|
||||||
|
u->format = filefmt;
|
||||||
|
|
||||||
|
return u;
|
||||||
|
}
|
||||||
|
|
||||||
|
int do_op(PROGRAMMER * pgm, struct avrpart * p, UPDATE * upd, int nowrite,
|
||||||
|
int verify)
|
||||||
|
{
|
||||||
|
struct avrpart * v;
|
||||||
|
AVRMEM * mem;
|
||||||
|
int size, vsize;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
mem = avr_locate_mem(p, upd->memtype);
|
||||||
|
if (mem == NULL) {
|
||||||
|
fprintf(stderr, "\"%s\" memory type not defined for part \"%s\"\n",
|
||||||
|
upd->memtype, p->desc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (upd->op == DEVICE_READ) {
|
||||||
|
/*
|
||||||
|
* read out the specified device memory and write it to a file
|
||||||
|
*/
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: reading %s memory:\n",
|
||||||
|
progname, mem->desc);
|
||||||
|
}
|
||||||
|
report_progress(0,1,"Reading");
|
||||||
|
rc = avr_read(pgm, p, upd->memtype, 0, 1);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: failed to read all of %s memory, rc=%d\n",
|
||||||
|
progname, mem->desc, rc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
report_progress(1,1,NULL);
|
||||||
|
size = rc;
|
||||||
|
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: writing output file \"%s\"\n",
|
||||||
|
progname,
|
||||||
|
strcmp(upd->filename, "-")==0 ? "<stdout>" : upd->filename);
|
||||||
|
}
|
||||||
|
rc = fileio(FIO_WRITE, upd->filename, upd->format, p, upd->memtype, size);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: write to file '%s' failed\n",
|
||||||
|
progname, upd->filename);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (upd->op == DEVICE_WRITE) {
|
||||||
|
/*
|
||||||
|
* write the selected device memory using data from a file; first
|
||||||
|
* read the data from the specified file
|
||||||
|
*/
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: reading input file \"%s\"\n",
|
||||||
|
progname,
|
||||||
|
strcmp(upd->filename, "-")==0 ? "<stdin>" : upd->filename);
|
||||||
|
}
|
||||||
|
rc = fileio(FIO_READ, upd->filename, upd->format, p, upd->memtype, -1);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: read from file '%s' failed\n",
|
||||||
|
progname, upd->filename);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
size = rc;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* write the buffer contents to the selected memory type
|
||||||
|
*/
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: writing %s (%d bytes):\n",
|
||||||
|
progname, mem->desc, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!nowrite) {
|
||||||
|
report_progress(0,1,"Writing");
|
||||||
|
rc = avr_write(pgm, p, upd->memtype, size, 1);
|
||||||
|
report_progress(1,1,NULL);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
/*
|
||||||
|
* test mode, don't actually write to the chip, output the buffer
|
||||||
|
* to stdout in intel hex instead
|
||||||
|
*/
|
||||||
|
rc = fileio(FIO_WRITE, "-", FMT_IHEX, p, upd->memtype, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: failed to write %s memory, rc=%d\n",
|
||||||
|
progname, mem->desc, rc);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
vsize = rc;
|
||||||
|
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: %d bytes of %s written\n", progname,
|
||||||
|
vsize, mem->desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (upd->op == DEVICE_VERIFY) {
|
||||||
|
/*
|
||||||
|
* verify that the in memory file (p->mem[AVR_M_FLASH|AVR_M_EEPROM])
|
||||||
|
* is the same as what is on the chip
|
||||||
|
*/
|
||||||
|
pgm->vfy_led(pgm, ON);
|
||||||
|
|
||||||
|
v = avr_dup_part(p);
|
||||||
|
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: verifying %s memory against %s:\n",
|
||||||
|
progname, mem->desc, upd->filename);
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: load data %s data from input file %s:\n",
|
||||||
|
progname, mem->desc, upd->filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = fileio(FIO_READ, upd->filename, upd->format, p, upd->memtype, -1);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: read from file '%s' failed\n",
|
||||||
|
progname, upd->filename);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
size = rc;
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: input file %s contains %d bytes\n",
|
||||||
|
progname, upd->filename, size);
|
||||||
|
fprintf(stderr, "%s: reading on-chip %s data:\n",
|
||||||
|
progname, mem->desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
report_progress (0,1,"Reading");
|
||||||
|
rc = avr_read(pgm, v, upd->memtype, size, 1);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: failed to read all of %s memory, rc=%d\n",
|
||||||
|
progname, mem->desc, rc);
|
||||||
|
pgm->err_led(pgm, ON);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
report_progress (1,1,NULL);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: verifying ...\n", progname);
|
||||||
|
}
|
||||||
|
rc = avr_verify(p, v, upd->memtype, size);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "%s: verification error; content mismatch\n",
|
||||||
|
progname);
|
||||||
|
pgm->err_led(pgm, ON);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (quell_progress < 2) {
|
||||||
|
fprintf(stderr, "%s: %d bytes of %s verified\n",
|
||||||
|
progname, rc, mem->desc);
|
||||||
|
}
|
||||||
|
|
||||||
|
pgm->vfy_led(pgm, OFF);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, "%s: invalid update operation (%d) requested\n",
|
||||||
|
progname, upd->op);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2000-2005 Brian S. Dean <bsd@bsdhome.com>
|
||||||
|
* Copyright (C) 2007 Joerg Wunsch
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
#ifndef update_h
|
||||||
|
#define update_h
|
||||||
|
|
||||||
|
enum {
|
||||||
|
DEVICE_READ,
|
||||||
|
DEVICE_WRITE,
|
||||||
|
DEVICE_VERIFY
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct update_t {
|
||||||
|
char * memtype;
|
||||||
|
int op;
|
||||||
|
char * filename;
|
||||||
|
int format;
|
||||||
|
} UPDATE;
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
extern UPDATE * parse_op(char * s);
|
||||||
|
extern UPDATE * dup_update(UPDATE * upd);
|
||||||
|
extern UPDATE * new_update(int op, char * memtype, int filefmt,
|
||||||
|
char * filename);
|
||||||
|
extern int do_op(PROGRAMMER * pgm, struct avrpart * p, UPDATE * upd,
|
||||||
|
int nowrite, int verify);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,483 @@
|
||||||
|
/*
|
||||||
|
* avrdude - A Downloader/Uploader for AVR device programmers
|
||||||
|
* Copyright (C) 2005,2006 Joerg Wunsch
|
||||||
|
* Copyright (C) 2006 David Moore
|
||||||
|
*
|
||||||
|
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* $Id$ */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* USB interface via libusb for avrdude.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "ac_cfg.h"
|
||||||
|
#if defined(HAVE_LIBUSB)
|
||||||
|
|
||||||
|
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
|
||||||
|
#include <usb.h>
|
||||||
|
|
||||||
|
#include "avrdude.h"
|
||||||
|
#include "serial.h"
|
||||||
|
#include "usbdevs.h"
|
||||||
|
|
||||||
|
#if defined(WIN32NATIVE)
|
||||||
|
/* someone has defined "interface" to "struct" in Cygwin */
|
||||||
|
# undef interface
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static char usbbuf[USBDEV_MAX_XFER];
|
||||||
|
static int buflen = -1, bufptr;
|
||||||
|
|
||||||
|
static int usb_interface;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The "baud" parameter is meaningless for USB devices, so we reuse it
|
||||||
|
* to pass the desired USB device ID.
|
||||||
|
*/
|
||||||
|
static int usbdev_open(char * port, long baud, union filedescriptor *fd)
|
||||||
|
{
|
||||||
|
char string[256];
|
||||||
|
char product[256];
|
||||||
|
struct usb_bus *bus;
|
||||||
|
struct usb_device *dev;
|
||||||
|
usb_dev_handle *udev;
|
||||||
|
char *serno, *cp2;
|
||||||
|
int i;
|
||||||
|
size_t x;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The syntax for usb devices is defined as:
|
||||||
|
*
|
||||||
|
* -P usb[:serialnumber]
|
||||||
|
*
|
||||||
|
* See if we've got a serial number passed here. The serial number
|
||||||
|
* might contain colons which we remove below, and we compare it
|
||||||
|
* right-to-left, so only the least significant nibbles need to be
|
||||||
|
* specified.
|
||||||
|
*/
|
||||||
|
if ((serno = strchr(port, ':')) != NULL)
|
||||||
|
{
|
||||||
|
/* first, drop all colons there if any */
|
||||||
|
cp2 = ++serno;
|
||||||
|
|
||||||
|
while ((cp2 = strchr(cp2, ':')) != NULL)
|
||||||
|
{
|
||||||
|
x = strlen(cp2) - 1;
|
||||||
|
memmove(cp2, cp2 + 1, x);
|
||||||
|
cp2[x] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen(serno) > 12)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): invalid serial number \"%s\"\n",
|
||||||
|
progname, serno);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
usb_init();
|
||||||
|
|
||||||
|
usb_find_busses();
|
||||||
|
usb_find_devices();
|
||||||
|
|
||||||
|
for (bus = usb_get_busses(); bus; bus = bus->next)
|
||||||
|
{
|
||||||
|
for (dev = bus->devices; dev; dev = dev->next)
|
||||||
|
{
|
||||||
|
udev = usb_open(dev);
|
||||||
|
if (udev)
|
||||||
|
{
|
||||||
|
if (dev->descriptor.idVendor == USB_VENDOR_ATMEL &&
|
||||||
|
dev->descriptor.idProduct == (unsigned short)baud)
|
||||||
|
{
|
||||||
|
/* yeah, we found something */
|
||||||
|
if (usb_get_string_simple(udev,
|
||||||
|
dev->descriptor.iSerialNumber,
|
||||||
|
string, sizeof(string)) < 0)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usb_open(): cannot read serial number \"%s\"\n",
|
||||||
|
progname, usb_strerror());
|
||||||
|
/*
|
||||||
|
* On some systems, libusb appears to have
|
||||||
|
* problems sending control messages. Catch the
|
||||||
|
* benign case where the user did not request a
|
||||||
|
* particular serial number, so we could
|
||||||
|
* continue anyway.
|
||||||
|
*/
|
||||||
|
if (serno != NULL)
|
||||||
|
exit(1); /* no chance */
|
||||||
|
else
|
||||||
|
strcpy(string, "[unknown]");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (usb_get_string_simple(udev,
|
||||||
|
dev->descriptor.iProduct,
|
||||||
|
product, sizeof(product)) < 0)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usb_open(): cannot read product name \"%s\"\n",
|
||||||
|
progname, usb_strerror());
|
||||||
|
strcpy(product, "[unnamed product]");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (verbose)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): Found %s, serno: %s\n",
|
||||||
|
progname, product, string);
|
||||||
|
if (serno != NULL)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* See if the serial number requested by the
|
||||||
|
* user matches what we found, matching
|
||||||
|
* right-to-left.
|
||||||
|
*/
|
||||||
|
x = strlen(string) - strlen(serno);
|
||||||
|
if (strcasecmp(string + x, serno) != 0)
|
||||||
|
{
|
||||||
|
if (verbose > 2)
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): serial number doesn't match\n",
|
||||||
|
progname);
|
||||||
|
usb_close(udev);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dev->config == NULL)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): USB device has no configuration\n",
|
||||||
|
progname);
|
||||||
|
goto trynext;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (usb_set_configuration(udev, dev->config[0].bConfigurationValue))
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): error setting configuration %d: %s\n",
|
||||||
|
progname, dev->config[0].bConfigurationValue,
|
||||||
|
usb_strerror());
|
||||||
|
goto trynext;
|
||||||
|
}
|
||||||
|
|
||||||
|
usb_interface = dev->config[0].interface[0].altsetting[0].bInterfaceNumber;
|
||||||
|
if (usb_claim_interface(udev, usb_interface))
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): error claiming interface %d: %s\n",
|
||||||
|
progname, usb_interface, usb_strerror());
|
||||||
|
goto trynext;
|
||||||
|
}
|
||||||
|
|
||||||
|
fd->usb.handle = udev;
|
||||||
|
fd->usb.ep = -1;
|
||||||
|
/* Try finding out what our read endpoint is. */
|
||||||
|
for (i = 0; i < dev->config[0].interface[0].altsetting[0].bNumEndpoints; i++)
|
||||||
|
{
|
||||||
|
int possible_ep = dev->config[0].interface[0].altsetting[0].
|
||||||
|
endpoint[i].bEndpointAddress;
|
||||||
|
|
||||||
|
if ((possible_ep & USB_ENDPOINT_DIR_MASK) != 0)
|
||||||
|
{
|
||||||
|
if (verbose > 1)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): using read endpoint 0x%02x\n",
|
||||||
|
progname, possible_ep);
|
||||||
|
}
|
||||||
|
fd->usb.ep = possible_ep;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (fd->usb.ep == -1)
|
||||||
|
{
|
||||||
|
fprintf(stderr,
|
||||||
|
"%s: usbdev_open(): cannot find a read endpoint, using 0x%02x\n",
|
||||||
|
progname, USBDEV_BULK_EP_READ);
|
||||||
|
fd->usb.ep = USBDEV_BULK_EP_READ;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
trynext:
|
||||||
|
usb_close(udev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: usbdev_open(): did not find any%s USB device \"%s\"\n",
|
||||||
|
progname, serno? " (matching)": "", port);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void usbdev_close(union filedescriptor *fd)
|
||||||
|
{
|
||||||
|
usb_dev_handle *udev = (usb_dev_handle *)fd->usb.handle;
|
||||||
|
|
||||||
|
(void)usb_release_interface(udev, usb_interface);
|
||||||
|
|
||||||
|
#if !( defined(__FreeBSD__) ) // || ( defined(__APPLE__) && defined(__MACH__) ) )
|
||||||
|
/*
|
||||||
|
* Without this reset, the AVRISP mkII seems to stall the second
|
||||||
|
* time we try to connect to it. This is not necessary on
|
||||||
|
* FreeBSD.
|
||||||
|
*/
|
||||||
|
usb_reset(udev);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
usb_close(udev);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int usbdev_send(union filedescriptor *fd, unsigned char *bp, size_t mlen)
|
||||||
|
{
|
||||||
|
usb_dev_handle *udev = (usb_dev_handle *)fd->usb.handle;
|
||||||
|
int rv;
|
||||||
|
int i = mlen;
|
||||||
|
unsigned char * p = bp;
|
||||||
|
int tx_size;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Split the frame into multiple packets. It's important to make
|
||||||
|
* sure we finish with a short packet, or else the device won't know
|
||||||
|
* the frame is finished. For example, if we need to send 64 bytes,
|
||||||
|
* we must send a packet of length 64 followed by a packet of length
|
||||||
|
* 0.
|
||||||
|
*/
|
||||||
|
do {
|
||||||
|
tx_size = (mlen < USBDEV_MAX_XFER)? mlen: USBDEV_MAX_XFER;
|
||||||
|
rv = usb_bulk_write(udev, USBDEV_BULK_EP_WRITE, (char *)bp, tx_size, 100000);
|
||||||
|
if (rv != tx_size)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: usbdev_send(): wrote %d out of %d bytes, err = %s\n",
|
||||||
|
progname, rv, tx_size, usb_strerror());
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
bp += tx_size;
|
||||||
|
mlen -= tx_size;
|
||||||
|
} while (tx_size == USBDEV_MAX_XFER);
|
||||||
|
|
||||||
|
if (verbose > 3)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Sent: ", progname);
|
||||||
|
|
||||||
|
while (i) {
|
||||||
|
unsigned char c = *p;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
|
||||||
|
p++;
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* As calls to usb_bulk_read() result in exactly one USB request, we
|
||||||
|
* have to buffer the read results ourselves, so the single-char read
|
||||||
|
* requests performed by the upper layers will be handled. In order
|
||||||
|
* to do this, we maintain a private buffer of what we've got so far,
|
||||||
|
* and transparently issue another USB read request if the buffer is
|
||||||
|
* empty and more data are requested.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
usb_fill_buf(usb_dev_handle *udev, int ep)
|
||||||
|
{
|
||||||
|
int rv;
|
||||||
|
|
||||||
|
rv = usb_bulk_read(udev, ep, usbbuf, USBDEV_MAX_XFER, 100000);
|
||||||
|
if (rv < 0)
|
||||||
|
{
|
||||||
|
if (verbose > 1)
|
||||||
|
fprintf(stderr, "%s: usb_fill_buf(): usb_bulk_read() error %s\n",
|
||||||
|
progname, usb_strerror());
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
buflen = rv;
|
||||||
|
bufptr = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usbdev_recv(union filedescriptor *fd, unsigned char *buf, size_t nbytes)
|
||||||
|
{
|
||||||
|
usb_dev_handle *udev = (usb_dev_handle *)fd->usb.handle;
|
||||||
|
int i, amnt;
|
||||||
|
unsigned char * p = buf;
|
||||||
|
|
||||||
|
for (i = 0; nbytes > 0;)
|
||||||
|
{
|
||||||
|
if (buflen <= bufptr)
|
||||||
|
{
|
||||||
|
if (usb_fill_buf(udev, fd->usb.ep) < 0)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
amnt = buflen - bufptr > nbytes? nbytes: buflen - bufptr;
|
||||||
|
memcpy(buf + i, usbbuf + bufptr, amnt);
|
||||||
|
bufptr += amnt;
|
||||||
|
nbytes -= amnt;
|
||||||
|
i += amnt;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (verbose > 4)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "%s: Recv: ", progname);
|
||||||
|
|
||||||
|
while (i) {
|
||||||
|
unsigned char c = *p;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
|
||||||
|
p++;
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This version of recv keeps reading packets until we receive a short
|
||||||
|
* packet. Then, the entire frame is assembled and returned to the
|
||||||
|
* user. The length will be unknown in advance, so we return the
|
||||||
|
* length as the return value of this function, or -1 in case of an
|
||||||
|
* error.
|
||||||
|
*
|
||||||
|
* This is used for the AVRISP mkII device.
|
||||||
|
*/
|
||||||
|
static int usbdev_recv_frame(union filedescriptor *fd, unsigned char *buf, size_t nbytes)
|
||||||
|
{
|
||||||
|
usb_dev_handle *udev = (usb_dev_handle *)fd->usb.handle;
|
||||||
|
int rv, n;
|
||||||
|
int i;
|
||||||
|
unsigned char * p = buf;
|
||||||
|
|
||||||
|
n = 0;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
rv = usb_bulk_read(udev, fd->usb.ep, usbbuf,
|
||||||
|
USBDEV_MAX_XFER, 100000);
|
||||||
|
if (rv < 0)
|
||||||
|
{
|
||||||
|
if (verbose > 1)
|
||||||
|
fprintf(stderr, "%s: usbdev_recv_frame(): usb_bulk_read(): %s\n",
|
||||||
|
progname, usb_strerror());
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rv <= nbytes)
|
||||||
|
{
|
||||||
|
memcpy (buf, usbbuf, rv);
|
||||||
|
buf += rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
n += rv;
|
||||||
|
nbytes -= rv;
|
||||||
|
}
|
||||||
|
while (rv == USBDEV_MAX_XFER);
|
||||||
|
|
||||||
|
if (nbytes < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (verbose > 3)
|
||||||
|
{
|
||||||
|
i = n;
|
||||||
|
fprintf(stderr, "%s: Recv: ", progname);
|
||||||
|
|
||||||
|
while (i) {
|
||||||
|
unsigned char c = *p;
|
||||||
|
if (isprint(c)) {
|
||||||
|
fprintf(stderr, "%c ", c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(stderr, ". ");
|
||||||
|
}
|
||||||
|
fprintf(stderr, "[%02x] ", c);
|
||||||
|
|
||||||
|
p++;
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
}
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usbdev_drain(union filedescriptor *fd, int display)
|
||||||
|
{
|
||||||
|
usb_dev_handle *udev = (usb_dev_handle *)fd->usb.handle;
|
||||||
|
int rv;
|
||||||
|
|
||||||
|
do {
|
||||||
|
rv = usb_bulk_read(udev, fd->usb.ep, usbbuf, USBDEV_MAX_XFER, 100);
|
||||||
|
if (rv > 0 && verbose >= 4)
|
||||||
|
fprintf(stderr, "%s: usbdev_drain(): flushed %d characters\n",
|
||||||
|
progname, rv);
|
||||||
|
} while (rv > 0);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Device descriptor for the JTAG ICE mkII.
|
||||||
|
*/
|
||||||
|
struct serial_device usb_serdev =
|
||||||
|
{
|
||||||
|
.open = usbdev_open,
|
||||||
|
.close = usbdev_close,
|
||||||
|
.send = usbdev_send,
|
||||||
|
.recv = usbdev_recv,
|
||||||
|
.drain = usbdev_drain,
|
||||||
|
.flags = SERDEV_FL_NONE,
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Device descriptor for the AVRISP mkII.
|
||||||
|
*/
|
||||||
|
struct serial_device usb_serdev_frame =
|
||||||
|
{
|
||||||
|
.open = usbdev_open,
|
||||||
|
.close = usbdev_close,
|
||||||
|
.send = usbdev_send,
|
||||||
|
.recv = usbdev_recv_frame,
|
||||||
|
.drain = usbdev_drain,
|
||||||
|
.flags = SERDEV_FL_NONE,
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* HAVE_LIBUSB */
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue