mirror of
https://gitea.osmocom.org/sim-card/simtrace2.git
synced 2026-03-17 21:58:33 +03:00
Compare commits
218 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
71cfc2b90d | ||
|
|
15e943ab8b | ||
|
|
b7f94dcb9f | ||
|
|
c90de6983c | ||
|
|
6f41349db9 | ||
|
|
208890ad6e | ||
|
|
964cda309d | ||
|
|
331fa5a237 | ||
|
|
6fada5604b | ||
|
|
931fe558df | ||
|
|
d401b12136 | ||
|
|
389a4040d5 | ||
|
|
5db9402a5f | ||
|
|
ac7e73a579 | ||
|
|
7233cf803a | ||
|
|
cdcdcc9f6d | ||
|
|
e876bf53e8 | ||
|
|
f7f1ea864d | ||
|
|
3feadfa910 | ||
|
|
9acff5ee5a | ||
|
|
e5efbb156c | ||
|
|
ede87e067d | ||
|
|
acb7bd9fbe | ||
|
|
4b487b836a | ||
|
|
e0265462d8 | ||
|
|
d14970f95b | ||
|
|
b1a81c130e | ||
|
|
298a5ba722 | ||
|
|
c3ef475ea5 | ||
|
|
155f57abcf | ||
|
|
bc62335768 | ||
|
|
63490361d2 | ||
|
|
6228d187da | ||
|
|
a634c0efee | ||
|
|
bb9b0dc8e8 | ||
|
|
b7e326cad3 | ||
|
|
2fdcf3b38d | ||
|
|
7e5cda5732 | ||
|
|
032fc5f844 | ||
|
|
b1f99c909c | ||
|
|
53b4e593aa | ||
|
|
dc85fbc3e1 | ||
|
|
66ffb6d493 | ||
|
|
5b5d24ebf3 | ||
|
|
faf1e88e48 | ||
|
|
64f69fc4ac | ||
|
|
6303c39a00 | ||
|
|
ad0958e9e3 | ||
|
|
4f3a0356a4 | ||
|
|
7d5d011095 | ||
|
|
1dbcf62295 | ||
|
|
e5f891a825 | ||
|
|
a6bd7178b5 | ||
|
|
ba15387b09 | ||
|
|
c171112994 | ||
|
|
29200c6223 | ||
|
|
80d9476602 | ||
|
|
e2b0f971e5 | ||
|
|
e07640c35a | ||
|
|
82b628524f | ||
|
|
cb78b55c26 | ||
|
|
77ff745cca | ||
|
|
bd9d94994d | ||
|
|
95fac3aeab | ||
|
|
c17bf77ecc | ||
|
|
c394109964 | ||
|
|
888f196595 | ||
|
|
f7f61cd10f | ||
|
|
0e95f53588 | ||
|
|
76ef811a4e | ||
|
|
fecfa2aa51 | ||
|
|
c428516efa | ||
|
|
37e7861c4d | ||
|
|
ff3d84922d | ||
|
|
9547e419eb | ||
|
|
8daba9cc9a | ||
|
|
d5f583da60 | ||
|
|
04ccb770fd | ||
|
|
910e6830b9 | ||
|
|
9cccb2bab5 | ||
|
|
8a4fba5ea2 | ||
|
|
1b39fd31ee | ||
|
|
0f4abf5eaa | ||
|
|
1836ac0761 | ||
|
|
dd36d9b010 | ||
|
|
8b8e58b00e | ||
|
|
8f70c3eb1f | ||
|
|
ebe672e926 | ||
|
|
d8ebd6ab77 | ||
|
|
738a04aefb | ||
|
|
2a44dc598d | ||
|
|
ec396bf402 | ||
|
|
697199676e | ||
|
|
8e84f8125c | ||
|
|
d1c6536154 | ||
|
|
352809992c | ||
|
|
a71a6f48cb | ||
|
|
57b60d23cf | ||
|
|
b60538888f | ||
|
|
680bdaba96 | ||
|
|
f66af0c640 | ||
|
|
5f6b8717a4 | ||
|
|
ac0843af83 | ||
|
|
31ed8029b7 | ||
|
|
3b7624c120 | ||
|
|
f000831d72 | ||
|
|
b37bda0b55 | ||
|
|
c6b968067d | ||
|
|
a95bb1e85d | ||
|
|
da5578bd85 | ||
|
|
d975411511 | ||
|
|
a2fccba96c | ||
|
|
411428eb5e | ||
|
|
f82f0f6eff | ||
|
|
55f0612c8e | ||
|
|
33d1eb73fd | ||
|
|
93717e43b3 | ||
|
|
9a12d68c74 | ||
|
|
efbcf38afb | ||
|
|
9918c2840e | ||
|
|
d24e9bde26 | ||
|
|
2bdaa73aff | ||
|
|
fcf2743552 | ||
|
|
d44cb80bc5 | ||
|
|
9e29a3eb37 | ||
|
|
bf6b1b1a3e | ||
|
|
4cbdc7cf18 | ||
|
|
86f48fc962 | ||
|
|
1200a5228f | ||
|
|
fe763b7698 | ||
|
|
c9bd715289 | ||
|
|
35e8bdf879 | ||
|
|
709a431ab9 | ||
|
|
30f90a78fc | ||
|
|
7406337a7f | ||
|
|
6e3f1121e6 | ||
|
|
de97fd25bd | ||
|
|
012940f48e | ||
|
|
e2f84f6a8b | ||
|
|
4d6a4b949c | ||
|
|
a1012b170a | ||
|
|
42bd026416 | ||
|
|
638cec820f | ||
|
|
00ec89d73f | ||
|
|
cf59919494 | ||
|
|
11914d9658 | ||
|
|
3113e3d2e5 | ||
|
|
9def7631ba | ||
|
|
ca9e4bf4ba | ||
|
|
8210ec3f62 | ||
|
|
8fa6ff5979 | ||
|
|
866d20b10e | ||
|
|
4fe99fad59 | ||
|
|
7be52ec1ce | ||
|
|
a2b367633c | ||
|
|
927ffb46eb | ||
|
|
d7a6de57d2 | ||
|
|
353351ddf4 | ||
|
|
4091d78c4b | ||
|
|
ee62a9da56 | ||
|
|
45ad62d8d4 | ||
|
|
2c4e2af21f | ||
|
|
7b73462442 | ||
|
|
36abece0b1 | ||
|
|
f9997e9d26 | ||
|
|
05cc7f6531 | ||
|
|
46893451de | ||
|
|
cb6e20596e | ||
|
|
bb2eb19fb1 | ||
|
|
9d90d284ed | ||
|
|
ebe8b2069c | ||
|
|
105c1dce4f | ||
|
|
1cfc2614dd | ||
|
|
a9bca48914 | ||
|
|
eac1bec428 | ||
|
|
51c128bc35 | ||
|
|
869dbfafe4 | ||
|
|
80303c135b | ||
|
|
d86cab0080 | ||
|
|
b73f0a00bc | ||
|
|
f5869d4a59 | ||
|
|
4136c242a8 | ||
|
|
318309f30f | ||
|
|
0828b914e7 | ||
|
|
76be7c806e | ||
|
|
a38a126361 | ||
|
|
6822716428 | ||
|
|
a93f7273b3 | ||
|
|
432ba5140e | ||
|
|
849d20e29e | ||
|
|
e48d6f2321 | ||
|
|
af616ec4e2 | ||
|
|
6051e126da | ||
|
|
7f62c24532 | ||
|
|
75cf93e04d | ||
|
|
2afd57f00a | ||
|
|
0633b25974 | ||
|
|
c1e2254854 | ||
|
|
27f5fc681c | ||
|
|
7b250bfc8d | ||
|
|
ed75c62acf | ||
|
|
5c081911a9 | ||
|
|
5e6e8dcbde | ||
|
|
c35998e20d | ||
|
|
ba2ad563cc | ||
|
|
fc87c24326 | ||
|
|
f231541601 | ||
|
|
119624f46f | ||
|
|
7b36306113 | ||
|
|
eb50c9f914 | ||
|
|
965d5c918a | ||
|
|
514c6d1da0 | ||
|
|
e7f2f9a5e0 | ||
|
|
4e837d45db | ||
|
|
b52b886186 | ||
|
|
c47fc5febf | ||
|
|
02d0ec6e08 | ||
|
|
3b646955b9 |
32
Makefile
Normal file
32
Makefile
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
all: fw utils
|
||||||
|
|
||||||
|
define RULES
|
||||||
|
fw-$(1)-$(2):
|
||||||
|
make -C firmware BOARD=$(1) APP=$(2)
|
||||||
|
fw-$(1)-$(2)-clean:
|
||||||
|
make -C firmware BOARD=$(1) APP=$(2) clean
|
||||||
|
endef
|
||||||
|
|
||||||
|
$(eval $(call RULES,simtrace,dfu))
|
||||||
|
$(eval $(call RULES,simtrace,trace))
|
||||||
|
$(eval $(call RULES,simtrace,cardem))
|
||||||
|
$(eval $(call RULES,qmod,dfu))
|
||||||
|
$(eval $(call RULES,qmod,cardem))
|
||||||
|
|
||||||
|
fw-clean: fw-simtrace-dfu-clean fw-simtrace-trace-clean fw-simtrace-cardem-clean fw-qmod-dfu-clean fw-qmod-cardem-clean
|
||||||
|
fw: fw-simtrace-dfu fw-simtrace-trace fw-simtrace-cardem fw-qmod-dfu fw-qmod-cardem
|
||||||
|
|
||||||
|
utils:
|
||||||
|
(cd host && \
|
||||||
|
autoreconf -fi && \
|
||||||
|
./configure --prefix=/usr --disable-werror && \
|
||||||
|
make)
|
||||||
|
|
||||||
|
clean: fw-clean
|
||||||
|
if [ -e host/Makefile ]; then \
|
||||||
|
make -C host clean; \
|
||||||
|
fi
|
||||||
|
|
||||||
|
install:
|
||||||
|
make -C firmware install
|
||||||
|
make -C host install
|
||||||
39
README.md
Normal file
39
README.md
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
SIMtrace v2.0
|
||||||
|
=============
|
||||||
|
|
||||||
|
This is the repository for the next-generation SIMtrace devices,
|
||||||
|
providing abilities to trace the communication between (U)SIM card and
|
||||||
|
phone, remote (U)SIM card forward, (U)SIM man-in-the-middle, and more.
|
||||||
|
|
||||||
|
This is under heavy development, and right now it is not surprising if
|
||||||
|
things still break on a daily basis.
|
||||||
|
|
||||||
|
NOTE: Nothing in this repository applies to the SIMtrace v1.x hardware
|
||||||
|
or its associated firmware. SIMtrace v1.x is based on a different CPU /
|
||||||
|
microcontroller architecture and uses a completely different software
|
||||||
|
stack and host software.
|
||||||
|
|
||||||
|
Supported Hardware
|
||||||
|
------------------
|
||||||
|
|
||||||
|
At this point, the primary development target is still the OWHW + sysmoQMOD
|
||||||
|
device, but we expect to add support for a SAM3 based SIMtrace hardware
|
||||||
|
board soon.
|
||||||
|
|
||||||
|
The goal is to support the following devices:
|
||||||
|
|
||||||
|
* Osmocom SIMtrace 1.x with SAM3 controller
|
||||||
|
** this is open hardware and schematics / PCB design is published
|
||||||
|
* sysmocom sysmoQMOD (with 4 Modems, 4 SIM slots and 2 SAM3)
|
||||||
|
** this is a proprietary device, publicly available from sysmocom
|
||||||
|
* sysmocom OWHW (with 2 Modems and 1 SAM3 onboard)
|
||||||
|
** this is not publicly available hardware, but still supported
|
||||||
|
|
||||||
|
This Repository
|
||||||
|
---------------
|
||||||
|
|
||||||
|
This repository contains several directory
|
||||||
|
|
||||||
|
* firmware - the firmware to run on the actual devices
|
||||||
|
* hardware - some information related to the hardware
|
||||||
|
* host - Programs to use on the USB host to interface with the hardware
|
||||||
90
contrib/jenkins.sh
Executable file
90
contrib/jenkins.sh
Executable file
@@ -0,0 +1,90 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
TOPDIR=`pwd`
|
||||||
|
|
||||||
|
if ! [ -x "$(command -v osmo-build-dep.sh)" ]; then
|
||||||
|
echo "Error: We need to have scripts/osmo-deps.sh from http://git.osmocom.org/osmo-ci/ in PATH !"
|
||||||
|
exit 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
publish="$1"
|
||||||
|
base="$PWD"
|
||||||
|
deps="$base/deps"
|
||||||
|
inst="$deps/install"
|
||||||
|
export deps inst
|
||||||
|
|
||||||
|
osmo-clean-workspace.sh
|
||||||
|
|
||||||
|
mkdir "$deps" || true
|
||||||
|
|
||||||
|
osmo-build-dep.sh libosmocore "" '--disable-doxygen --enable-gnutls'
|
||||||
|
|
||||||
|
export PKG_CONFIG_PATH="$inst/lib/pkgconfig:$PKG_CONFIG_PATH"
|
||||||
|
export LD_LIBRARY_PATH="$inst/lib"
|
||||||
|
|
||||||
|
BUILDS=""
|
||||||
|
BUILDS+="simtrace/dfu simtrace/cardem simtrace/trace " # simtrace/triple_play
|
||||||
|
BUILDS+="qmod/dfu qmod/cardem "
|
||||||
|
BUILDS+="owhw/dfu owhw/cardem "
|
||||||
|
|
||||||
|
cd $TOPDIR/firmware
|
||||||
|
for build in $BUILDS; do
|
||||||
|
board=`echo $build | cut -d "/" -f 1`
|
||||||
|
app=`echo $build | cut -d "/" -f 2`
|
||||||
|
echo
|
||||||
|
echo "=============== $board / $app START =============="
|
||||||
|
make BOARD="$board" APP="$app"
|
||||||
|
echo "=============== $board / $app RES:$? =============="
|
||||||
|
done
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "=============== FIRMWARE TESTS ==========="
|
||||||
|
cd $TOPDIR/firmware/test
|
||||||
|
make clean
|
||||||
|
make
|
||||||
|
./card_emu_test
|
||||||
|
make clean
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "=============== HOST START =============="
|
||||||
|
cd $TOPDIR/host
|
||||||
|
autoreconf --install --force
|
||||||
|
./configure --enable-sanitize --enable-werror
|
||||||
|
$MAKE $PARALLEL_MAKE
|
||||||
|
#$MAKE distcheck || cat-testlogs.sh
|
||||||
|
make dist
|
||||||
|
|
||||||
|
#if [ "$WITH_MANUALS" = "1" ] && [ "$PUBLISH" = "1" ]; then
|
||||||
|
# make -C "$base/doc/manuals" publish
|
||||||
|
#fi
|
||||||
|
|
||||||
|
if [ "x$publish" = "x--publish" ]; then
|
||||||
|
echo
|
||||||
|
echo "=============== UPLOAD BUILD =============="
|
||||||
|
|
||||||
|
cat > "/build/known_hosts" <<EOF
|
||||||
|
[rita.osmocom.org]:48 ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQDDgQ9HntlpWNmh953a2Gc8NysKE4orOatVT1wQkyzhARnfYUerRuwyNr1GqMyBKdSI9amYVBXJIOUFcpV81niA7zQRUs66bpIMkE9/rHxBd81SkorEPOIS84W4vm3SZtuNqa+fADcqe88Hcb0ZdTzjKILuwi19gzrQyME2knHY71EOETe9Yow5RD2hTIpB5ecNxI0LUKDq+Ii8HfBvndPBIr0BWYDugckQ3Bocf+yn/tn2/GZieFEyFpBGF/MnLbAAfUKIdeyFRX7ufaiWWz5yKAfEhtziqdAGZaXNaLG6gkpy3EixOAy6ZXuTAk3b3Y0FUmDjhOHllbPmTOcKMry9
|
||||||
|
[rita.osmocom.org]:48 ecdsa-sha2-nistp256 AAAAE2VjZHNhLXNoYTItbmlzdHAyNTYAAAAIbmlzdHAyNTYAAABBBPdWn1kEousXuKsZ+qJEZTt/NSeASxCrUfNDW3LWtH+d8Ust7ZuKp/vuyG+5pe5pwpPOgFu7TjN+0lVjYJVXH54=
|
||||||
|
[rita.osmocom.org]:48 ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIK8iivY70EiR5NiGChV39gRLjNpC8lvu1ZdHtdMw2zuX
|
||||||
|
EOF
|
||||||
|
SSH_COMMAND="ssh -o 'UserKnownHostsFile=/build/known_hosts' -p 48"
|
||||||
|
rsync --archive --verbose --compress --delete --rsh "$SSH_COMMAND" $TOPDIR/firmware/bin/*-latest.{bin,elf} binaries@rita.osmocom.org:web-files/simtrace2/firmware/latest/
|
||||||
|
rsync --archive --verbose --compress --rsh "$SSH_COMMAND" --exclude $TOPDIR/firmware/bin/*-latest.{bin,elf} $TOPDIR/firmware/bin/*-*-*-*.{bin,elf} binaries@rita.osmocom.org:web-files/simtrace2/firmware/all/
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "=============== HOST CLEAN =============="
|
||||||
|
$MAKE maintainer-clean
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "=============== FIRMWARE CLEAN =============="
|
||||||
|
cd $TOPDIR/firmware/
|
||||||
|
for build in $BUILDS; do
|
||||||
|
board=`echo $build | cut -d "/" -f 1`
|
||||||
|
app=`echo $build | cut -d "/" -f 2`
|
||||||
|
make BOARD="$board" APP="$app" clean
|
||||||
|
done
|
||||||
|
|
||||||
|
osmo-clean-workspace.sh
|
||||||
17
debian/changelog
vendored
Normal file
17
debian/changelog
vendored
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
simtrace2 (0.5.2) UNRELEASED; urgency=medium
|
||||||
|
|
||||||
|
* adapt to host tools in autotools
|
||||||
|
|
||||||
|
-- Harald Welte <lafore@gnumonks.org> Thu, 28 Nov 2019 00:44:57 +0100
|
||||||
|
|
||||||
|
simtrace2 (0.5.1) unstable; urgency=medium
|
||||||
|
|
||||||
|
* Backwards-compatibility with older (released, non-master) libosmocore
|
||||||
|
|
||||||
|
-- Harald Welte <lafore@gnumonks.org> Sun, 26 Aug 2018 11:50:36 +0200
|
||||||
|
|
||||||
|
simtrace2 (0.5) unstable; urgency=medium
|
||||||
|
|
||||||
|
* Initial debian package release.
|
||||||
|
|
||||||
|
-- Harald Welte <lafore@gnumonks.org> Sun, 26 Aug 2018 10:37:19 +0200
|
||||||
1
debian/compat
vendored
Normal file
1
debian/compat
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
9
|
||||||
65
debian/control
vendored
Normal file
65
debian/control
vendored
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
Source: simtrace2
|
||||||
|
Maintainer: Harald Welte <laforge@gnumonks.org>
|
||||||
|
Section: devel
|
||||||
|
Priority: optional
|
||||||
|
Build-Depends: debhelper (>= 9),
|
||||||
|
autotools-dev,
|
||||||
|
autoconf,
|
||||||
|
automake,
|
||||||
|
libtool,
|
||||||
|
pkg-config,
|
||||||
|
git,
|
||||||
|
dh-autoreconf,
|
||||||
|
libosmocore-dev,
|
||||||
|
libpcsclite-dev,
|
||||||
|
libnewlib-arm-none-eabi,
|
||||||
|
libusb-1.0-0-dev,
|
||||||
|
gcc-arm-none-eabi
|
||||||
|
Standards-Version: 3.9.8
|
||||||
|
Vcs-Git: git://git.osmocom.org/simtrace2.git
|
||||||
|
Vcs-Browser: http://git.osmocom.org/simtrace2/
|
||||||
|
Homepage: http://osmocom.org/projects/simtrace2/wiki
|
||||||
|
|
||||||
|
Package: simtrace2-firmware
|
||||||
|
Section: devel
|
||||||
|
Architecture: all
|
||||||
|
Recommends: dfu-util
|
||||||
|
Description: Firmware for SAM3 based SIMtrace2 USB Devices.
|
||||||
|
Open Source firmware for the Cortex-M3 microcontroller in the
|
||||||
|
"Osmocom SIMtrace2" USB-attached peripheral device. Will only work in
|
||||||
|
SAM3S-based SIMtrace2, not in its SAM7S-based predecessor SIMtrace!
|
||||||
|
|
||||||
|
Package: simtrace2-utils
|
||||||
|
Section: devel
|
||||||
|
Architecture: any
|
||||||
|
Multi-Arch: same
|
||||||
|
Depends: ${shlibs:Depends}, ${misc:Depends}, libosmo-simtrace2-0
|
||||||
|
Recommends: simtrace2-firmware
|
||||||
|
Description: Host utilities to communicate with SIMtrace2 USB Devices.
|
||||||
|
|
||||||
|
Package: libosmo-simtrace2-0
|
||||||
|
Section: libs
|
||||||
|
Architecture: any
|
||||||
|
Multi-Arch: same
|
||||||
|
Depends: ${shlibs:Depends}, ${misc:Depends}
|
||||||
|
Description: Osmocom SIMtrace2 library
|
||||||
|
This library contains core "driver" functionality to interface with the
|
||||||
|
Osmocom SIMtrace2 (and compatible) USB device firmware. It enables
|
||||||
|
applications to implement SIM card / smart card tracing as well as
|
||||||
|
SIM / smart card emulation functions.
|
||||||
|
|
||||||
|
Package: libosmo-simtrace2-dev
|
||||||
|
Section: libdevel
|
||||||
|
Architecture: any
|
||||||
|
Multi-Arch: same
|
||||||
|
Depends: libosmo-simtrace2-0, ${misc:Depends}
|
||||||
|
Description: Development headers for Osmocom SIMtrace2 library
|
||||||
|
This library contains core "driver" functionality to interface with the
|
||||||
|
Osmocom SIMtrace2 (and compatible) USB device firmware. It enables
|
||||||
|
applications to implement SIM card / smart card tracing as well as
|
||||||
|
SIM / smart card emulation functions.
|
||||||
|
.
|
||||||
|
The header files provided by this package may be used to develop
|
||||||
|
with any of the libosmocore libraries.
|
||||||
|
.
|
||||||
|
Also static libraries are installed with this package.
|
||||||
1
debian/libosmo-simtrace2-0.install
vendored
Normal file
1
debian/libosmo-simtrace2-0.install
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
usr/lib/libosmo-simtrace2*.so.*
|
||||||
5
debian/libosmo-simtrace2-dev.install
vendored
Normal file
5
debian/libosmo-simtrace2-dev.install
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
usr/include/*
|
||||||
|
usr/lib/lib*.a
|
||||||
|
usr/lib/lib*.so
|
||||||
|
usr/lib/lib*.la
|
||||||
|
usr/lib/pkgconfig/*
|
||||||
19
debian/rules
vendored
Executable file
19
debian/rules
vendored
Executable file
@@ -0,0 +1,19 @@
|
|||||||
|
#!/usr/bin/make -f
|
||||||
|
|
||||||
|
# Uncomment this to turn on verbose mode.
|
||||||
|
#export DH_VERBOSE=1
|
||||||
|
|
||||||
|
DEBIAN := $(shell dpkg-parsechangelog | grep ^Version: | cut -d' ' -f2)
|
||||||
|
DEBVERS := $(shell echo '$(DEBIAN)' | cut -d- -f1)
|
||||||
|
VERSION := $(shell echo '$(DEBVERS)' | sed -e 's/[+-].*//' -e 's/~//g')
|
||||||
|
|
||||||
|
export DEB_BUILD_MAINT_OPTIONS = hardening=+all
|
||||||
|
|
||||||
|
export DEB_LDFLAGS_MAINT_STRIP = -Wl,-Bsymbolic-functions
|
||||||
|
|
||||||
|
|
||||||
|
%:
|
||||||
|
dh $@
|
||||||
|
|
||||||
|
override_dh_autoreconf:
|
||||||
|
cd host && dh_autoreconf
|
||||||
1
debian/simtrace2-firmware.install
vendored
Normal file
1
debian/simtrace2-firmware.install
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
usr/share/simtrace2/*.bin
|
||||||
1
debian/simtrace2-utils.install
vendored
Normal file
1
debian/simtrace2-utils.install
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
usr/bin/simtrace2-*
|
||||||
@@ -28,6 +28,7 @@
|
|||||||
|
|
||||||
# Makefile for compiling the Getting Started with SAM3S Microcontrollers project
|
# Makefile for compiling the Getting Started with SAM3S Microcontrollers project
|
||||||
|
|
||||||
|
GIT_VERSION=$(shell $(TOP)/git-version-gen $(TOP)/.tarvers)
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
# User-modifiable options
|
# User-modifiable options
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
@@ -43,7 +44,7 @@ MEMORIES ?= flash dfu
|
|||||||
# Output file basename
|
# Output file basename
|
||||||
APP ?= dfu
|
APP ?= dfu
|
||||||
|
|
||||||
# Output directories
|
# Output directories and filename
|
||||||
OUTPUT = $(BOARD)-$(APP)
|
OUTPUT = $(BOARD)-$(APP)
|
||||||
BIN = bin
|
BIN = bin
|
||||||
OBJ = obj/$(BOARD)
|
OBJ = obj/$(BOARD)
|
||||||
@@ -73,7 +74,6 @@ GDB = $(CROSS_COMPILE)gdb
|
|||||||
NM = $(CROSS_COMPILE)nm
|
NM = $(CROSS_COMPILE)nm
|
||||||
|
|
||||||
TOP=..
|
TOP=..
|
||||||
GIT_VERSION=$(shell $(TOP)/git-version-gen $(TOP)/.tarvers)
|
|
||||||
|
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
# Files
|
# Files
|
||||||
@@ -120,15 +120,15 @@ C_OBJECTS = $(C_FILES:%.c=%.o)
|
|||||||
# TRACE_LEVEL_NO_TRACE 0
|
# TRACE_LEVEL_NO_TRACE 0
|
||||||
TRACE_LEVEL ?= 4
|
TRACE_LEVEL ?= 4
|
||||||
|
|
||||||
DEBUG_PHONE_SNIFF?=0
|
# allow asserting the peer SAM3S ERASE signal to completely erase the flash
|
||||||
|
# only applicable for qmod board
|
||||||
|
ALLOW_PEER_ERASE?=0
|
||||||
|
|
||||||
#CFLAGS+=-DUSB_NO_DEBUG=1
|
#CFLAGS+=-DUSB_NO_DEBUG=1
|
||||||
|
|
||||||
# Optimization level, put in comment for debugging
|
# Optimization level, put in comment for debugging
|
||||||
OPTIMIZATION ?= -Os
|
OPTIMIZATION ?= -Os
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Flags
|
# Flags
|
||||||
INCLUDES_USB = -I$(AT91LIB)/usb/include -I$(AT91LIB)
|
INCLUDES_USB = -I$(AT91LIB)/usb/include -I$(AT91LIB)
|
||||||
|
|
||||||
@@ -141,13 +141,12 @@ INCLUDES += -Ilibosmocore/include
|
|||||||
INCLUDES += -Isrc_simtrace -Iinclude
|
INCLUDES += -Isrc_simtrace -Iinclude
|
||||||
INCLUDES += -Iapps/$(APP)
|
INCLUDES += -Iapps/$(APP)
|
||||||
|
|
||||||
CFLAGS += -Wall -Wchar-subscripts -Wcomment -Wimplicit-int #-Wformat=2
|
CFLAGS += -Wall -Wchar-subscripts -Wcomment -Wimplicit-int -Wformat=2
|
||||||
CFLAGS += -Werror-implicit-function-declaration -Wmain -Wparentheses
|
CFLAGS += -Werror-implicit-function-declaration -Wmain -Wparentheses
|
||||||
CFLAGS += -Wsequence-point -Wreturn-type -Wswitch -Wtrigraphs #-Wunused
|
CFLAGS += -Wsequence-point -Wreturn-type -Wswitch -Wtrigraphs #-Wunused
|
||||||
CFLAGS += -Wuninitialized -Wunknown-pragmas -Wfloat-equal -Wundef
|
CFLAGS += -Wuninitialized -Wunknown-pragmas -Wfloat-equal #-Wundef
|
||||||
CFLAGS += -Wshadow -Wpointer-arith -Wbad-function-cast -Wwrite-strings
|
CFLAGS += -Wshadow -Wpointer-arith -Wbad-function-cast -Wwrite-strings
|
||||||
CFLAGS += -Wsign-compare -Waggregate-return
|
CFLAGS += -Waggregate-return #-Wsign-compare
|
||||||
CFLAGS += -Wformat=0
|
|
||||||
CFLAGS += -Wmissing-format-attribute -Wno-deprecated-declarations
|
CFLAGS += -Wmissing-format-attribute -Wno-deprecated-declarations
|
||||||
CFLAGS += #-Wpacked
|
CFLAGS += #-Wpacked
|
||||||
CFLAGS += -Wredundant-decls -Wnested-externs #-Winline -Wlong-long
|
CFLAGS += -Wredundant-decls -Wnested-externs #-Winline -Wlong-long
|
||||||
@@ -164,7 +163,7 @@ CFLAGS += -Wno-suggest-attribute=noreturn
|
|||||||
#CFLAGS += -Wa,-a,-ad
|
#CFLAGS += -Wa,-a,-ad
|
||||||
CFLAGS += -D__ARM
|
CFLAGS += -D__ARM
|
||||||
CFLAGS += --param max-inline-insns-single=500 -mcpu=cortex-m3 -mthumb # -mfix-cortex-m3-ldrd
|
CFLAGS += --param max-inline-insns-single=500 -mcpu=cortex-m3 -mthumb # -mfix-cortex-m3-ldrd
|
||||||
CFLAGS += -ffunction-sections -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP) -DTRACE_LEVEL=$(TRACE_LEVEL) -DDEBUG_PHONE_SNIFF=$(DEBUG_PHONE_SNIFF)
|
CFLAGS += -ffunction-sections -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP) -DTRACE_LEVEL=$(TRACE_LEVEL) -DALLOW_PEER_ERASE=$(ALLOW_PEER_ERASE)
|
||||||
CFLAGS += -DGIT_VERSION=\"$(GIT_VERSION)\"
|
CFLAGS += -DGIT_VERSION=\"$(GIT_VERSION)\"
|
||||||
CFLAGS += -DBOARD=\"$(BOARD)\" -DBOARD_$(BOARD)
|
CFLAGS += -DBOARD=\"$(BOARD)\" -DBOARD_$(BOARD)
|
||||||
CFLAGS += -DAPPLICATION=\"$(APP)\" -DAPPLICATION_$(APP)
|
CFLAGS += -DAPPLICATION=\"$(APP)\" -DAPPLICATION_$(APP)
|
||||||
@@ -172,8 +171,7 @@ ASFLAGS = -mcpu=cortex-m3 -mthumb -Wall -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP)
|
|||||||
LDFLAGS = -mcpu=cortex-m3 -mthumb -Wl,--cref -Wl,--check-sections -Wl,--gc-sections -Wl,--entry=ResetException -Wl,--unresolved-symbols=report-all -Wl,--warn-common -Wl,--warn-section-align -Wl,--warn-unresolved-symbols $(LIB)
|
LDFLAGS = -mcpu=cortex-m3 -mthumb -Wl,--cref -Wl,--check-sections -Wl,--gc-sections -Wl,--entry=ResetException -Wl,--unresolved-symbols=report-all -Wl,--warn-common -Wl,--warn-section-align -Wl,--warn-unresolved-symbols $(LIB)
|
||||||
#LD_OPTIONAL=-Wl,--print-gc-sections -Wl,--stats
|
#LD_OPTIONAL=-Wl,--print-gc-sections -Wl,--stats
|
||||||
|
|
||||||
|
# Append BIN directories to output filename
|
||||||
# Append OBJ and BIN directories to output filename
|
|
||||||
OUTPUT := $(BIN)/$(OUTPUT)
|
OUTPUT := $(BIN)/$(OUTPUT)
|
||||||
|
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
@@ -192,7 +190,7 @@ $(OUTPUT)-combined.bin: $(BIN)/$(BOARD)-dfu-flash-padded.bin $(OUTPUT)-dfu.bin
|
|||||||
cat $^ > $@
|
cat $^ > $@
|
||||||
|
|
||||||
$(BIN) $(OBJ):
|
$(BIN) $(OBJ):
|
||||||
mkdir $@
|
mkdir -p $@
|
||||||
|
|
||||||
usbstring/usbstring: usbstring/usbstring.c
|
usbstring/usbstring: usbstring/usbstring.c
|
||||||
gcc $^ -o $@
|
gcc $^ -o $@
|
||||||
@@ -206,8 +204,12 @@ ASM_OBJECTS_$(1) = $(addprefix $(OBJ)/$(1)_, $(ASM_OBJECTS))
|
|||||||
|
|
||||||
$(1): $$(ASM_OBJECTS_$(1)) $$(C_OBJECTS_$(1))
|
$(1): $$(ASM_OBJECTS_$(1)) $$(C_OBJECTS_$(1))
|
||||||
@$(CC) $(LDFLAGS) $(LD_OPTIONAL) -T"libboard/common/resources/$(CHIP)/$$@.ld" -Wl,-Map,$(OUTPUT)-$$@.map -o $(OUTPUT)-$$@.elf $$^ $(LIBS)
|
@$(CC) $(LDFLAGS) $(LD_OPTIONAL) -T"libboard/common/resources/$(CHIP)/$$@.ld" -Wl,-Map,$(OUTPUT)-$$@.map -o $(OUTPUT)-$$@.elf $$^ $(LIBS)
|
||||||
|
cp $(OUTPUT)-$$@.elf $(OUTPUT)-$$@-$(GIT_VERSION).elf
|
||||||
|
cp $(OUTPUT)-$$@.elf $(OUTPUT)-$$@-latest.elf
|
||||||
@$(NM) $(OUTPUT)-$$@.elf >$(OUTPUT)-$$@.elf.txt
|
@$(NM) $(OUTPUT)-$$@.elf >$(OUTPUT)-$$@.elf.txt
|
||||||
@$(OBJCOPY) -O binary $(OUTPUT)-$$@.elf $(OUTPUT)-$$@.bin
|
@$(OBJCOPY) -O binary $(OUTPUT)-$$@.elf $(OUTPUT)-$$@.bin
|
||||||
|
cp $(OUTPUT)-$$@.bin $(OUTPUT)-$$@-$(GIT_VERSION).bin
|
||||||
|
cp $(OUTPUT)-$$@.bin $(OUTPUT)-$$@-latest.bin
|
||||||
@$(SIZE) $$^ $(OUTPUT)-$$@.elf
|
@$(SIZE) $$^ $(OUTPUT)-$$@.elf
|
||||||
|
|
||||||
$$(C_OBJECTS_$(1)): $(OBJ)/$(1)_%.o: %.c Makefile $(OBJ) $(BIN)
|
$$(C_OBJECTS_$(1)): $(OBJ)/$(1)_%.o: %.c Makefile $(OBJ) $(BIN)
|
||||||
@@ -234,3 +236,7 @@ log:
|
|||||||
|
|
||||||
clean:
|
clean:
|
||||||
-rm -fR $(OBJ)/*.o $(BIN)/*.bin $(BIN)/*.elf $(BIN)/*.elf.txt $(BIN)/*.map $(BIN)/*.lst `find . -name \*.p`
|
-rm -fR $(OBJ)/*.o $(BIN)/*.bin $(BIN)/*.elf $(BIN)/*.elf.txt $(BIN)/*.map $(BIN)/*.lst `find . -name \*.p`
|
||||||
|
|
||||||
|
install:
|
||||||
|
mkdir -p $(DESTDIR)/usr/share/simtrace2
|
||||||
|
cp $(BIN)/*.bin $(BIN)/*.elf $(BIN)/*.elf.txt $(BIN)/*.map $(DESTDIR)/usr/share/simtrace2
|
||||||
|
|||||||
@@ -1,70 +1,87 @@
|
|||||||
|
This is the source code for SIMtrace 2 firmwares.
|
||||||
|
|
||||||
== BOARDS
|
= Hardware
|
||||||
|
|
||||||
A board defines a given circuit board, i.e. SIMtrace, OWHW, QMOD
|
== Micro-Controller
|
||||||
|
|
||||||
It defines the given hardware model for which the program is to be
|
The firmware is for Microchip (formerly Atmel) ATSAM3S4B micro-controllers (MCU).
|
||||||
compiled.
|
Product page: https://www.microchip.com/wwwproducts/en/ATSAM3S4B
|
||||||
|
Datasheet: http://ww1.microchip.com/downloads/en/DeviceDoc/Atmel-6500-32-bit-Cortex-M3-Microcontroller-SAM3S4-SAM3S2-SAM3S1_Datasheet.pdf
|
||||||
|
|
||||||
|
Note: The SAM3S is now not recommended for new designs.
|
||||||
|
It can be replaced by the pin-compatible SAM4S.
|
||||||
|
The MCU can be specified using the environment variable `CHIP` (set to `sam3s4` per default) for future MCU support.
|
||||||
|
|
||||||
|
== Boards
|
||||||
|
|
||||||
|
The SIMtrace 2 firmware supports multiple boards.
|
||||||
|
A board defines a given circuit board.
|
||||||
|
While compiling the firmware, set the target board using the `BOARD` environment variable (set to `qmod` per default).
|
||||||
|
The supported boards correspond to sub-folders under `libboard`.
|
||||||
|
|
||||||
Current boards supported are:
|
Current boards supported are:
|
||||||
* simtrace: The good old Osmocom SIMtrace PCB with SAM3 instead of
|
|
||||||
SAM7, open hardware.
|
|
||||||
* qmod: A sysmocom-proprietary quad mPCIe carrier board, publicly available
|
|
||||||
* owhw: An undisclosed sysmocom-internal board, not publicly available
|
|
||||||
|
|
||||||
== APPLICATIONS
|
* `simtrace`: The good old Osmocom SIMtrace PCB with SAM3 instead of SAM7, open hardware.
|
||||||
|
* `qmod`: A sysmocom-proprietary quad mPCIe carrier board, publicly available
|
||||||
|
* `owhw`: An undisclosed sysmocom-internal board, not publicly available
|
||||||
|
* `octsimtest`: A sysmocom-proprietary production testing board, not publicly available
|
||||||
|
|
||||||
An application is a specific piece of software with given
|
= Firmware
|
||||||
functionality.
|
|
||||||
|
|
||||||
== ENVIRONMENTS
|
== Library
|
||||||
|
|
||||||
|
The firmware uses the manufacturer provided Software Package (SoftPack) micro-controller library.
|
||||||
|
The original library is available at https://www.microchip.com/design-centers/32-bit/softpacks/legacy-softpacks .
|
||||||
|
Version 2.1 from 2001 is used: http://ww1.microchip.com/downloads/en/DeviceDoc/SAM3S_softpack_2.1_for_CodeSourcery_2010q1.zip
|
||||||
|
The SIMtrace 2 project uses the `libboard_sam3s-ek`, `libchip_sam3s`, and `usb` sub-libraries, saved in `atmel_softpack_libraries` (with local modifications).
|
||||||
|
|
||||||
|
Note: SoftPack is the legacy micro-controller library.
|
||||||
|
This library is now replaced by the Advanced Software Framework (ASF): https://www.microchip.com/avr-support/advanced-software-framework-(asf) .
|
||||||
|
The SAM3S ASF documentation is available at http://asf.atmel.com/docs/latest/sam3s/html/index.html .
|
||||||
|
|
||||||
|
== Applications
|
||||||
|
|
||||||
|
An application is a specific piece of software with given functionality.
|
||||||
|
While compiling the firmware, set the target application using the `APP` environment variable (set to `dfu` per default).
|
||||||
|
The supported applications correspond to sub-folder under `apps`.
|
||||||
|
|
||||||
|
Current applications supported are:
|
||||||
|
|
||||||
|
* `dfu`: The USB DFU bootloader to flash further main appliction firmwares.
|
||||||
|
* `ccid`: To use SIMtrace 2 as USB CCID smartcard reader.
|
||||||
|
* `cardem`: To provide remote SIM operation capabilities.
|
||||||
|
* `trace`: To monitor the communication between a SIM card and a phone (corresponds to the functionality provide by the first SIMtrace)
|
||||||
|
* `triple_play`: To support the three previous functionalities, using USB configurations.
|
||||||
|
* `gpio_test`: internal test code
|
||||||
|
|
||||||
|
== Memories
|
||||||
|
|
||||||
|
Firmwares can be run from several memory locations:
|
||||||
|
|
||||||
An environment is a runtime environment, typically defined by a linker
|
|
||||||
script. The current runtime environments include
|
|
||||||
* flash: Run natively from start of flash memory
|
* flash: Run natively from start of flash memory
|
||||||
* dfu: Run after a DFU bootloader from an offset after the first 16k
|
* dfu: Run after a DFU bootloader from an offset after the first 16k of flash (the first 16k are reserved for the bootloader)
|
||||||
of flash (the first 16k are reserved for the bootloader)
|
|
||||||
* ram: Run from within the RAM of the chip, downloaded via JTAG/SWD
|
* ram: Run from within the RAM of the chip, downloaded via JTAG/SWD
|
||||||
|
|
||||||
|
|
||||||
== Building
|
== Building
|
||||||
|
|
||||||
A given software build is made for a specific combination of an APP
|
A given firmware build is made for a specific combination of an application `APP` running in a certain memory `MEM` on a given board `BOARD`.
|
||||||
running in a certain ENVIRONMENT on a given BOARD.
|
When building using `make`, set the target application using the `APP` environment variable and target board using the `BOARD` environment variable, e.g.:
|
||||||
|
|
||||||
A Makefile is provided. It will create output files in the format
|
|
||||||
bin/$(BOARD)-$(APP)-$(ENV).{elf,bin}
|
|
||||||
|
|
||||||
You can specify the APP and BOARD to build when calling make, like
|
|
||||||
e.g.
|
|
||||||
* make APP=cardem BOARD=qmod
|
* make APP=cardem BOARD=qmod
|
||||||
* make APP=dfu BOARD=qmod
|
* make APP=dfu BOARD=qmod
|
||||||
|
|
||||||
|
The Makefile will create output files in the format: `bin/$(BOARD)-$(APP)-$(MEM).{elf,bin}`
|
||||||
|
|
||||||
The level of debug messages can be altered at compile time:
|
The level of debug messages can be altered at compile time:
|
||||||
```
|
```
|
||||||
$ make TRACE_LEVEL=4
|
$ make TRACE_LEVEL=4
|
||||||
```
|
```
|
||||||
Accepted values: 0 (NO_TRACE) to 5 (DEBUG)
|
Accepted values: 0 (NO_TRACE) to 5 (DEBUG)
|
||||||
|
|
||||||
== Flashing
|
The qmod specific option `ALLOW_PEER_ERASE` controls if the UART debug command to assert the peer SAM3S ERASE line is present in the code.
|
||||||
|
Per default this is set to 0 to prevent accidentally erasing all firmware, including the DFU bootloader, which would then need to be flashed using SAM-BA or JTAG/SWD.
|
||||||
|
Setting `ALLOW_PEER_ERASE` to 1 enables back the debug command and should be used only for debugging or development purposes.
|
||||||
|
|
||||||
For flashing the firmware, there are at least two options.
|
= Flashing
|
||||||
|
|
||||||
=== Using JTAG + OpenOCD to flash the DFU bootloader
|
To flash a firmware image follow the instructions provided in the [wiki](https://projects.osmocom.org/projects/simtrace2/wiki/).
|
||||||
|
|
||||||
The first one is using openocd and a JTAG key.
|
|
||||||
For this option, a JTAG connector has to be soldered onto the board, which is not attached per default.
|
|
||||||
|
|
||||||
```
|
|
||||||
$ openocd -f openocd/openocd.cfg -c "init" -c "halt" -c "flash write_bank 0 ./bin/$(BOARD)-dfu-flash.bin 0" -c "reset" -c "shutdown"
|
|
||||||
```
|
|
||||||
|
|
||||||
=== Using bossac to flash the DFU bootloader
|
|
||||||
|
|
||||||
The second option is using rumba for flashing. No further hardware has to be provided for this option.
|
|
||||||
|
|
||||||
FIXME
|
|
||||||
|
|
||||||
=== Using DFU to flash application
|
|
||||||
|
|
||||||
FIXME
|
|
||||||
|
|||||||
22
firmware/TODO.txt
Normal file
22
firmware/TODO.txt
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
|
||||||
|
== Important DFU topics / reliability ==
|
||||||
|
x some kind of DFU fallback / boot cycle counter ?
|
||||||
|
* CRC check of image before attempting to execute it ?
|
||||||
|
x Keep WDT running while in DFU or app
|
||||||
|
? USB control request for flash erase
|
||||||
|
|
||||||
|
== QModem related ==
|
||||||
|
x new vendor/product ID for hub and SAM3s
|
||||||
|
* board-specfic string descriptors
|
||||||
|
* re-mapping of USB ports in EEPROM
|
||||||
|
|
||||||
|
== Lower Priority ==
|
||||||
|
* unique serial number in iSerial?
|
||||||
|
* printing of banner from generic function
|
||||||
|
* board_main_top() automatically before calling main()
|
||||||
|
x compile-time USB string generation
|
||||||
|
* shared USB strings for DFU and runtime
|
||||||
|
* version detection voltage ranges
|
||||||
|
* locking of bootloader pages?
|
||||||
|
* debug console command for switch-to-dfu
|
||||||
|
* read CPU reset cause (and time?) via USB
|
||||||
@@ -1,3 +1,3 @@
|
|||||||
C_FILES += $(C_LIBUSB_RT)
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
C_FILES += card_emu.c ccid.c iso7816_4.c iso7816_fidi.c mitm.c mode_cardemu.c simtrace_iso7816.c sniffer.c tc_etu.c usb.c
|
C_FILES += card_emu.c cciddriver.c iso7816_4.c iso7816_fidi.c mitm.c mode_cardemu.c mode_ccid.c simtrace_iso7816.c sniffer.c tc_etu.c usb.c
|
||||||
|
|||||||
@@ -1,4 +1,22 @@
|
|||||||
// FIXME: Copyright license here
|
/* SIMtrace 2 firmware card emulation application
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
@@ -6,7 +24,7 @@
|
|||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "simtrace.h"
|
#include "simtrace.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
#include "osmocom/core/timer.h"
|
#include <osmocom/core/timer.h>
|
||||||
|
|
||||||
unsigned int g_unique_id[4];
|
unsigned int g_unique_id[4];
|
||||||
|
|
||||||
@@ -84,17 +102,23 @@ static volatile enum confNum simtrace_config = CFG_NUM_CCID;
|
|||||||
void USBDDriverCallbacks_ConfigurationChanged(uint8_t cfgnum)
|
void USBDDriverCallbacks_ConfigurationChanged(uint8_t cfgnum)
|
||||||
{
|
{
|
||||||
TRACE_INFO_WP("cfgChanged%d ", cfgnum);
|
TRACE_INFO_WP("cfgChanged%d ", cfgnum);
|
||||||
simtrace_config = cfgnum;
|
if (cfgnum < ARRAY_SIZE(config_func_ptrs)) {
|
||||||
|
simtrace_config = cfgnum;
|
||||||
|
} else {
|
||||||
|
TRACE_ERROR("trying to set out of bounds config %u\r\n", cfgnum);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void USART1_IrqHandler(void)
|
void USART1_IrqHandler(void)
|
||||||
{
|
{
|
||||||
config_func_ptrs[simtrace_config].usart1_irq();
|
if (config_func_ptrs[simtrace_config].usart1_irq)
|
||||||
|
config_func_ptrs[simtrace_config].usart1_irq();
|
||||||
}
|
}
|
||||||
|
|
||||||
void USART0_IrqHandler(void)
|
void USART0_IrqHandler(void)
|
||||||
{
|
{
|
||||||
config_func_ptrs[simtrace_config].usart0_irq();
|
if (config_func_ptrs[simtrace_config].usart0_irq)
|
||||||
|
config_func_ptrs[simtrace_config].usart0_irq();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* returns '1' in case we should break any endless loop */
|
/* returns '1' in case we should break any endless loop */
|
||||||
@@ -106,7 +130,8 @@ static void check_exec_dbg_cmd(void)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
ch = UART_GetChar();
|
ch = UART_GetChar();
|
||||||
|
/* We must echo the character to make python fdexpect happy, which we use in factory testing */
|
||||||
|
fputc(ch, stdout);
|
||||||
board_exec_dbg_cmd(ch);
|
board_exec_dbg_cmd(ch);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -123,24 +148,38 @@ extern int main(void)
|
|||||||
led_init();
|
led_init();
|
||||||
led_blink(LED_RED, BLINK_3O_5F);
|
led_blink(LED_RED, BLINK_3O_5F);
|
||||||
|
|
||||||
/* Enable watchdog for 500ms, with no window */
|
/* Enable watchdog for 2000ms, with no window */
|
||||||
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
(WDT_GetPeriod(500) << 16) | WDT_GetPeriod(500));
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
|
||||||
PIO_InitializeInterrupts(0);
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
EEFC_ReadUniqueID(g_unique_id);
|
EEFC_ReadUniqueID(g_unique_id);
|
||||||
|
|
||||||
printf("\n\r\n\r"
|
printf("\n\r\n\r"
|
||||||
"=============================================================================\n\r"
|
"=============================================================================\n\r"
|
||||||
"SIMtrace2 firmware " GIT_VERSION " (C) 2010-2016 by Harald Welte\n\r"
|
"SIMtrace2 firmware " GIT_VERSION " (C) 2010-2016 by Harald Welte\n\r"
|
||||||
"=============================================================================\n\r");
|
"=============================================================================\n\r");
|
||||||
|
|
||||||
TRACE_INFO("Chip ID: 0x%08x (Ext 0x%08x)\n\r", CHIPID->CHIPID_CIDR, CHIPID->CHIPID_EXID);
|
#if (TRACE_LEVEL >= TRACE_LEVEL_INFO)
|
||||||
|
TRACE_INFO("Chip ID: 0x%08lx (Ext 0x%08lx)\n\r", CHIPID->CHIPID_CIDR, CHIPID->CHIPID_EXID);
|
||||||
TRACE_INFO("Serial Nr. %08x-%08x-%08x-%08x\n\r",
|
TRACE_INFO("Serial Nr. %08x-%08x-%08x-%08x\n\r",
|
||||||
g_unique_id[0], g_unique_id[1],
|
g_unique_id[0], g_unique_id[1],
|
||||||
g_unique_id[2], g_unique_id[3]);
|
g_unique_id[2], g_unique_id[3]);
|
||||||
TRACE_INFO("Reset Cause: 0x%x\n\r", (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos);
|
uint8_t reset_cause = (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos;
|
||||||
|
static const char* reset_causes[] = {
|
||||||
|
"general reset (first power-up reset)",
|
||||||
|
"backup reset (return from backup mode)",
|
||||||
|
"watchdog reset (watchdog fault occurred)",
|
||||||
|
"software reset (processor reset required by the software)",
|
||||||
|
"user reset (NRST pin detected low)",
|
||||||
|
};
|
||||||
|
if (reset_cause < ARRAY_SIZE(reset_causes)) {
|
||||||
|
TRACE_INFO("Reset Cause: %s\n\r", reset_causes[reset_cause]);
|
||||||
|
} else {
|
||||||
|
TRACE_INFO("Reset Cause: 0x%lx\n\r", (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
board_main_top();
|
board_main_top();
|
||||||
|
|
||||||
@@ -169,7 +208,9 @@ extern int main(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
TRACE_INFO("calling init of config %u...\n\r", simtrace_config);
|
TRACE_INFO("calling init of config %u...\n\r", simtrace_config);
|
||||||
config_func_ptrs[simtrace_config].init();
|
if (config_func_ptrs[simtrace_config].init) {
|
||||||
|
config_func_ptrs[simtrace_config].init();
|
||||||
|
}
|
||||||
last_simtrace_config = simtrace_config;
|
last_simtrace_config = simtrace_config;
|
||||||
|
|
||||||
TRACE_INFO("entering main loop...\n\r");
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
@@ -197,11 +238,17 @@ extern int main(void)
|
|||||||
if (last_simtrace_config != simtrace_config) {
|
if (last_simtrace_config != simtrace_config) {
|
||||||
TRACE_INFO("USB config chg %u -> %u\n\r",
|
TRACE_INFO("USB config chg %u -> %u\n\r",
|
||||||
last_simtrace_config, simtrace_config);
|
last_simtrace_config, simtrace_config);
|
||||||
config_func_ptrs[last_simtrace_config].exit();
|
if (config_func_ptrs[last_simtrace_config].exit) {
|
||||||
config_func_ptrs[simtrace_config].init();
|
config_func_ptrs[last_simtrace_config].exit();
|
||||||
|
}
|
||||||
|
if (config_func_ptrs[simtrace_config].init) {
|
||||||
|
config_func_ptrs[simtrace_config].init();
|
||||||
|
}
|
||||||
last_simtrace_config = simtrace_config;
|
last_simtrace_config = simtrace_config;
|
||||||
} else {
|
} else {
|
||||||
config_func_ptrs[simtrace_config].run();
|
if (config_func_ptrs[simtrace_config].run) {
|
||||||
|
config_func_ptrs[simtrace_config].run();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,28 @@
|
|||||||
|
/* SIMtrace 2 firmware USB DFU bootloader
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*/
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
#include "usb/device/dfu/dfu.h"
|
#include "usb/device/dfu/dfu.h"
|
||||||
#include "usb/common/dfu/usb_dfu.h"
|
#include "usb/common/dfu/usb_dfu.h"
|
||||||
#include "manifest.h"
|
#include "manifest.h"
|
||||||
|
#include "USBD_HAL.h"
|
||||||
|
|
||||||
#include <osmocom/core/timer.h>
|
#include <osmocom/core/timer.h>
|
||||||
|
|
||||||
@@ -10,6 +30,14 @@
|
|||||||
#define ALTIF_FLASH 1
|
#define ALTIF_FLASH 1
|
||||||
|
|
||||||
unsigned int g_unique_id[4];
|
unsigned int g_unique_id[4];
|
||||||
|
/* remember if the watchdog has been configured in the main loop so we can kick it in the ISR */
|
||||||
|
static bool watchdog_configured = false;
|
||||||
|
|
||||||
|
/* There is not enough space in the 16 KiB DFU bootloader to include led.h functions */
|
||||||
|
#ifdef PINS_LEDS
|
||||||
|
/** LED pin configurations */
|
||||||
|
static const Pin pinsLeds[] = { PINS_LEDS } ;
|
||||||
|
#endif
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Callbacks
|
* Callbacks
|
||||||
@@ -21,46 +49,81 @@ unsigned int g_unique_id[4];
|
|||||||
#define IFLASH_END ((uint8_t *)IFLASH_ADDR + IFLASH_SIZE)
|
#define IFLASH_END ((uint8_t *)IFLASH_ADDR + IFLASH_SIZE)
|
||||||
#define IRAM_END ((uint8_t *)IRAM_ADDR + IRAM_SIZE)
|
#define IRAM_END ((uint8_t *)IRAM_ADDR + IRAM_SIZE)
|
||||||
|
|
||||||
/* incoming call-back: Host has transfered 'len' bytes (stored at
|
/* incoming call-back: Host has transferred 'len' bytes (stored at
|
||||||
* 'data'), which we shall write to 'offset' into the partition
|
* 'data'), which we shall write to 'offset' into the partition
|
||||||
* associated with 'altif'. Guaranted to be les than
|
* associated with 'altif'. Guaranted to be less than
|
||||||
* BOARD_DFU_PAGE_SIZE */
|
* BOARD_DFU_PAGE_SIZE */
|
||||||
int USBDFU_handle_dnload(uint8_t altif, unsigned int offset,
|
int USBDFU_handle_dnload(uint8_t altif, unsigned int offset,
|
||||||
uint8_t *data, unsigned int len)
|
uint8_t *data, unsigned int len)
|
||||||
{
|
{
|
||||||
uint32_t addr;
|
uint32_t addr;
|
||||||
|
unsigned int i;
|
||||||
int rc;
|
int rc;
|
||||||
|
/* address of the last allocated variable on the stack */
|
||||||
|
uint32_t stack_addr = (uint32_t)&rc;
|
||||||
|
/* kick the dog to have enough time to flash */
|
||||||
|
if (watchdog_configured) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
}
|
||||||
|
|
||||||
printf("dnload(altif=%u, offset=%u, len=%u)\n\r", altif, offset, len);
|
printf("dnload(altif=%u, offset=%u, len=%u)\n\r", altif, offset, len);
|
||||||
|
|
||||||
|
#ifdef PINS_LEDS
|
||||||
|
PIO_Clear(&pinsLeds[LED_NUM_RED]);
|
||||||
|
#endif
|
||||||
|
|
||||||
switch (altif) {
|
switch (altif) {
|
||||||
case ALTIF_RAM:
|
case ALTIF_RAM:
|
||||||
addr = RAM_ADDR(offset);
|
addr = RAM_ADDR(offset);
|
||||||
if (addr > IRAM_ADDR + IRAM_SIZE) {
|
if (addr < IRAM_ADDR || addr + len >= IRAM_ADDR + IRAM_SIZE || addr + len >= stack_addr) {
|
||||||
g_dfu->state = DFU_STATE_dfuERROR;
|
g_dfu->state = DFU_STATE_dfuERROR;
|
||||||
g_dfu->status = DFU_STATUS_errADDRESS;
|
g_dfu->status = DFU_STATUS_errADDRESS;
|
||||||
return DFU_RET_STALL;
|
rc = DFU_RET_STALL;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
memcpy((void *)addr, data, len);
|
memcpy((void *)addr, data, len);
|
||||||
return DFU_RET_ZLP;
|
rc = DFU_RET_ZLP;
|
||||||
|
break;
|
||||||
case ALTIF_FLASH:
|
case ALTIF_FLASH:
|
||||||
addr = FLASH_ADDR(offset);
|
addr = FLASH_ADDR(offset);
|
||||||
if (addr > IFLASH_ADDR + IFLASH_SIZE) {
|
if (addr < IFLASH_ADDR || addr + len >= IFLASH_ADDR + IFLASH_SIZE) {
|
||||||
g_dfu->state = DFU_STATE_dfuERROR;
|
g_dfu->state = DFU_STATE_dfuERROR;
|
||||||
g_dfu->status = DFU_STATUS_errADDRESS;
|
g_dfu->status = DFU_STATUS_errADDRESS;
|
||||||
return DFU_RET_STALL;
|
rc = DFU_RET_STALL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
rc = FLASHD_Unlock(addr, addr + len, 0, 0);
|
||||||
|
if (rc != 0) {
|
||||||
|
TRACE_ERROR("DFU download flash unlock failed\n\r");
|
||||||
|
rc = DFU_RET_STALL;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
rc = FLASHD_Write(addr, data, len);
|
rc = FLASHD_Write(addr, data, len);
|
||||||
if (rc != 0) {
|
if (rc != 0) {
|
||||||
/* FIXME: set error codes */
|
TRACE_ERROR("DFU download flash erase failed\n\r");
|
||||||
return DFU_RET_STALL;
|
rc = DFU_RET_STALL;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return DFU_RET_ZLP;
|
for (i = 0; i < len; i++) {
|
||||||
|
if (((uint8_t*)addr)[i]!=data[i]) {
|
||||||
|
TRACE_ERROR("DFU download flash data written not correct\n\r");
|
||||||
|
rc = DFU_RET_STALL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
rc = DFU_RET_ZLP;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
/* FIXME: set error codes */
|
|
||||||
TRACE_ERROR("DFU download for unknown AltIf %d\n\r", altif);
|
TRACE_ERROR("DFU download for unknown AltIf %d\n\r", altif);
|
||||||
return DFU_RET_STALL;
|
rc = DFU_RET_STALL;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef PINS_LEDS
|
||||||
|
PIO_Set(&pinsLeds[LED_NUM_RED]);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* incoming call-back: Host has requested to read back 'req_len' bytes
|
/* incoming call-back: Host has requested to read back 'req_len' bytes
|
||||||
@@ -105,31 +168,10 @@ int USBDFU_handle_upload(uint8_t altif, unsigned int offset,
|
|||||||
return req_len;
|
return req_len;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int uart_has_loopback_jumper(void)
|
/* can be overridden by board specific code, e.g. by pushbutton */
|
||||||
|
WEAK int board_override_enter_dfu(void)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
return 0;
|
||||||
const Pin uart_loopback_pins[] = {
|
|
||||||
{PIO_PA9A_URXD0, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT},
|
|
||||||
{PIO_PA10A_UTXD0, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
|
||||||
};
|
|
||||||
|
|
||||||
/* Configure UART pins as I/O */
|
|
||||||
PIO_Configure(uart_loopback_pins, PIO_LISTSIZE(uart_loopback_pins));
|
|
||||||
|
|
||||||
for (i = 0; i < 10; i++) {
|
|
||||||
/* Set TxD high; abort if RxD doesn't go high either */
|
|
||||||
PIO_Set(&uart_loopback_pins[1]);
|
|
||||||
if (!PIO_Get(&uart_loopback_pins[0]))
|
|
||||||
return 0;
|
|
||||||
/* Set TxD low, abort if RxD doesn't go low either */
|
|
||||||
PIO_Clear(&uart_loopback_pins[1]);
|
|
||||||
if (PIO_Get(&uart_loopback_pins[0]))
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
/* if we reached here, RxD always follows TxD and thus a
|
|
||||||
* loopback jumper has been placed on RxD/TxD, and we will boot
|
|
||||||
* into DFU unconditionally */
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* using this function we can determine if we should enter DFU mode
|
/* using this function we can determine if we should enter DFU mode
|
||||||
@@ -137,22 +179,30 @@ static int uart_has_loopback_jumper(void)
|
|||||||
int USBDFU_OverrideEnterDFU(void)
|
int USBDFU_OverrideEnterDFU(void)
|
||||||
{
|
{
|
||||||
uint32_t *app_part = (uint32_t *)FLASH_ADDR(0);
|
uint32_t *app_part = (uint32_t *)FLASH_ADDR(0);
|
||||||
|
/* at the first call we are before the text segment has been relocated,
|
||||||
|
* so g_dfu is not initialized yet */
|
||||||
|
g_dfu = &_g_dfu;
|
||||||
|
if (USB_DFU_MAGIC == g_dfu->magic) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
/* If the loopback jumper is set, we enter DFU mode */
|
/* If the loopback jumper is set, we enter DFU mode */
|
||||||
if (uart_has_loopback_jumper())
|
if (board_override_enter_dfu()) {
|
||||||
return 1;
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
/* if the first word of the application partition doesn't look
|
/* if the first word of the application partition doesn't look
|
||||||
* like a stack pointer (i.e. point to RAM), enter DFU mode */
|
* like a stack pointer (i.e. point to RAM), enter DFU mode */
|
||||||
if ((app_part[0] < IRAM_ADDR) ||
|
if ((app_part[0] < IRAM_ADDR) || ((uint8_t *)app_part[0] > IRAM_END)) {
|
||||||
((uint8_t *)app_part[0] > IRAM_END))
|
return 3;
|
||||||
return 1;
|
}
|
||||||
|
|
||||||
/* if the second word of the application partition doesn't look
|
/* if the second word of the application partition doesn't look
|
||||||
* like a function from flash (reset vector), enter DFU mode */
|
* like a function from flash (reset vector), enter DFU mode */
|
||||||
if (((uint32_t *)app_part[1] < app_part) ||
|
if (((uint32_t *)app_part[1] < app_part) ||
|
||||||
((uint8_t *)app_part[1] > IFLASH_END))
|
((uint8_t *)app_part[1] > IFLASH_END)) {
|
||||||
return 1;
|
return 4;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -180,21 +230,23 @@ extern int main(void)
|
|||||||
unsigned int i = 0;
|
unsigned int i = 0;
|
||||||
uint32_t reset_cause = (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos;
|
uint32_t reset_cause = (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos;
|
||||||
|
|
||||||
#if 0
|
/* Enable watchdog for 2000ms, with no window */
|
||||||
led_init();
|
|
||||||
led_blink(LED_GREEN, BLINK_3O_30F);
|
|
||||||
led_blink(LED_RED, BLINK_3O_30F);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Enable watchdog for 500ms, with no window */
|
|
||||||
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
(WDT_GetPeriod(500) << 16) | WDT_GetPeriod(500));
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
watchdog_configured = true;
|
||||||
|
|
||||||
|
#ifdef PINS_LEDS
|
||||||
|
/* Configure LED */
|
||||||
|
PIO_Configure(pinsLeds, sizeof(pinsLeds));
|
||||||
|
PIO_Set(&pinsLeds[LED_NUM_RED]);
|
||||||
|
PIO_Clear(&pinsLeds[LED_NUM_GREEN]);
|
||||||
|
#endif
|
||||||
|
|
||||||
PIO_InitializeInterrupts(0);
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
EEFC_ReadUniqueID(g_unique_id);
|
EEFC_ReadUniqueID(g_unique_id);
|
||||||
|
|
||||||
printf("\n\r\n\r"
|
printf("\n\r\n\r"
|
||||||
"=============================================================================\n\r"
|
"=============================================================================\n\r"
|
||||||
"DFU bootloader %s for board %s (C) 2010-2017 by Harald Welte\n\r"
|
"DFU bootloader %s for board %s (C) 2010-2017 by Harald Welte\n\r"
|
||||||
"=============================================================================\n\r",
|
"=============================================================================\n\r",
|
||||||
@@ -204,7 +256,36 @@ extern int main(void)
|
|||||||
TRACE_INFO("Serial Nr. %08x-%08x-%08x-%08x\n\r",
|
TRACE_INFO("Serial Nr. %08x-%08x-%08x-%08x\n\r",
|
||||||
g_unique_id[0], g_unique_id[1],
|
g_unique_id[0], g_unique_id[1],
|
||||||
g_unique_id[2], g_unique_id[3]);
|
g_unique_id[2], g_unique_id[3]);
|
||||||
TRACE_INFO("Reset Cause: 0x%x\n\r", reset_cause);
|
TRACE_INFO("Reset Cause: 0x%lx\n\r", reset_cause);
|
||||||
|
|
||||||
|
#if (TRACE_LEVEL >= TRACE_LEVEL_INFO)
|
||||||
|
/* Find out why we are in the DFU bootloader, and not the main application */
|
||||||
|
TRACE_INFO("DFU bootloader start reason: ");
|
||||||
|
switch (USBDFU_OverrideEnterDFU()) {
|
||||||
|
case 0:
|
||||||
|
/* 0 normally means that there is no override, but we are in the bootloader,
|
||||||
|
* thus the first check in board_cstartup_gnu did return something else than 0.
|
||||||
|
* this can only be g_dfu->magic which is erased when the segment are
|
||||||
|
* relocated, which happens in board_cstartup_gnu just after USBDFU_OverrideEnterDFU.
|
||||||
|
* no static variable can be used to store this case since this will also be overwritten
|
||||||
|
*/
|
||||||
|
case 1:
|
||||||
|
TRACE_INFO_WP("DFU switch requested by main application\n\r");
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
TRACE_INFO_WP("bootloader forced (button pressed or jumper set)\n\r");
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
TRACE_INFO_WP("stack pointer (first application word) does no point in RAM\n\r");
|
||||||
|
break;
|
||||||
|
case 4: // the is no reason
|
||||||
|
TRACE_INFO_WP("reset vector (second application word) does no point in flash\n\r");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
TRACE_INFO_WP("unknown\n\r");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/* clear g_dfu on power-up reset */
|
/* clear g_dfu on power-up reset */
|
||||||
if (reset_cause == 0)
|
if (reset_cause == 0)
|
||||||
@@ -213,6 +294,19 @@ extern int main(void)
|
|||||||
board_main_top();
|
board_main_top();
|
||||||
|
|
||||||
TRACE_INFO("USB init...\n\r");
|
TRACE_INFO("USB init...\n\r");
|
||||||
|
/* Signal USB reset by disabling the pull-up on USB D+ for at least 10 ms */
|
||||||
|
#ifdef PIN_USB_PULLUP
|
||||||
|
const Pin usb_dp_pullup = PIN_USB_PULLUP;
|
||||||
|
PIO_Configure(&usb_dp_pullup, 1);
|
||||||
|
PIO_Set(&usb_dp_pullup);
|
||||||
|
#endif
|
||||||
|
USBD_HAL_Suspend();
|
||||||
|
mdelay(20);
|
||||||
|
#ifdef PIN_USB_PULLUP
|
||||||
|
PIO_Clear(&usb_dp_pullup);
|
||||||
|
#endif
|
||||||
|
USBD_HAL_Activate();
|
||||||
|
|
||||||
USBDFU_Initialize(&dfu_descriptors);
|
USBDFU_Initialize(&dfu_descriptors);
|
||||||
|
|
||||||
while (USBD_GetState() < USBD_STATE_CONFIGURED) {
|
while (USBD_GetState() < USBD_STATE_CONFIGURED) {
|
||||||
@@ -229,7 +323,9 @@ extern int main(void)
|
|||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Initialize the flash to be able to write it, using the IAP ROM code */
|
||||||
FLASHD_Initialize(BOARD_MCK, 1);
|
FLASHD_Initialize(BOARD_MCK, 1);
|
||||||
|
|
||||||
TRACE_INFO("entering main loop...\n\r");
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
while (1) {
|
while (1) {
|
||||||
WDT_Restart(WDT);
|
WDT_Restart(WDT);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
sysmocom - s.f.m.c. GmbH
|
sysmocom - s.f.m.c. GmbH
|
||||||
SIMtrace 2 compatible device
|
SIMtrace 2 compatible device
|
||||||
DFU (Device Firmare Upgrade)
|
DFU (Device Firmware Upgrade)
|
||||||
RAM
|
RAM
|
||||||
Flash (Application Partition)
|
Flash (Application Partition)
|
||||||
|
|||||||
3
firmware/apps/freq_ctr/Makefile
Normal file
3
firmware/apps/freq_ctr/Makefile
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
|
C_FILES += freq_ctr.c
|
||||||
55
firmware/apps/freq_ctr/freq_ctr.c
Normal file
55
firmware/apps/freq_ctr/freq_ctr.c
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
#include <stdint.h>
|
||||||
|
#include "utils.h"
|
||||||
|
#include "tc_etu.h"
|
||||||
|
#include "chip.h"
|
||||||
|
|
||||||
|
|
||||||
|
/* pins for Channel 0 of TC-block 0 */
|
||||||
|
#define PIN_TIOA0 {PIO_PA0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/* pins for Channel 1 of TC-block 0 */
|
||||||
|
#define PIN_TIOA1 {PIO_PA15, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
#define PIN_TCLK1 {PIO_PA28, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
static const Pin pins_tc[] = { PIN_TIOA0, PIN_TIOA1, PIN_TCLK1 };
|
||||||
|
|
||||||
|
static TcChannel *tc1 = &TC0->TC_CHANNEL[1];
|
||||||
|
|
||||||
|
void TC1_IrqHandler(void)
|
||||||
|
{
|
||||||
|
uint32_t sr = tc1->TC_SR;
|
||||||
|
printf("TC1=%lu; SR=0x%08lx\r\n", tc1->TC_RA, sr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void freq_ctr_init(void)
|
||||||
|
{
|
||||||
|
TcChannel *tc0 = &TC0->TC_CHANNEL[0];
|
||||||
|
|
||||||
|
PIO_Configure(pins_tc, ARRAY_SIZE(pins_tc));
|
||||||
|
|
||||||
|
PMC_EnablePeripheral(ID_TC0);
|
||||||
|
PMC_EnablePeripheral(ID_TC1);
|
||||||
|
|
||||||
|
/* route TCLK1 to XC1 */
|
||||||
|
TC0->TC_BMR &= ~TC_BMR_TC1XC1S_Msk;
|
||||||
|
TC0->TC_BMR |= TC_BMR_TC1XC1S_TCLK1;
|
||||||
|
|
||||||
|
/* TC0 in wveform mode: Run from SCLK. Raise TIOA on RA; lower TIOA on RC + trigger */
|
||||||
|
tc0->TC_CMR = TC_CMR_TCCLKS_TIMER_CLOCK5 | TC_CMR_BURST_NONE |
|
||||||
|
TC_CMR_EEVTEDG_NONE | TC_CMR_WAVSEL_UP_RC | TC_CMR_WAVE |
|
||||||
|
TC_CMR_ACPA_SET | TC_CMR_ACPC_CLEAR;
|
||||||
|
tc0->TC_RA = 16384; /* set high at 16384 */
|
||||||
|
tc0->TC_RC = 32786; /* set low at 32786 */
|
||||||
|
|
||||||
|
/* TC1 in capture mode: Run from XC1. Trigger on TIOA rising. Load RA on rising */
|
||||||
|
tc1->TC_CMR = TC_CMR_TCCLKS_XC1 | TC_CMR_BURST_NONE |
|
||||||
|
TC_CMR_ETRGEDG_RISING | TC_CMR_ABETRG | TC_CMR_LDRA_RISING;
|
||||||
|
/* Interrupt us if the external trigger happens */
|
||||||
|
tc1->TC_IER = TC_IER_ETRGS;
|
||||||
|
NVIC_EnableIRQ(TC1_IRQn);
|
||||||
|
|
||||||
|
TC0->TC_BCR = TC_BCR_SYNC;
|
||||||
|
|
||||||
|
tc0->TC_CCR = TC_CCR_CLKEN|TC_CCR_SWTRG;
|
||||||
|
tc1->TC_CCR = TC_CCR_CLKEN|TC_CCR_SWTRG;
|
||||||
|
}
|
||||||
54
firmware/apps/freq_ctr/main.c
Normal file
54
firmware/apps/freq_ctr/main.c
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "osmocom/core/timer.h"
|
||||||
|
|
||||||
|
extern void freq_ctr_init(void);
|
||||||
|
|
||||||
|
/* returns '1' in case we should break any endless loop */
|
||||||
|
static void check_exec_dbg_cmd(void)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
if (!UART_IsRxReady())
|
||||||
|
return;
|
||||||
|
|
||||||
|
ch = UART_GetChar();
|
||||||
|
|
||||||
|
board_exec_dbg_cmd(ch);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
extern int main(void)
|
||||||
|
{
|
||||||
|
led_init();
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
|
||||||
|
/* Enable watchdog for 2000 ms, with no window */
|
||||||
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
|
||||||
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
|
|
||||||
|
printf("\n\r\n\r"
|
||||||
|
"=============================================================================\n\r"
|
||||||
|
"Freq Ctr firmware " GIT_VERSION " (C) 2019 by Harald Welte\n\r"
|
||||||
|
"=============================================================================\n\r");
|
||||||
|
|
||||||
|
board_main_top();
|
||||||
|
|
||||||
|
TRACE_INFO("starting frequency counter...\n\r");
|
||||||
|
freq_ctr_init();
|
||||||
|
|
||||||
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
|
while (1) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
check_exec_dbg_cmd();
|
||||||
|
osmo_timers_prepare();
|
||||||
|
osmo_timers_update();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
10
firmware/apps/freq_ctr/usb_strings.txt
Normal file
10
firmware/apps/freq_ctr/usb_strings.txt
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
sysmocom - s.f.m.c. GmbH
|
||||||
|
SIMtrace 2 compatible device
|
||||||
|
SIMtrace Sniffer
|
||||||
|
SIMtrace CCID
|
||||||
|
SIMtrace Phone
|
||||||
|
SIMtrace MITM
|
||||||
|
CardEmulator Modem 1
|
||||||
|
CardEmulator Modem 2
|
||||||
|
CardEmulator Modem 3
|
||||||
|
CardEmulator Modem 4
|
||||||
3
firmware/apps/gpio_test/Makefile
Normal file
3
firmware/apps/gpio_test/Makefile
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
|
C_FILES += gpio_test.c
|
||||||
8
firmware/apps/gpio_test/gpio_test.c
Normal file
8
firmware/apps/gpio_test/gpio_test.c
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
#include <stdint.h>
|
||||||
|
#include "utils.h"
|
||||||
|
#include "chip.h"
|
||||||
|
|
||||||
|
void gpio_test_init(void)
|
||||||
|
{
|
||||||
|
printf("FIXME run tests here\n\r");
|
||||||
|
}
|
||||||
54
firmware/apps/gpio_test/main.c
Normal file
54
firmware/apps/gpio_test/main.c
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "osmocom/core/timer.h"
|
||||||
|
|
||||||
|
extern void gpio_test_init(void);
|
||||||
|
|
||||||
|
/* returns '1' in case we should break any endless loop */
|
||||||
|
static void check_exec_dbg_cmd(void)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
if (!UART_IsRxReady())
|
||||||
|
return;
|
||||||
|
|
||||||
|
ch = UART_GetChar();
|
||||||
|
|
||||||
|
board_exec_dbg_cmd(ch);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
extern int main(void)
|
||||||
|
{
|
||||||
|
led_init();
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
|
||||||
|
/* Enable watchdog for 2000 ms, with no window */
|
||||||
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
|
||||||
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
|
|
||||||
|
printf("\n\r\n\r"
|
||||||
|
"=============================================================================\n\r"
|
||||||
|
"GPIO Test firmware " GIT_VERSION " (C) 2019 Sysmocom GmbH\n\r"
|
||||||
|
"=============================================================================\n\r");
|
||||||
|
|
||||||
|
board_main_top();
|
||||||
|
|
||||||
|
TRACE_INFO("starting gpio test...\n\r");
|
||||||
|
gpio_test_init();
|
||||||
|
|
||||||
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
|
while (1) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
check_exec_dbg_cmd();
|
||||||
|
osmo_timers_prepare();
|
||||||
|
osmo_timers_update();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
10
firmware/apps/gpio_test/usb_strings.txt
Normal file
10
firmware/apps/gpio_test/usb_strings.txt
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
sysmocom - s.f.m.c. GmbH
|
||||||
|
SIMtrace 2 compatible device
|
||||||
|
SIMtrace Sniffer
|
||||||
|
SIMtrace CCID
|
||||||
|
SIMtrace Phone
|
||||||
|
SIMtrace MITM
|
||||||
|
CardEmulator Modem 1
|
||||||
|
CardEmulator Modem 2
|
||||||
|
CardEmulator Modem 3
|
||||||
|
CardEmulator Modem 4
|
||||||
3
firmware/apps/trace/Makefile
Normal file
3
firmware/apps/trace/Makefile
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
|
C_FILES += card_emu.c cciddriver.c iso7816_4.c iso7816_fidi.c mitm.c mode_cardemu.c mode_ccid.c simtrace_iso7816.c sniffer.c tc_etu.c usb.c
|
||||||
@@ -0,0 +1,236 @@
|
|||||||
|
/* SIMtrace 2 firmware sniffer application
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
/*------------------------------------------------------------------------------
|
||||||
|
* Headers
|
||||||
|
*------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "simtrace.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "osmocom/core/timer.h"
|
||||||
|
|
||||||
|
unsigned int g_unique_id[4];
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------------
|
||||||
|
* Internal variables
|
||||||
|
*------------------------------------------------------------------------------*/
|
||||||
|
typedef struct {
|
||||||
|
/* static initialization, called whether or not the usb config is active */
|
||||||
|
void (*configure) (void);
|
||||||
|
/* initialization function after the config was selected */
|
||||||
|
void (*init) (void);
|
||||||
|
/* de-initialization before selecting new config */
|
||||||
|
void (*exit) (void);
|
||||||
|
/* main loop content for given configuration */
|
||||||
|
void (*run) (void);
|
||||||
|
/* Interrupt handler for USART0 */
|
||||||
|
void (*usart0_irq) (void);
|
||||||
|
/* Interrupt handler for USART1 */
|
||||||
|
void (*usart1_irq) (void);
|
||||||
|
} conf_func;
|
||||||
|
|
||||||
|
static const conf_func config_func_ptrs[] = {
|
||||||
|
/* array slot 0 is empty, usb configs start at 1 */
|
||||||
|
#ifdef HAVE_SNIFFER
|
||||||
|
[CFG_NUM_SNIFF] = {
|
||||||
|
.configure = Sniffer_configure,
|
||||||
|
.init = Sniffer_init,
|
||||||
|
.exit = Sniffer_exit,
|
||||||
|
.run = Sniffer_run,
|
||||||
|
.usart0_irq = Sniffer_usart0_irq,
|
||||||
|
.usart1_irq = Sniffer_usart1_irq,
|
||||||
|
},
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_CCID
|
||||||
|
[CFG_NUM_CCID] = {
|
||||||
|
.configure = CCID_configure,
|
||||||
|
.init = CCID_init,
|
||||||
|
.exit = CCID_exit,
|
||||||
|
.run = CCID_run,
|
||||||
|
},
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_CARDEM
|
||||||
|
[CFG_NUM_PHONE] = {
|
||||||
|
.configure = mode_cardemu_configure,
|
||||||
|
.init = mode_cardemu_init,
|
||||||
|
.exit = mode_cardemu_exit,
|
||||||
|
.run = mode_cardemu_run,
|
||||||
|
.usart0_irq = mode_cardemu_usart0_irq,
|
||||||
|
.usart1_irq = mode_cardemu_usart1_irq,
|
||||||
|
},
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_MITM
|
||||||
|
[CFG_NUM_MITM] = {
|
||||||
|
.configure = MITM_configure,
|
||||||
|
.init = MITM_init,
|
||||||
|
.exit = MITM_exit,
|
||||||
|
.run = MITM_run,
|
||||||
|
},
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------------
|
||||||
|
* Internal variables
|
||||||
|
*------------------------------------------------------------------------------*/
|
||||||
|
#if defined(HAVE_SNIFFER)
|
||||||
|
static volatile enum confNum simtrace_config = CFG_NUM_SNIFF;
|
||||||
|
#elif defined(HAVE_CARDEM)
|
||||||
|
static volatile enum confNum simtrace_config = CFG_NUM_PHONE;
|
||||||
|
#elif defined(HAVE_CCID)
|
||||||
|
static volatile enum confNum simtrace_config = CFG_NUM_CCID;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*----------------------------------------------------------------------------
|
||||||
|
* Callbacks
|
||||||
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
void USBDDriverCallbacks_ConfigurationChanged(uint8_t cfgnum)
|
||||||
|
{
|
||||||
|
TRACE_INFO_WP("cfgChanged%d ", cfgnum);
|
||||||
|
simtrace_config = cfgnum;
|
||||||
|
}
|
||||||
|
|
||||||
|
void USART1_IrqHandler(void)
|
||||||
|
{
|
||||||
|
if (config_func_ptrs[simtrace_config].usart1_irq)
|
||||||
|
config_func_ptrs[simtrace_config].usart1_irq();
|
||||||
|
}
|
||||||
|
|
||||||
|
void USART0_IrqHandler(void)
|
||||||
|
{
|
||||||
|
if (config_func_ptrs[simtrace_config].usart0_irq)
|
||||||
|
config_func_ptrs[simtrace_config].usart0_irq();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* returns '1' in case we should break any endless loop */
|
||||||
|
static void check_exec_dbg_cmd(void)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
if (!UART_IsRxReady())
|
||||||
|
return;
|
||||||
|
|
||||||
|
ch = UART_GetChar();
|
||||||
|
|
||||||
|
board_exec_dbg_cmd(ch);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------------
|
||||||
|
* Main
|
||||||
|
*------------------------------------------------------------------------------*/
|
||||||
|
#define MAX_USB_ITER BOARD_MCK/72 // This should be around a second
|
||||||
|
extern int main(void)
|
||||||
|
{
|
||||||
|
uint8_t isUsbConnected = 0;
|
||||||
|
enum confNum last_simtrace_config = simtrace_config;
|
||||||
|
unsigned int i = 0;
|
||||||
|
|
||||||
|
/* Configure LED output
|
||||||
|
* red on = power
|
||||||
|
* red blink = error
|
||||||
|
* green on = running
|
||||||
|
* green blink = activity
|
||||||
|
*/
|
||||||
|
led_init();
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
|
||||||
|
/* Enable watchdog for 2000 ms, with no window */
|
||||||
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
|
||||||
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
|
EEFC_ReadUniqueID(g_unique_id);
|
||||||
|
|
||||||
|
printf("\n\r\n\r"
|
||||||
|
"=============================================================================\n\r"
|
||||||
|
"SIMtrace2 firmware " GIT_VERSION " (C) 2010-2016 by Harald Welte\n\r"
|
||||||
|
"=============================================================================\n\r");
|
||||||
|
|
||||||
|
TRACE_INFO("Chip ID: 0x%08lx (Ext 0x%08lx)\n\r", CHIPID->CHIPID_CIDR, CHIPID->CHIPID_EXID);
|
||||||
|
TRACE_INFO("Serial Nr. %08x-%08x-%08x-%08x\n\r",
|
||||||
|
g_unique_id[0], g_unique_id[1],
|
||||||
|
g_unique_id[2], g_unique_id[3]);
|
||||||
|
TRACE_INFO("Reset Cause: 0x%lx\n\r", (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos);
|
||||||
|
TRACE_INFO("USB configuration used: %d\n\r", simtrace_config);
|
||||||
|
|
||||||
|
board_main_top();
|
||||||
|
|
||||||
|
TRACE_INFO("USB init...\n\r");
|
||||||
|
SIMtrace_USB_Initialize();
|
||||||
|
|
||||||
|
while (USBD_GetState() < USBD_STATE_CONFIGURED) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
check_exec_dbg_cmd();
|
||||||
|
#if 0
|
||||||
|
if (i >= MAX_USB_ITER * 3) {
|
||||||
|
TRACE_ERROR("Resetting board (USB could "
|
||||||
|
"not be configured)\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
TRACE_INFO("calling configure of all configurations...\n\r");
|
||||||
|
for (i = 1; i < ARRAY_SIZE(config_func_ptrs); i++) {
|
||||||
|
if (config_func_ptrs[i].configure)
|
||||||
|
config_func_ptrs[i].configure();
|
||||||
|
}
|
||||||
|
|
||||||
|
TRACE_INFO("calling init of config %u...\n\r", simtrace_config);
|
||||||
|
config_func_ptrs[simtrace_config].init();
|
||||||
|
last_simtrace_config = simtrace_config;
|
||||||
|
|
||||||
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
|
while (1) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
#if TRACE_LEVEL >= TRACE_LEVEL_DEBUG
|
||||||
|
const char rotor[] = { '-', '\\', '|', '/' };
|
||||||
|
putchar('\b');
|
||||||
|
putchar(rotor[i++ % ARRAY_SIZE(rotor)]);
|
||||||
|
#endif
|
||||||
|
check_exec_dbg_cmd();
|
||||||
|
osmo_timers_prepare();
|
||||||
|
osmo_timers_update();
|
||||||
|
|
||||||
|
if (USBD_GetState() < USBD_STATE_CONFIGURED) {
|
||||||
|
|
||||||
|
if (isUsbConnected) {
|
||||||
|
isUsbConnected = 0;
|
||||||
|
}
|
||||||
|
} else if (isUsbConnected == 0) {
|
||||||
|
TRACE_INFO("USB is now configured\n\r");
|
||||||
|
|
||||||
|
isUsbConnected = 1;
|
||||||
|
}
|
||||||
|
if (last_simtrace_config != simtrace_config) {
|
||||||
|
TRACE_INFO("USB config chg %u -> %u\n\r",
|
||||||
|
last_simtrace_config, simtrace_config);
|
||||||
|
config_func_ptrs[last_simtrace_config].exit();
|
||||||
|
config_func_ptrs[simtrace_config].init();
|
||||||
|
last_simtrace_config = simtrace_config;
|
||||||
|
} else {
|
||||||
|
config_func_ptrs[simtrace_config].run();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
10
firmware/apps/trace/usb_strings.txt
Normal file
10
firmware/apps/trace/usb_strings.txt
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
sysmocom - s.f.m.c. GmbH
|
||||||
|
SIMtrace 2 compatible device
|
||||||
|
SIMtrace Sniffer
|
||||||
|
SIMtrace CCID
|
||||||
|
SIMtrace Phone
|
||||||
|
SIMtrace MITM
|
||||||
|
CardEmulator Modem 1
|
||||||
|
CardEmulator Modem 2
|
||||||
|
CardEmulator Modem 3
|
||||||
|
CardEmulator Modem 4
|
||||||
@@ -1,3 +1,3 @@
|
|||||||
C_FILES += $(C_LIBUSB_RT)
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
C_FILES += card_emu.c ccid.c iso7816_4.c iso7816_fidi.c mitm.c mode_cardemu.c simtrace_iso7816.c sniffer.c tc_etu.c usb.c
|
C_FILES += card_emu.c iso7816_4.c iso7816_fidi.c mitm.c mode_cardemu.c mode_ccid.c simtrace_iso7816.c sniffer.c tc_etu.c usb.c
|
||||||
|
|||||||
@@ -1,4 +1,21 @@
|
|||||||
// FIXME: Copyright license here
|
/* SIMtrace 2 firmware card emulation, CCID, and sniffer application
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.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
|
||||||
|
*/
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
@@ -7,7 +24,7 @@
|
|||||||
#include "simtrace.h"
|
#include "simtrace.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
#include "req_ctx.h"
|
#include "req_ctx.h"
|
||||||
#include "osmocom/core/timer.h"
|
#include <osmocom/core/timer.h>
|
||||||
|
|
||||||
unsigned int g_unique_id[4];
|
unsigned int g_unique_id[4];
|
||||||
|
|
||||||
@@ -134,7 +151,7 @@ extern int main(void)
|
|||||||
|
|
||||||
EEFC_ReadUniqueID(g_unique_id);
|
EEFC_ReadUniqueID(g_unique_id);
|
||||||
|
|
||||||
printf("\r\n\r\n"
|
printf("\r\n\r\n"
|
||||||
"=============================================================================\r\n"
|
"=============================================================================\r\n"
|
||||||
"SIMtrace2 firmware " GIT_REVISION " (C) 2010-2017 by Harald Welte\r\n"
|
"SIMtrace2 firmware " GIT_REVISION " (C) 2010-2017 by Harald Welte\r\n"
|
||||||
"=============================================================================\r\n");
|
"=============================================================================\r\n");
|
||||||
|
|||||||
@@ -1,113 +1,113 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* \section Purpose
|
* \section Purpose
|
||||||
*
|
*
|
||||||
* Interface for configuration the Enhanced Embedded Flash Controller (EEFC) peripheral.
|
* Interface for configuration the Enhanced Embedded Flash Controller (EEFC) peripheral.
|
||||||
*
|
*
|
||||||
* \section Usage
|
* \section Usage
|
||||||
*
|
*
|
||||||
* -# Enable/disable %flash ready interrupt sources using EFC_EnableFrdyIt()
|
* -# Enable/disable %flash ready interrupt sources using EFC_EnableFrdyIt()
|
||||||
* and EFC_DisableFrdyIt().
|
* and EFC_DisableFrdyIt().
|
||||||
* -# Translates the given address into which EEFC, page and offset values
|
* -# Translates the given address into which EEFC, page and offset values
|
||||||
* for difference density %flash memory using EFC_TranslateAddress().
|
* for difference density %flash memory using EFC_TranslateAddress().
|
||||||
* -# Computes the address of a %flash access given the EFC, page and offset
|
* -# Computes the address of a %flash access given the EFC, page and offset
|
||||||
* for difference density %flash memory using EFC_ComputeAddress().
|
* for difference density %flash memory using EFC_ComputeAddress().
|
||||||
* -# Start the executing command with EFC_StartCommand()
|
* -# Start the executing command with EFC_StartCommand()
|
||||||
* -# Retrieve the current status of the EFC using EFC_GetStatus().
|
* -# Retrieve the current status of the EFC using EFC_GetStatus().
|
||||||
* -# Retrieve the result of the last executed command with EFC_GetResult().
|
* -# Retrieve the result of the last executed command with EFC_GetResult().
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _EEFC_
|
#ifndef _EEFC_
|
||||||
#define _EEFC_
|
#define _EEFC_
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Definitions
|
* Definitions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
/* EFC command */
|
/* EFC command */
|
||||||
#define EFC_FCMD_GETD 0x00
|
#define EFC_FCMD_GETD 0x00
|
||||||
#define EFC_FCMD_WP 0x01
|
#define EFC_FCMD_WP 0x01
|
||||||
#define EFC_FCMD_WPL 0x02
|
#define EFC_FCMD_WPL 0x02
|
||||||
#define EFC_FCMD_EWP 0x03
|
#define EFC_FCMD_EWP 0x03
|
||||||
#define EFC_FCMD_EWPL 0x04
|
#define EFC_FCMD_EWPL 0x04
|
||||||
#define EFC_FCMD_EA 0x05
|
#define EFC_FCMD_EA 0x05
|
||||||
#define EFC_FCMD_SLB 0x08
|
#define EFC_FCMD_SLB 0x08
|
||||||
#define EFC_FCMD_CLB 0x09
|
#define EFC_FCMD_CLB 0x09
|
||||||
#define EFC_FCMD_GLB 0x0A
|
#define EFC_FCMD_GLB 0x0A
|
||||||
#define EFC_FCMD_SFB 0x0B
|
#define EFC_FCMD_SFB 0x0B
|
||||||
#define EFC_FCMD_CFB 0x0C
|
#define EFC_FCMD_CFB 0x0C
|
||||||
#define EFC_FCMD_GFB 0x0D
|
#define EFC_FCMD_GFB 0x0D
|
||||||
#define EFC_FCMD_STUI 0x0E /* Start unique ID */
|
#define EFC_FCMD_STUI 0x0E /* Start unique ID */
|
||||||
#define EFC_FCMD_SPUI 0x0F /* Stop unique ID */
|
#define EFC_FCMD_SPUI 0x0F /* Stop unique ID */
|
||||||
|
|
||||||
/* The IAP function entry addreass */
|
/* The IAP function entry addreass */
|
||||||
#define CHIP_FLASH_IAP_ADDRESS (0x00800008)
|
#define CHIP_FLASH_IAP_ADDRESS (0x00800008)
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
extern void EFC_EnableFrdyIt( Efc* efc ) ;
|
extern void EFC_EnableFrdyIt( Efc* efc ) ;
|
||||||
|
|
||||||
extern void EFC_DisableFrdyIt( Efc* efc ) ;
|
extern void EFC_DisableFrdyIt( Efc* efc ) ;
|
||||||
|
|
||||||
extern void EFC_SetWaitState( Efc* efc, uint8_t cycles ) ;
|
extern void EFC_SetWaitState( Efc* efc, uint8_t cycles ) ;
|
||||||
|
|
||||||
extern void EFC_TranslateAddress( Efc** pEfc, uint32_t dwAddress, uint16_t *pwPage, uint16_t *pwOffset ) ;
|
extern void EFC_TranslateAddress( Efc** pEfc, uint32_t dwAddress, uint16_t *pwPage, uint16_t *pwOffset ) ;
|
||||||
|
|
||||||
extern void EFC_ComputeAddress( Efc* efc, uint16_t wPage, uint16_t wOffset, uint32_t *pdwAddress ) ;
|
extern void EFC_ComputeAddress( Efc* efc, uint16_t wPage, uint16_t wOffset, uint32_t *pdwAddress ) ;
|
||||||
|
|
||||||
extern void EFC_StartCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument ) ;
|
extern void EFC_StartCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument ) ;
|
||||||
|
|
||||||
extern uint32_t EFC_PerformCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument, uint32_t dwUseIAP ) ;
|
extern uint32_t EFC_PerformCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument, uint32_t dwUseIAP ) ;
|
||||||
|
|
||||||
extern uint32_t EFC_GetStatus( Efc* efc ) ;
|
extern uint32_t EFC_GetStatus( Efc* efc ) ;
|
||||||
|
|
||||||
extern uint32_t EFC_GetResult( Efc* efc ) ;
|
extern uint32_t EFC_GetResult( Efc* efc ) ;
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif /* #ifndef _EEFC_ */
|
#endif /* #ifndef _EEFC_ */
|
||||||
|
|
||||||
|
|||||||
@@ -1,79 +1,79 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* The flash driver provides the unified interface for flash program operations.
|
* The flash driver provides the unified interface for flash program operations.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _FLASHD_
|
#ifndef _FLASHD_
|
||||||
#define _FLASHD_
|
#define _FLASHD_
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
extern void FLASHD_Initialize( uint32_t dwMCk, uint32_t dwUseIAP ) ;
|
extern void FLASHD_Initialize( uint32_t dwMCk, uint32_t dwUseIAP ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_Erase( uint32_t dwAddress ) ;
|
extern uint32_t FLASHD_Erase( uint32_t dwAddress ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_Write( uint32_t dwAddress, const void *pvBuffer, uint32_t dwSize ) ;
|
extern uint32_t FLASHD_Write( uint32_t dwAddress, const void *pvBuffer, uint32_t dwSize ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_Lock( uint32_t dwStart, uint32_t dwEnd, uint32_t *pdwActualStart, uint32_t *pdwActualEnd ) ;
|
extern uint32_t FLASHD_Lock( uint32_t dwStart, uint32_t dwEnd, uint32_t *pdwActualStart, uint32_t *pdwActualEnd ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_Unlock( uint32_t dwStart, uint32_t dwEnd, uint32_t *pdwActualStart, uint32_t *pdwActualEnd ) ;
|
extern uint32_t FLASHD_Unlock( uint32_t dwStart, uint32_t dwEnd, uint32_t *pdwActualStart, uint32_t *pdwActualEnd ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_IsLocked( uint32_t dwStart, uint32_t dwEnd ) ;
|
extern uint32_t FLASHD_IsLocked( uint32_t dwStart, uint32_t dwEnd ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_SetGPNVM( uint8_t gpnvm ) ;
|
extern uint32_t FLASHD_SetGPNVM( uint8_t gpnvm ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_ClearGPNVM( uint8_t gpnvm ) ;
|
extern uint32_t FLASHD_ClearGPNVM( uint8_t gpnvm ) ;
|
||||||
|
|
||||||
extern uint32_t FLASHD_IsGPNVMSet( uint8_t gpnvm ) ;
|
extern uint32_t FLASHD_IsGPNVMSet( uint8_t gpnvm ) ;
|
||||||
|
|
||||||
#define FLASHD_IsSecurityBitSet() FLASHD_IsGPNVMSet( 0 )
|
#define FLASHD_IsSecurityBitSet() FLASHD_IsGPNVMSet( 0 )
|
||||||
|
|
||||||
#define FLASHD_SetSecurityBit() FLASHD_SetGPNVM( 0 )
|
#define FLASHD_SetSecurityBit() FLASHD_SetGPNVM( 0 )
|
||||||
|
|
||||||
extern uint32_t FLASHD_ReadUniqueID( uint32_t* pdwUniqueID ) ;
|
extern uint32_t FLASHD_ReadUniqueID( uint32_t* pdwUniqueID ) ;
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif /* #ifndef _FLASHD_ */
|
#endif /* #ifndef _FLASHD_ */
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,290 +1,290 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** \addtogroup efc_module Working with EEFC
|
/** \addtogroup efc_module Working with EEFC
|
||||||
* The EEFC driver provides the interface to configure and use the EEFC
|
* The EEFC driver provides the interface to configure and use the EEFC
|
||||||
* peripheral.
|
* peripheral.
|
||||||
*
|
*
|
||||||
* The user needs to set the number of wait states depending on the frequency used.\n
|
* The user needs to set the number of wait states depending on the frequency used.\n
|
||||||
* Configure number of cycles for flash read/write operations in the FWS field of EEFC_FMR.
|
* Configure number of cycles for flash read/write operations in the FWS field of EEFC_FMR.
|
||||||
*
|
*
|
||||||
* It offers a function to send flash command to EEFC and waits for the
|
* It offers a function to send flash command to EEFC and waits for the
|
||||||
* flash to be ready.
|
* flash to be ready.
|
||||||
*
|
*
|
||||||
* To send flash command, the user could do in either of following way:
|
* To send flash command, the user could do in either of following way:
|
||||||
* <ul>
|
* <ul>
|
||||||
* <li>Write a correct key, command and argument in EEFC_FCR. </li>
|
* <li>Write a correct key, command and argument in EEFC_FCR. </li>
|
||||||
* <li>Or, Use IAP (In Application Programming) function which is executed from
|
* <li>Or, Use IAP (In Application Programming) function which is executed from
|
||||||
* ROM directly, this allows flash programming to be done by code running in flash.</li>
|
* ROM directly, this allows flash programming to be done by code running in flash.</li>
|
||||||
* <li>Once the command is achieved, it can be detected even by polling EEFC_FSR or interrupt.
|
* <li>Once the command is achieved, it can be detected even by polling EEFC_FSR or interrupt.
|
||||||
* </ul>
|
* </ul>
|
||||||
*
|
*
|
||||||
* The command argument could be a page number,GPNVM number or nothing, it depends on
|
* The command argument could be a page number,GPNVM number or nothing, it depends on
|
||||||
* the command itself. Some useful functions in this driver could help user tranlate physical
|
* the command itself. Some useful functions in this driver could help user tranlate physical
|
||||||
* flash address into a page number and vice verse.
|
* flash address into a page number and vice verse.
|
||||||
*
|
*
|
||||||
* For more accurate information, please look at the EEFC section of the
|
* For more accurate information, please look at the EEFC section of the
|
||||||
* Datasheet.
|
* Datasheet.
|
||||||
*
|
*
|
||||||
* Related files :\n
|
* Related files :\n
|
||||||
* \ref efc.c\n
|
* \ref efc.c\n
|
||||||
* \ref efc.h.\n
|
* \ref efc.h.\n
|
||||||
*/
|
*/
|
||||||
/*@{*/
|
/*@{*/
|
||||||
/*@}*/
|
/*@}*/
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* Implementation of Enhanced Embedded Flash Controller (EEFC).
|
* Implementation of Enhanced Embedded Flash Controller (EEFC).
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
#include "efc.h"
|
#include "efc.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enables the flash ready interrupt source on the EEFC peripheral.
|
* \brief Enables the flash ready interrupt source on the EEFC peripheral.
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
*/
|
*/
|
||||||
extern void EFC_EnableFrdyIt( Efc* efc )
|
extern void EFC_EnableFrdyIt( Efc* efc )
|
||||||
{
|
{
|
||||||
efc->EEFC_FMR |= EEFC_FMR_FRDY ;
|
efc->EEFC_FMR |= EEFC_FMR_FRDY ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disables the flash ready interrupt source on the EEFC peripheral.
|
* \brief Disables the flash ready interrupt source on the EEFC peripheral.
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
*/
|
*/
|
||||||
|
|
||||||
extern void EFC_DisableFrdyIt( Efc* efc )
|
extern void EFC_DisableFrdyIt( Efc* efc )
|
||||||
{
|
{
|
||||||
efc->EEFC_FMR &= ~((uint32_t)EEFC_FMR_FRDY) ;
|
efc->EEFC_FMR &= ~((uint32_t)EEFC_FMR_FRDY) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Set read/write wait state on the EEFC perpherial.
|
* \brief Set read/write wait state on the EEFC perpherial.
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
* \param cycles the number of wait states in cycle.
|
* \param cycles the number of wait states in cycle.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
extern void EFC_SetWaitState( Efc* efc, uint8_t ucCycles )
|
extern void EFC_SetWaitState( Efc* efc, uint8_t ucCycles )
|
||||||
{
|
{
|
||||||
uint32_t dwValue ;
|
uint32_t dwValue ;
|
||||||
|
|
||||||
dwValue = efc->EEFC_FMR ;
|
dwValue = efc->EEFC_FMR ;
|
||||||
dwValue &= ~((uint32_t)EEFC_FMR_FWS_Msk) ;
|
dwValue &= ~((uint32_t)EEFC_FMR_FWS_Msk) ;
|
||||||
dwValue |= EEFC_FMR_FWS(ucCycles);
|
dwValue |= EEFC_FMR_FWS(ucCycles);
|
||||||
efc->EEFC_FMR = dwValue ;
|
efc->EEFC_FMR = dwValue ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Returns the current status of the EEFC.
|
* \brief Returns the current status of the EEFC.
|
||||||
*
|
*
|
||||||
* \note Keep in mind that this function clears the value of some status bits (LOCKE, PROGE).
|
* \note Keep in mind that this function clears the value of some status bits (LOCKE, PROGE).
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
*/
|
*/
|
||||||
extern uint32_t EFC_GetStatus( Efc* efc )
|
extern uint32_t EFC_GetStatus( Efc* efc )
|
||||||
{
|
{
|
||||||
return efc->EEFC_FSR ;
|
return efc->EEFC_FSR ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Returns the result of the last executed command.
|
* \brief Returns the result of the last executed command.
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
*/
|
*/
|
||||||
extern uint32_t EFC_GetResult( Efc* efc )
|
extern uint32_t EFC_GetResult( Efc* efc )
|
||||||
{
|
{
|
||||||
return efc->EEFC_FRR ;
|
return efc->EEFC_FRR ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Translates the given address page and offset values.
|
* \brief Translates the given address page and offset values.
|
||||||
* \note The resulting values are stored in the provided variables if they are not null.
|
* \note The resulting values are stored in the provided variables if they are not null.
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
* \param address Address to translate.
|
* \param address Address to translate.
|
||||||
* \param pPage First page accessed.
|
* \param pPage First page accessed.
|
||||||
* \param pOffset Byte offset in first page.
|
* \param pOffset Byte offset in first page.
|
||||||
*/
|
*/
|
||||||
extern void EFC_TranslateAddress( Efc** ppEfc, uint32_t dwAddress, uint16_t* pwPage, uint16_t* pwOffset )
|
extern void EFC_TranslateAddress( Efc** ppEfc, uint32_t dwAddress, uint16_t* pwPage, uint16_t* pwOffset )
|
||||||
{
|
{
|
||||||
Efc *pEfc ;
|
Efc *pEfc ;
|
||||||
uint16_t wPage ;
|
uint16_t wPage ;
|
||||||
uint16_t wOffset ;
|
uint16_t wOffset ;
|
||||||
|
|
||||||
assert( dwAddress >= IFLASH_ADDR ) ;
|
assert( dwAddress >= IFLASH_ADDR ) ;
|
||||||
assert( dwAddress <= (IFLASH_ADDR + IFLASH_SIZE) ) ;
|
assert( dwAddress <= (IFLASH_ADDR + IFLASH_SIZE) ) ;
|
||||||
|
|
||||||
pEfc = EFC ;
|
pEfc = EFC ;
|
||||||
wPage = (dwAddress - IFLASH_ADDR) / IFLASH_PAGE_SIZE;
|
wPage = (dwAddress - IFLASH_ADDR) / IFLASH_PAGE_SIZE;
|
||||||
wOffset = (dwAddress - IFLASH_ADDR) % IFLASH_PAGE_SIZE;
|
wOffset = (dwAddress - IFLASH_ADDR) % IFLASH_PAGE_SIZE;
|
||||||
|
|
||||||
TRACE_DEBUG( "Translated 0x%08X to page=%d and offset=%d\n\r", dwAddress, wPage, wOffset ) ;
|
TRACE_DEBUG( "Translated 0x%08X to page=%d and offset=%d\n\r", dwAddress, wPage, wOffset ) ;
|
||||||
/* Store values */
|
/* Store values */
|
||||||
if ( pEfc )
|
if ( pEfc )
|
||||||
{
|
{
|
||||||
*ppEfc = pEfc ;
|
*ppEfc = pEfc ;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( pwPage )
|
if ( pwPage )
|
||||||
{
|
{
|
||||||
*pwPage = wPage ;
|
*pwPage = wPage ;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( pwOffset )
|
if ( pwOffset )
|
||||||
{
|
{
|
||||||
*pwOffset = wOffset ;
|
*pwOffset = wOffset ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Computes the address of a flash access given the page and offset.
|
* \brief Computes the address of a flash access given the page and offset.
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
* \param page Page number.
|
* \param page Page number.
|
||||||
* \param offset Byte offset inside page.
|
* \param offset Byte offset inside page.
|
||||||
* \param pAddress Computed address (optional).
|
* \param pAddress Computed address (optional).
|
||||||
*/
|
*/
|
||||||
extern void EFC_ComputeAddress( Efc *efc, uint16_t wPage, uint16_t wOffset, uint32_t *pdwAddress )
|
extern void EFC_ComputeAddress( Efc *efc, uint16_t wPage, uint16_t wOffset, uint32_t *pdwAddress )
|
||||||
{
|
{
|
||||||
uint32_t dwAddress ;
|
uint32_t dwAddress ;
|
||||||
|
|
||||||
assert( efc ) ;
|
assert( efc ) ;
|
||||||
assert( wPage <= IFLASH_NB_OF_PAGES ) ;
|
assert( wPage <= IFLASH_NB_OF_PAGES ) ;
|
||||||
assert( wOffset < IFLASH_PAGE_SIZE ) ;
|
assert( wOffset < IFLASH_PAGE_SIZE ) ;
|
||||||
|
|
||||||
/* Compute address */
|
/* Compute address */
|
||||||
dwAddress = IFLASH_ADDR + wPage * IFLASH_PAGE_SIZE + wOffset ;
|
dwAddress = IFLASH_ADDR + wPage * IFLASH_PAGE_SIZE + wOffset ;
|
||||||
|
|
||||||
/* Store result */
|
/* Store result */
|
||||||
if ( pdwAddress != NULL )
|
if ( pdwAddress != NULL )
|
||||||
{
|
{
|
||||||
*pdwAddress = dwAddress ;
|
*pdwAddress = dwAddress ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Starts the executing the given command on the EEFC and returns as soon as the command is started.
|
* \brief Starts the executing the given command on the EEFC and returns as soon as the command is started.
|
||||||
*
|
*
|
||||||
* \note It does NOT set the FMCN field automatically.
|
* \note It does NOT set the FMCN field automatically.
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
* \param command Command to execute.
|
* \param command Command to execute.
|
||||||
* \param argument Command argument (should be 0 if not used).
|
* \param argument Command argument (should be 0 if not used).
|
||||||
*/
|
*/
|
||||||
extern void EFC_StartCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument )
|
extern void EFC_StartCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument )
|
||||||
{
|
{
|
||||||
/* Check command & argument */
|
/* Check command & argument */
|
||||||
switch ( dwCommand )
|
switch ( dwCommand )
|
||||||
{
|
{
|
||||||
case EFC_FCMD_WP:
|
case EFC_FCMD_WP:
|
||||||
case EFC_FCMD_WPL:
|
case EFC_FCMD_WPL:
|
||||||
case EFC_FCMD_EWP:
|
case EFC_FCMD_EWP:
|
||||||
case EFC_FCMD_EWPL:
|
case EFC_FCMD_EWPL:
|
||||||
case EFC_FCMD_SLB:
|
case EFC_FCMD_SLB:
|
||||||
case EFC_FCMD_CLB:
|
case EFC_FCMD_CLB:
|
||||||
assert( dwArgument < IFLASH_NB_OF_PAGES ) ;
|
assert( dwArgument < IFLASH_NB_OF_PAGES ) ;
|
||||||
break ;
|
break ;
|
||||||
|
|
||||||
case EFC_FCMD_SFB:
|
case EFC_FCMD_SFB:
|
||||||
case EFC_FCMD_CFB:
|
case EFC_FCMD_CFB:
|
||||||
assert( dwArgument < 2 ) ;
|
assert( dwArgument < 2 ) ;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EFC_FCMD_GETD:
|
case EFC_FCMD_GETD:
|
||||||
case EFC_FCMD_EA:
|
case EFC_FCMD_EA:
|
||||||
case EFC_FCMD_GLB:
|
case EFC_FCMD_GLB:
|
||||||
case EFC_FCMD_GFB:
|
case EFC_FCMD_GFB:
|
||||||
case EFC_FCMD_STUI:
|
case EFC_FCMD_STUI:
|
||||||
assert( dwArgument == 0 ) ;
|
assert( dwArgument == 0 ) ;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default: assert( 0 ) ;
|
default: assert( 0 ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Start command Embedded flash */
|
/* Start command Embedded flash */
|
||||||
assert( (efc->EEFC_FSR & EEFC_FMR_FRDY) == EEFC_FMR_FRDY ) ;
|
assert( (efc->EEFC_FSR & EEFC_FMR_FRDY) == EEFC_FMR_FRDY ) ;
|
||||||
efc->EEFC_FCR = EEFC_FCR_FKEY(0x5A) | EEFC_FCR_FARG(dwArgument) | EEFC_FCR_FCMD(dwCommand) ;
|
efc->EEFC_FCR = EEFC_FCR_FKEY(0x5A) | EEFC_FCR_FARG(dwArgument) | EEFC_FCR_FCMD(dwCommand) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Performs the given command and wait until its completion (or an error).
|
* \brief Performs the given command and wait until its completion (or an error).
|
||||||
*
|
*
|
||||||
* \param efc Pointer to a Efc instance
|
* \param efc Pointer to a Efc instance
|
||||||
* \param command Command to perform.
|
* \param command Command to perform.
|
||||||
* \param argument Optional command argument.
|
* \param argument Optional command argument.
|
||||||
*
|
*
|
||||||
* \return 0 if successful, otherwise returns an error code.
|
* \return 0 if successful, otherwise returns an error code.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
extern uint32_t EFC_PerformCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument, uint32_t dwUseIAP )
|
extern uint32_t EFC_PerformCommand( Efc* efc, uint32_t dwCommand, uint32_t dwArgument, uint32_t dwUseIAP )
|
||||||
{
|
{
|
||||||
if ( dwUseIAP != 0 )
|
if ( dwUseIAP != 0 )
|
||||||
{
|
{
|
||||||
/* Pointer on IAP function in ROM */
|
/* Pointer on IAP function in ROM */
|
||||||
static uint32_t (*IAP_PerformCommand)( uint32_t, uint32_t ) ;
|
static uint32_t (*IAP_PerformCommand)( uint32_t, uint32_t ) ;
|
||||||
|
|
||||||
IAP_PerformCommand = (uint32_t (*)( uint32_t, uint32_t )) *((uint32_t*)CHIP_FLASH_IAP_ADDRESS ) ;
|
IAP_PerformCommand = (uint32_t (*)( uint32_t, uint32_t )) *((uint32_t*)CHIP_FLASH_IAP_ADDRESS ) ;
|
||||||
IAP_PerformCommand( 0, EEFC_FCR_FKEY(0x5A) | EEFC_FCR_FARG(dwArgument) | EEFC_FCR_FCMD(dwCommand) ) ;
|
IAP_PerformCommand( 0, EEFC_FCR_FKEY(0x5A) | EEFC_FCR_FARG(dwArgument) | EEFC_FCR_FCMD(dwCommand) ) ;
|
||||||
|
|
||||||
return (efc->EEFC_FSR & (EEFC_FSR_FLOCKE | EEFC_FSR_FCMDE)) ;
|
return (efc->EEFC_FSR & (EEFC_FSR_FLOCKE | EEFC_FSR_FCMDE)) ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
uint32_t dwStatus ;
|
uint32_t dwStatus ;
|
||||||
|
|
||||||
efc->EEFC_FCR = EEFC_FCR_FKEY(0x5A) | EEFC_FCR_FARG(dwArgument) | EEFC_FCR_FCMD(dwCommand) ;
|
efc->EEFC_FCR = EEFC_FCR_FKEY(0x5A) | EEFC_FCR_FARG(dwArgument) | EEFC_FCR_FCMD(dwCommand) ;
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
dwStatus = efc->EEFC_FSR ;
|
dwStatus = efc->EEFC_FSR ;
|
||||||
}
|
}
|
||||||
while ( (dwStatus & EEFC_FSR_FRDY) != EEFC_FSR_FRDY ) ;
|
while ( (dwStatus & EEFC_FSR_FRDY) != EEFC_FSR_FRDY ) ;
|
||||||
|
|
||||||
return ( dwStatus & (EEFC_FSR_FLOCKE | EEFC_FSR_FCMDE) ) ;
|
return ( dwStatus & (EEFC_FSR_FLOCKE | EEFC_FSR_FCMDE) ) ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1,453 +1,453 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2010, Atmel Corporation
|
* Copyright (c) 2010, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** \file */
|
/** \file */
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
#include "pio.h"
|
#include "pio.h"
|
||||||
#include "pmc.h"
|
#include "pmc.h"
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local functions
|
* Local functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
||||||
* peripheral A. Optionally, the corresponding internal pull-up(s) can be enabled.
|
* peripheral A. Optionally, the corresponding internal pull-up(s) can be enabled.
|
||||||
*
|
*
|
||||||
* \param pio Pointer to a PIO controller.
|
* \param pio Pointer to a PIO controller.
|
||||||
* \param mask Bitmask of one or more pin(s) to configure.
|
* \param mask Bitmask of one or more pin(s) to configure.
|
||||||
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
||||||
* configured.
|
* configured.
|
||||||
*/
|
*/
|
||||||
static void PIO_SetPeripheralA(
|
static void PIO_SetPeripheralA(
|
||||||
Pio *pio,
|
Pio *pio,
|
||||||
unsigned int mask,
|
unsigned int mask,
|
||||||
unsigned char enablePullUp)
|
unsigned char enablePullUp)
|
||||||
{
|
{
|
||||||
unsigned int abcdsr;
|
unsigned int abcdsr;
|
||||||
/* Disable interrupts on the pin(s) */
|
/* Disable interrupts on the pin(s) */
|
||||||
pio->PIO_IDR = mask;
|
pio->PIO_IDR = mask;
|
||||||
|
|
||||||
/* Enable the pull-up(s) if necessary */
|
/* Enable the pull-up(s) if necessary */
|
||||||
if (enablePullUp) {
|
if (enablePullUp) {
|
||||||
pio->PIO_PUER = mask;
|
pio->PIO_PUER = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_PUDR = mask;
|
pio->PIO_PUDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
abcdsr = pio->PIO_ABCDSR[0];
|
abcdsr = pio->PIO_ABCDSR[0];
|
||||||
pio->PIO_ABCDSR[0] &= (~mask & abcdsr);
|
pio->PIO_ABCDSR[0] &= (~mask & abcdsr);
|
||||||
abcdsr = pio->PIO_ABCDSR[1];
|
abcdsr = pio->PIO_ABCDSR[1];
|
||||||
pio->PIO_ABCDSR[1] &= (~mask & abcdsr);
|
pio->PIO_ABCDSR[1] &= (~mask & abcdsr);
|
||||||
pio->PIO_PDR = mask;
|
pio->PIO_PDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
||||||
* peripheral B. Optionally, the corresponding internal pull-up(s) can be enabled.
|
* peripheral B. Optionally, the corresponding internal pull-up(s) can be enabled.
|
||||||
*
|
*
|
||||||
* \param pio Pointer to a PIO controller.
|
* \param pio Pointer to a PIO controller.
|
||||||
* \param mask Bitmask of one or more pin(s) to configure.
|
* \param mask Bitmask of one or more pin(s) to configure.
|
||||||
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
||||||
* configured.
|
* configured.
|
||||||
*/
|
*/
|
||||||
static void PIO_SetPeripheralB(
|
static void PIO_SetPeripheralB(
|
||||||
Pio *pio,
|
Pio *pio,
|
||||||
unsigned int mask,
|
unsigned int mask,
|
||||||
unsigned char enablePullUp)
|
unsigned char enablePullUp)
|
||||||
{
|
{
|
||||||
unsigned int abcdsr;
|
unsigned int abcdsr;
|
||||||
/* Disable interrupts on the pin(s) */
|
/* Disable interrupts on the pin(s) */
|
||||||
pio->PIO_IDR = mask;
|
pio->PIO_IDR = mask;
|
||||||
|
|
||||||
/* Enable the pull-up(s) if necessary */
|
/* Enable the pull-up(s) if necessary */
|
||||||
if (enablePullUp) {
|
if (enablePullUp) {
|
||||||
|
|
||||||
pio->PIO_PUER = mask;
|
pio->PIO_PUER = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_PUDR = mask;
|
pio->PIO_PUDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
abcdsr = pio->PIO_ABCDSR[0];
|
abcdsr = pio->PIO_ABCDSR[0];
|
||||||
pio->PIO_ABCDSR[0] = (mask | abcdsr);
|
pio->PIO_ABCDSR[0] = (mask | abcdsr);
|
||||||
abcdsr = pio->PIO_ABCDSR[1];
|
abcdsr = pio->PIO_ABCDSR[1];
|
||||||
pio->PIO_ABCDSR[1] &= (~mask & abcdsr);
|
pio->PIO_ABCDSR[1] &= (~mask & abcdsr);
|
||||||
|
|
||||||
pio->PIO_PDR = mask;
|
pio->PIO_PDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
||||||
* peripheral C. Optionally, the corresponding internal pull-up(s) can be enabled.
|
* peripheral C. Optionally, the corresponding internal pull-up(s) can be enabled.
|
||||||
*
|
*
|
||||||
* \param pio Pointer to a PIO controller.
|
* \param pio Pointer to a PIO controller.
|
||||||
* \param mask Bitmask of one or more pin(s) to configure.
|
* \param mask Bitmask of one or more pin(s) to configure.
|
||||||
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
||||||
* configured.
|
* configured.
|
||||||
*/
|
*/
|
||||||
static void PIO_SetPeripheralC(
|
static void PIO_SetPeripheralC(
|
||||||
Pio *pio,
|
Pio *pio,
|
||||||
unsigned int mask,
|
unsigned int mask,
|
||||||
unsigned char enablePullUp)
|
unsigned char enablePullUp)
|
||||||
{
|
{
|
||||||
unsigned int abcdsr;
|
unsigned int abcdsr;
|
||||||
/* Disable interrupts on the pin(s) */
|
/* Disable interrupts on the pin(s) */
|
||||||
pio->PIO_IDR = mask;
|
pio->PIO_IDR = mask;
|
||||||
|
|
||||||
/* Enable the pull-up(s) if necessary */
|
/* Enable the pull-up(s) if necessary */
|
||||||
if (enablePullUp) {
|
if (enablePullUp) {
|
||||||
|
|
||||||
pio->PIO_PUER = mask;
|
pio->PIO_PUER = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_PUDR = mask;
|
pio->PIO_PUDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
abcdsr = pio->PIO_ABCDSR[0];
|
abcdsr = pio->PIO_ABCDSR[0];
|
||||||
pio->PIO_ABCDSR[0] &= (~mask & abcdsr);
|
pio->PIO_ABCDSR[0] &= (~mask & abcdsr);
|
||||||
abcdsr = pio->PIO_ABCDSR[1];
|
abcdsr = pio->PIO_ABCDSR[1];
|
||||||
pio->PIO_ABCDSR[1] = (mask | abcdsr);
|
pio->PIO_ABCDSR[1] = (mask | abcdsr);
|
||||||
|
|
||||||
pio->PIO_PDR = mask;
|
pio->PIO_PDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
* \brief Configures one or more pin(s) of a PIO controller as being controlled by
|
||||||
* peripheral D. Optionally, the corresponding internal pull-up(s) can be enabled.
|
* peripheral D. Optionally, the corresponding internal pull-up(s) can be enabled.
|
||||||
*
|
*
|
||||||
* \param pio Pointer to a PIO controller.
|
* \param pio Pointer to a PIO controller.
|
||||||
* \param mask Bitmask of one or more pin(s) to configure.
|
* \param mask Bitmask of one or more pin(s) to configure.
|
||||||
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
* \param enablePullUp Indicates if the pin(s) internal pull-up shall be
|
||||||
* configured.
|
* configured.
|
||||||
*/
|
*/
|
||||||
static void PIO_SetPeripheralD(
|
static void PIO_SetPeripheralD(
|
||||||
Pio *pio,
|
Pio *pio,
|
||||||
unsigned int mask,
|
unsigned int mask,
|
||||||
unsigned char enablePullUp)
|
unsigned char enablePullUp)
|
||||||
{
|
{
|
||||||
unsigned int abcdsr;
|
unsigned int abcdsr;
|
||||||
/* Disable interrupts on the pin(s) */
|
/* Disable interrupts on the pin(s) */
|
||||||
pio->PIO_IDR = mask;
|
pio->PIO_IDR = mask;
|
||||||
|
|
||||||
/* Enable the pull-up(s) if necessary */
|
/* Enable the pull-up(s) if necessary */
|
||||||
if (enablePullUp) {
|
if (enablePullUp) {
|
||||||
|
|
||||||
pio->PIO_PUER = mask;
|
pio->PIO_PUER = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_PUDR = mask;
|
pio->PIO_PUDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
abcdsr = pio->PIO_ABCDSR[0];
|
abcdsr = pio->PIO_ABCDSR[0];
|
||||||
pio->PIO_ABCDSR[0] = (mask | abcdsr);
|
pio->PIO_ABCDSR[0] = (mask | abcdsr);
|
||||||
abcdsr = pio->PIO_ABCDSR[1];
|
abcdsr = pio->PIO_ABCDSR[1];
|
||||||
pio->PIO_ABCDSR[1] = (mask | abcdsr);
|
pio->PIO_ABCDSR[1] = (mask | abcdsr);
|
||||||
|
|
||||||
pio->PIO_PDR = mask;
|
pio->PIO_PDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures one or more pin(s) or a PIO controller as inputs. Optionally,
|
* \brief Configures one or more pin(s) or a PIO controller as inputs. Optionally,
|
||||||
* the corresponding internal pull-up(s) and glitch filter(s) can be enabled.
|
* the corresponding internal pull-up(s) and glitch filter(s) can be enabled.
|
||||||
*
|
*
|
||||||
* \param pio Pointer to a PIO controller.
|
* \param pio Pointer to a PIO controller.
|
||||||
* \param mask Bitmask indicating which pin(s) to configure as input(s).
|
* \param mask Bitmask indicating which pin(s) to configure as input(s).
|
||||||
* \param enablePullUp Indicates if the internal pull-up(s) must be enabled.
|
* \param enablePullUp Indicates if the internal pull-up(s) must be enabled.
|
||||||
* \param enableFilter Indicates if the glitch filter(s) must be enabled.
|
* \param enableFilter Indicates if the glitch filter(s) must be enabled.
|
||||||
*/
|
*/
|
||||||
static void PIO_SetInput(
|
static void PIO_SetInput(
|
||||||
Pio *pio,
|
Pio *pio,
|
||||||
unsigned int mask,
|
unsigned int mask,
|
||||||
unsigned char attribute)
|
unsigned char attribute)
|
||||||
{
|
{
|
||||||
/* Disable interrupts */
|
/* Disable interrupts */
|
||||||
pio->PIO_IDR = mask;
|
pio->PIO_IDR = mask;
|
||||||
|
|
||||||
/* Enable pull-up(s) if necessary */
|
/* Enable pull-up(s) if necessary */
|
||||||
if (attribute & PIO_PULLUP)
|
if (attribute & PIO_PULLUP)
|
||||||
pio->PIO_PUER = mask;
|
pio->PIO_PUER = mask;
|
||||||
else
|
else
|
||||||
pio->PIO_PUDR = mask;
|
pio->PIO_PUDR = mask;
|
||||||
|
|
||||||
/* Enable Input Filter if necessary */
|
/* Enable Input Filter if necessary */
|
||||||
if (attribute & (PIO_DEGLITCH | PIO_DEBOUNCE))
|
if (attribute & (PIO_DEGLITCH | PIO_DEBOUNCE))
|
||||||
pio->PIO_IFER = mask;
|
pio->PIO_IFER = mask;
|
||||||
else
|
else
|
||||||
pio->PIO_IFDR = mask;
|
pio->PIO_IFDR = mask;
|
||||||
|
|
||||||
/* Enable de-glitch or de-bounce if necessary */
|
/* Enable de-glitch or de-bounce if necessary */
|
||||||
if (attribute & PIO_DEGLITCH)
|
if (attribute & PIO_DEGLITCH)
|
||||||
{
|
{
|
||||||
pio->PIO_IFSCDR = mask;
|
pio->PIO_IFSCDR = mask;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (attribute & PIO_DEBOUNCE)
|
if (attribute & PIO_DEBOUNCE)
|
||||||
{
|
{
|
||||||
pio->PIO_IFSCER = mask;
|
pio->PIO_IFSCER = mask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Configure pin as input */
|
/* Configure pin as input */
|
||||||
pio->PIO_ODR = mask;
|
pio->PIO_ODR = mask;
|
||||||
pio->PIO_PER = mask;
|
pio->PIO_PER = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures one or more pin(s) of a PIO controller as outputs, with the
|
* \brief Configures one or more pin(s) of a PIO controller as outputs, with the
|
||||||
* given default value. Optionally, the multi-drive feature can be enabled
|
* given default value. Optionally, the multi-drive feature can be enabled
|
||||||
* on the pin(s).
|
* on the pin(s).
|
||||||
*
|
*
|
||||||
* \param pio Pointer to a PIO controller.
|
* \param pio Pointer to a PIO controller.
|
||||||
* \param mask Bitmask indicating which pin(s) to configure.
|
* \param mask Bitmask indicating which pin(s) to configure.
|
||||||
* \param defaultValue Default level on the pin(s).
|
* \param defaultValue Default level on the pin(s).
|
||||||
* \param enableMultiDrive Indicates if the pin(s) shall be configured as
|
* \param enableMultiDrive Indicates if the pin(s) shall be configured as
|
||||||
* open-drain.
|
* open-drain.
|
||||||
* \param enablePullUp Indicates if the pin shall have its pull-up activated.
|
* \param enablePullUp Indicates if the pin shall have its pull-up activated.
|
||||||
*/
|
*/
|
||||||
static void PIO_SetOutput(
|
static void PIO_SetOutput(
|
||||||
Pio *pio,
|
Pio *pio,
|
||||||
unsigned int mask,
|
unsigned int mask,
|
||||||
unsigned char defaultValue,
|
unsigned char defaultValue,
|
||||||
unsigned char enableMultiDrive,
|
unsigned char enableMultiDrive,
|
||||||
unsigned char enablePullUp)
|
unsigned char enablePullUp)
|
||||||
{
|
{
|
||||||
/* Disable interrupts */
|
/* Disable interrupts */
|
||||||
pio->PIO_IDR = mask;
|
pio->PIO_IDR = mask;
|
||||||
|
|
||||||
/* Enable pull-up(s) if necessary */
|
/* Enable pull-up(s) if necessary */
|
||||||
if (enablePullUp) {
|
if (enablePullUp) {
|
||||||
|
|
||||||
pio->PIO_PUER = mask;
|
pio->PIO_PUER = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_PUDR = mask;
|
pio->PIO_PUDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Enable multi-drive if necessary */
|
/* Enable multi-drive if necessary */
|
||||||
if (enableMultiDrive) {
|
if (enableMultiDrive) {
|
||||||
|
|
||||||
pio->PIO_MDER = mask;
|
pio->PIO_MDER = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_MDDR = mask;
|
pio->PIO_MDDR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set default value */
|
/* Set default value */
|
||||||
if (defaultValue) {
|
if (defaultValue) {
|
||||||
|
|
||||||
pio->PIO_SODR = mask;
|
pio->PIO_SODR = mask;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
pio->PIO_CODR = mask;
|
pio->PIO_CODR = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Configure pin(s) as output(s) */
|
/* Configure pin(s) as output(s) */
|
||||||
pio->PIO_OER = mask;
|
pio->PIO_OER = mask;
|
||||||
pio->PIO_PER = mask;
|
pio->PIO_PER = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Global functions
|
* Global functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures a list of Pin instances, each of which can either hold a single
|
* \brief Configures a list of Pin instances, each of which can either hold a single
|
||||||
* pin or a group of pins, depending on the mask value; all pins are configured
|
* pin or a group of pins, depending on the mask value; all pins are configured
|
||||||
* by this function. The size of the array must also be provided and is easily
|
* by this function. The size of the array must also be provided and is easily
|
||||||
* computed using PIO_LISTSIZE whenever its length is not known in advance.
|
* computed using PIO_LISTSIZE whenever its length is not known in advance.
|
||||||
*
|
*
|
||||||
* \param list Pointer to a list of Pin instances.
|
* \param list Pointer to a list of Pin instances.
|
||||||
* \param size Size of the Pin list (calculated using PIO_LISTSIZE).
|
* \param size Size of the Pin list (calculated using PIO_LISTSIZE).
|
||||||
*
|
*
|
||||||
* \return 1 if the pins have been configured properly; otherwise 0.
|
* \return 1 if the pins have been configured properly; otherwise 0.
|
||||||
*/
|
*/
|
||||||
uint8_t PIO_Configure( const Pin *list, uint32_t size )
|
uint8_t PIO_Configure( const Pin *list, uint32_t size )
|
||||||
{
|
{
|
||||||
/* Configure pins */
|
/* Configure pins */
|
||||||
while ( size > 0 )
|
while ( size > 0 )
|
||||||
{
|
{
|
||||||
switch ( list->type )
|
switch ( list->type )
|
||||||
{
|
{
|
||||||
|
|
||||||
case PIO_PERIPH_A:
|
case PIO_PERIPH_A:
|
||||||
PIO_SetPeripheralA(list->pio,
|
PIO_SetPeripheralA(list->pio,
|
||||||
list->mask,
|
list->mask,
|
||||||
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PIO_PERIPH_B:
|
case PIO_PERIPH_B:
|
||||||
PIO_SetPeripheralB(list->pio,
|
PIO_SetPeripheralB(list->pio,
|
||||||
list->mask,
|
list->mask,
|
||||||
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PIO_PERIPH_C:
|
case PIO_PERIPH_C:
|
||||||
PIO_SetPeripheralC(list->pio,
|
PIO_SetPeripheralC(list->pio,
|
||||||
list->mask,
|
list->mask,
|
||||||
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PIO_PERIPH_D:
|
case PIO_PERIPH_D:
|
||||||
PIO_SetPeripheralD(list->pio,
|
PIO_SetPeripheralD(list->pio,
|
||||||
list->mask,
|
list->mask,
|
||||||
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
||||||
break;
|
break;
|
||||||
case PIO_INPUT:
|
case PIO_INPUT:
|
||||||
PMC_EnablePeripheral(list->id);
|
PMC_EnablePeripheral(list->id);
|
||||||
PIO_SetInput(list->pio,
|
PIO_SetInput(list->pio,
|
||||||
list->mask,
|
list->mask,
|
||||||
list->attribute);
|
list->attribute);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PIO_OUTPUT_0:
|
case PIO_OUTPUT_0:
|
||||||
case PIO_OUTPUT_1:
|
case PIO_OUTPUT_1:
|
||||||
PIO_SetOutput(list->pio,
|
PIO_SetOutput(list->pio,
|
||||||
list->mask,
|
list->mask,
|
||||||
(list->type == PIO_OUTPUT_1),
|
(list->type == PIO_OUTPUT_1),
|
||||||
(list->attribute & PIO_OPENDRAIN) ? 1 : 0,
|
(list->attribute & PIO_OPENDRAIN) ? 1 : 0,
|
||||||
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
(list->attribute & PIO_PULLUP) ? 1 : 0);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default: return 0;
|
default: return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
list++;
|
list++;
|
||||||
size--;
|
size--;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sets a high output level on all the PIOs defined in the given Pin instance.
|
* \brief Sets a high output level on all the PIOs defined in the given Pin instance.
|
||||||
* This has no immediate effects on PIOs that are not output, but the PIO
|
* This has no immediate effects on PIOs that are not output, but the PIO
|
||||||
* controller will memorize the value they are changed to outputs.
|
* controller will memorize the value they are changed to outputs.
|
||||||
*
|
*
|
||||||
* \param pin Pointer to a Pin instance describing one or more pins.
|
* \param pin Pointer to a Pin instance describing one or more pins.
|
||||||
*/
|
*/
|
||||||
void PIO_Set(const Pin *pin)
|
void PIO_Set(const Pin *pin)
|
||||||
{
|
{
|
||||||
pin->pio->PIO_SODR = pin->mask;
|
pin->pio->PIO_SODR = pin->mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sets a low output level on all the PIOs defined in the given Pin instance.
|
* \brief Sets a low output level on all the PIOs defined in the given Pin instance.
|
||||||
* This has no immediate effects on PIOs that are not output, but the PIO
|
* This has no immediate effects on PIOs that are not output, but the PIO
|
||||||
* controller will memorize the value they are changed to outputs.
|
* controller will memorize the value they are changed to outputs.
|
||||||
*
|
*
|
||||||
* \param pin Pointer to a Pin instance describing one or more pins.
|
* \param pin Pointer to a Pin instance describing one or more pins.
|
||||||
*/
|
*/
|
||||||
void PIO_Clear(const Pin *pin)
|
void PIO_Clear(const Pin *pin)
|
||||||
{
|
{
|
||||||
pin->pio->PIO_CODR = pin->mask;
|
pin->pio->PIO_CODR = pin->mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Returns 1 if one or more PIO of the given Pin instance currently have
|
* \brief Returns 1 if one or more PIO of the given Pin instance currently have
|
||||||
* a high level; otherwise returns 0. This method returns the actual value that
|
* a high level; otherwise returns 0. This method returns the actual value that
|
||||||
* is being read on the pin. To return the supposed output value of a pin, use
|
* is being read on the pin. To return the supposed output value of a pin, use
|
||||||
* PIO_GetOutputDataStatus() instead.
|
* PIO_GetOutputDataStatus() instead.
|
||||||
*
|
*
|
||||||
* \param pin Pointer to a Pin instance describing one or more pins.
|
* \param pin Pointer to a Pin instance describing one or more pins.
|
||||||
*
|
*
|
||||||
* \return 1 if the Pin instance contains at least one PIO that currently has
|
* \return 1 if the Pin instance contains at least one PIO that currently has
|
||||||
* a high level; otherwise 0.
|
* a high level; otherwise 0.
|
||||||
*/
|
*/
|
||||||
unsigned char PIO_Get( const Pin *pin )
|
unsigned char PIO_Get( const Pin *pin )
|
||||||
{
|
{
|
||||||
unsigned int reg ;
|
unsigned int reg ;
|
||||||
|
|
||||||
if ( (pin->type == PIO_OUTPUT_0) || (pin->type == PIO_OUTPUT_1) )
|
if ( (pin->type == PIO_OUTPUT_0) || (pin->type == PIO_OUTPUT_1) )
|
||||||
{
|
{
|
||||||
reg = pin->pio->PIO_ODSR ;
|
reg = pin->pio->PIO_ODSR ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
reg = pin->pio->PIO_PDSR ;
|
reg = pin->pio->PIO_PDSR ;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( (reg & pin->mask) == 0 )
|
if ( (reg & pin->mask) == 0 )
|
||||||
{
|
{
|
||||||
return 0 ;
|
return 0 ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
return 1 ;
|
return 1 ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Returns 1 if one or more PIO of the given Pin are configured to output a
|
* \brief Returns 1 if one or more PIO of the given Pin are configured to output a
|
||||||
* high level (even if they are not output).
|
* high level (even if they are not output).
|
||||||
* To get the actual value of the pin, use PIO_Get() instead.
|
* To get the actual value of the pin, use PIO_Get() instead.
|
||||||
*
|
*
|
||||||
* \param pin Pointer to a Pin instance describing one or more pins.
|
* \param pin Pointer to a Pin instance describing one or more pins.
|
||||||
*
|
*
|
||||||
* \return 1 if the Pin instance contains at least one PIO that is configured
|
* \return 1 if the Pin instance contains at least one PIO that is configured
|
||||||
* to output a high level; otherwise 0.
|
* to output a high level; otherwise 0.
|
||||||
*/
|
*/
|
||||||
unsigned char PIO_GetOutputDataStatus(const Pin *pin)
|
unsigned char PIO_GetOutputDataStatus(const Pin *pin)
|
||||||
{
|
{
|
||||||
if ((pin->pio->PIO_ODSR & pin->mask) == 0) {
|
if ((pin->pio->PIO_ODSR & pin->mask) == 0) {
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* \brief Configures Glitch or Debouncing filter for input.
|
* \brief Configures Glitch or Debouncing filter for input.
|
||||||
*
|
*
|
||||||
* \param pin Pointer to a Pin instance describing one or more pins.
|
* \param pin Pointer to a Pin instance describing one or more pins.
|
||||||
* \param cuttoff Cutt off frequency for debounce filter.
|
* \param cuttoff Cutt off frequency for debounce filter.
|
||||||
*/
|
*/
|
||||||
void PIO_SetDebounceFilter( const Pin *pin, uint32_t cuttoff )
|
void PIO_SetDebounceFilter( const Pin *pin, uint32_t cuttoff )
|
||||||
{
|
{
|
||||||
Pio *pio = pin->pio;
|
Pio *pio = pin->pio;
|
||||||
|
|
||||||
pio->PIO_IFSCER = pin->mask; /* set Debouncing, 0 bit field no effect */
|
pio->PIO_IFSCER = pin->mask; /* set Debouncing, 0 bit field no effect */
|
||||||
pio->PIO_SCDR = ((32678/(2*(cuttoff))) - 1) & 0x3FFF; /* the lowest 14 bits work */
|
pio->PIO_SCDR = ((32678/(2*(cuttoff))) - 1) & 0x3FFF; /* the lowest 14 bits work */
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,315 +1,315 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2008, Atmel Corporation
|
* Copyright (c) 2008, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* \file
|
* \file
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local definitions
|
* Local definitions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/* Maximum number of interrupt sources that can be defined. This
|
/* Maximum number of interrupt sources that can be defined. This
|
||||||
* constant can be increased, but the current value is the smallest possible
|
* constant can be increased, but the current value is the smallest possible
|
||||||
* that will be compatible with all existing projects. */
|
* that will be compatible with all existing projects. */
|
||||||
#define MAX_INTERRUPT_SOURCES 7
|
#define MAX_INTERRUPT_SOURCES 7
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local types
|
* Local types
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Describes a PIO interrupt source, including the PIO instance triggering the
|
* Describes a PIO interrupt source, including the PIO instance triggering the
|
||||||
* interrupt and the associated interrupt handler.
|
* interrupt and the associated interrupt handler.
|
||||||
*/
|
*/
|
||||||
typedef struct _InterruptSource
|
typedef struct _InterruptSource
|
||||||
{
|
{
|
||||||
/* Pointer to the source pin instance. */
|
/* Pointer to the source pin instance. */
|
||||||
const Pin *pPin;
|
const Pin *pPin;
|
||||||
|
|
||||||
/* Interrupt handler. */
|
/* Interrupt handler. */
|
||||||
void (*handler)( const Pin* ) ;
|
void (*handler)( const Pin* ) ;
|
||||||
} InterruptSource ;
|
} InterruptSource ;
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local variables
|
* Local variables
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/* List of interrupt sources. */
|
/* List of interrupt sources. */
|
||||||
static InterruptSource _aIntSources[MAX_INTERRUPT_SOURCES] ;
|
static InterruptSource _aIntSources[MAX_INTERRUPT_SOURCES] ;
|
||||||
|
|
||||||
/* Number of currently defined interrupt sources. */
|
/* Number of currently defined interrupt sources. */
|
||||||
static uint32_t _dwNumSources = 0;
|
static uint32_t _dwNumSources = 0;
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local Functions
|
* Local Functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
/*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------*/
|
||||||
/**
|
/**
|
||||||
* \brief Stub, to handling all PIO Capture interrupts, if not defined.
|
* \brief Stub, to handling all PIO Capture interrupts, if not defined.
|
||||||
*/
|
*/
|
||||||
/*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------*/
|
||||||
extern WEAK void PIO_CaptureHandler( void )
|
extern WEAK void PIO_CaptureHandler( void )
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Handles all interrupts on the given PIO controller.
|
* \brief Handles all interrupts on the given PIO controller.
|
||||||
* \param id PIO controller ID.
|
* \param id PIO controller ID.
|
||||||
* \param pPio PIO controller base address.
|
* \param pPio PIO controller base address.
|
||||||
*/
|
*/
|
||||||
extern void PioInterruptHandler( uint32_t id, Pio *pPio )
|
extern void PioInterruptHandler( uint32_t id, Pio *pPio )
|
||||||
{
|
{
|
||||||
uint32_t status;
|
uint32_t status;
|
||||||
uint32_t i;
|
uint32_t i;
|
||||||
|
|
||||||
/* Read PIO controller status */
|
/* Read PIO controller status */
|
||||||
status = pPio->PIO_ISR;
|
status = pPio->PIO_ISR;
|
||||||
status &= pPio->PIO_IMR;
|
status &= pPio->PIO_IMR;
|
||||||
|
|
||||||
/* Check pending events */
|
/* Check pending events */
|
||||||
if ( status != 0 )
|
if ( status != 0 )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "PIO interrupt on PIO controller #%" PRIu32 "\n\r", id ) ;
|
TRACE_DEBUG( "PIO interrupt on PIO controller #%" PRIu32 "\n\r", id ) ;
|
||||||
|
|
||||||
/* Find triggering source */
|
/* Find triggering source */
|
||||||
i = 0;
|
i = 0;
|
||||||
while ( status != 0 )
|
while ( status != 0 )
|
||||||
{
|
{
|
||||||
/* There cannot be an unconfigured source enabled. */
|
/* There cannot be an unconfigured source enabled. */
|
||||||
assert(i < _dwNumSources);
|
assert(i < _dwNumSources);
|
||||||
|
|
||||||
/* Source is configured on the same controller */
|
/* Source is configured on the same controller */
|
||||||
if (_aIntSources[i].pPin->id == id)
|
if (_aIntSources[i].pPin->id == id)
|
||||||
{
|
{
|
||||||
/* Source has PIOs whose statuses have changed */
|
/* Source has PIOs whose statuses have changed */
|
||||||
if ( (status & _aIntSources[i].pPin->mask) != 0 )
|
if ( (status & _aIntSources[i].pPin->mask) != 0 )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "Interrupt source #%" PRIu32 " triggered\n\r", i ) ;
|
TRACE_DEBUG( "Interrupt source #%" PRIu32 " triggered\n\r", i ) ;
|
||||||
|
|
||||||
_aIntSources[i].handler(_aIntSources[i].pPin);
|
_aIntSources[i].handler(_aIntSources[i].pPin);
|
||||||
status &= ~(_aIntSources[i].pPin->mask);
|
status &= ~(_aIntSources[i].pPin->mask);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Global Functions
|
* Global Functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Parallel IO Controller A interrupt handler
|
* \brief Parallel IO Controller A interrupt handler
|
||||||
* \Redefined PIOA interrupt handler for NVIC interrupt table.
|
* \Redefined PIOA interrupt handler for NVIC interrupt table.
|
||||||
*/
|
*/
|
||||||
extern void PIOA_IrqHandler( void )
|
extern void PIOA_IrqHandler( void )
|
||||||
{
|
{
|
||||||
if ( PIOA->PIO_PCISR != 0 )
|
if ( PIOA->PIO_PCISR != 0 )
|
||||||
{
|
{
|
||||||
PIO_CaptureHandler() ;
|
PIO_CaptureHandler() ;
|
||||||
}
|
}
|
||||||
|
|
||||||
PioInterruptHandler( ID_PIOA, PIOA ) ;
|
PioInterruptHandler( ID_PIOA, PIOA ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Parallel IO Controller B interrupt handler
|
* \brief Parallel IO Controller B interrupt handler
|
||||||
* \Redefined PIOB interrupt handler for NVIC interrupt table.
|
* \Redefined PIOB interrupt handler for NVIC interrupt table.
|
||||||
*/
|
*/
|
||||||
extern void PIOB_IrqHandler( void )
|
extern void PIOB_IrqHandler( void )
|
||||||
{
|
{
|
||||||
PioInterruptHandler( ID_PIOB, PIOB ) ;
|
PioInterruptHandler( ID_PIOB, PIOB ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Parallel IO Controller C interrupt handler
|
* \brief Parallel IO Controller C interrupt handler
|
||||||
* \Redefined PIOC interrupt handler for NVIC interrupt table.
|
* \Redefined PIOC interrupt handler for NVIC interrupt table.
|
||||||
*/
|
*/
|
||||||
extern void PIOC_IrqHandler( void )
|
extern void PIOC_IrqHandler( void )
|
||||||
{
|
{
|
||||||
PioInterruptHandler( ID_PIOC, PIOC ) ;
|
PioInterruptHandler( ID_PIOC, PIOC ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Initializes the PIO interrupt management logic
|
* \brief Initializes the PIO interrupt management logic
|
||||||
*
|
*
|
||||||
* The desired priority of PIO interrupts must be provided.
|
* The desired priority of PIO interrupts must be provided.
|
||||||
* Calling this function multiple times result in the reset of currently
|
* Calling this function multiple times result in the reset of currently
|
||||||
* configured interrupts.
|
* configured interrupts.
|
||||||
*
|
*
|
||||||
* \param priority PIO controller interrupts priority.
|
* \param priority PIO controller interrupts priority.
|
||||||
*/
|
*/
|
||||||
extern void PIO_InitializeInterrupts( uint32_t dwPriority )
|
extern void PIO_InitializeInterrupts( uint32_t dwPriority )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "PIO_Initialize()\n\r" ) ;
|
TRACE_DEBUG( "PIO_Initialize()\n\r" ) ;
|
||||||
|
|
||||||
/* Reset sources */
|
/* Reset sources */
|
||||||
_dwNumSources = 0 ;
|
_dwNumSources = 0 ;
|
||||||
|
|
||||||
/* Configure PIO interrupt sources */
|
/* Configure PIO interrupt sources */
|
||||||
TRACE_DEBUG( "PIO_Initialize: Configuring PIOA\n\r" ) ;
|
TRACE_DEBUG( "PIO_Initialize: Configuring PIOA\n\r" ) ;
|
||||||
PMC_EnablePeripheral( ID_PIOA ) ;
|
PMC_EnablePeripheral( ID_PIOA ) ;
|
||||||
PIOA->PIO_ISR ;
|
PIOA->PIO_ISR ;
|
||||||
PIOA->PIO_IDR = 0xFFFFFFFF ;
|
PIOA->PIO_IDR = 0xFFFFFFFF ;
|
||||||
NVIC_DisableIRQ( PIOA_IRQn ) ;
|
NVIC_DisableIRQ( PIOA_IRQn ) ;
|
||||||
NVIC_ClearPendingIRQ( PIOA_IRQn ) ;
|
NVIC_ClearPendingIRQ( PIOA_IRQn ) ;
|
||||||
NVIC_SetPriority( PIOA_IRQn, dwPriority ) ;
|
NVIC_SetPriority( PIOA_IRQn, dwPriority ) ;
|
||||||
NVIC_EnableIRQ( PIOA_IRQn ) ;
|
NVIC_EnableIRQ( PIOA_IRQn ) ;
|
||||||
|
|
||||||
TRACE_DEBUG( "PIO_Initialize: Configuring PIOB\n\r" ) ;
|
TRACE_DEBUG( "PIO_Initialize: Configuring PIOB\n\r" ) ;
|
||||||
PMC_EnablePeripheral( ID_PIOB ) ;
|
PMC_EnablePeripheral( ID_PIOB ) ;
|
||||||
PIOB->PIO_ISR ;
|
PIOB->PIO_ISR ;
|
||||||
PIOB->PIO_IDR = 0xFFFFFFFF ;
|
PIOB->PIO_IDR = 0xFFFFFFFF ;
|
||||||
NVIC_DisableIRQ( PIOB_IRQn ) ;
|
NVIC_DisableIRQ( PIOB_IRQn ) ;
|
||||||
NVIC_ClearPendingIRQ( PIOB_IRQn ) ;
|
NVIC_ClearPendingIRQ( PIOB_IRQn ) ;
|
||||||
NVIC_SetPriority( PIOB_IRQn, dwPriority ) ;
|
NVIC_SetPriority( PIOB_IRQn, dwPriority ) ;
|
||||||
NVIC_EnableIRQ( PIOB_IRQn ) ;
|
NVIC_EnableIRQ( PIOB_IRQn ) ;
|
||||||
|
|
||||||
TRACE_DEBUG( "PIO_Initialize: Configuring PIOC\n\r" ) ;
|
TRACE_DEBUG( "PIO_Initialize: Configuring PIOC\n\r" ) ;
|
||||||
PMC_EnablePeripheral( ID_PIOC ) ;
|
PMC_EnablePeripheral( ID_PIOC ) ;
|
||||||
PIOC->PIO_ISR ;
|
PIOC->PIO_ISR ;
|
||||||
PIOC->PIO_IDR = 0xFFFFFFFF ;
|
PIOC->PIO_IDR = 0xFFFFFFFF ;
|
||||||
NVIC_DisableIRQ( PIOC_IRQn ) ;
|
NVIC_DisableIRQ( PIOC_IRQn ) ;
|
||||||
NVIC_ClearPendingIRQ( PIOC_IRQn ) ;
|
NVIC_ClearPendingIRQ( PIOC_IRQn ) ;
|
||||||
NVIC_SetPriority( PIOC_IRQn, dwPriority ) ;
|
NVIC_SetPriority( PIOC_IRQn, dwPriority ) ;
|
||||||
NVIC_EnableIRQ( PIOC_IRQn ) ;
|
NVIC_EnableIRQ( PIOC_IRQn ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Configures a PIO or a group of PIO to generate an interrupt on status
|
* Configures a PIO or a group of PIO to generate an interrupt on status
|
||||||
* change. The provided interrupt handler will be called with the triggering
|
* change. The provided interrupt handler will be called with the triggering
|
||||||
* pin as its parameter (enabling different pin instances to share the same
|
* pin as its parameter (enabling different pin instances to share the same
|
||||||
* handler).
|
* handler).
|
||||||
* \param pPin Pointer to a Pin instance.
|
* \param pPin Pointer to a Pin instance.
|
||||||
* \param handler Interrupt handler function pointer.
|
* \param handler Interrupt handler function pointer.
|
||||||
*/
|
*/
|
||||||
extern void PIO_ConfigureIt( const Pin *pPin, void (*handler)( const Pin* ) )
|
extern void PIO_ConfigureIt( const Pin *pPin, void (*handler)( const Pin* ) )
|
||||||
{
|
{
|
||||||
Pio* pio ;
|
Pio* pio ;
|
||||||
InterruptSource* pSource ;
|
InterruptSource* pSource ;
|
||||||
|
|
||||||
TRACE_DEBUG( "PIO_ConfigureIt()\n\r" ) ;
|
TRACE_DEBUG( "PIO_ConfigureIt()\n\r" ) ;
|
||||||
|
|
||||||
assert( pPin ) ;
|
assert( pPin ) ;
|
||||||
pio = pPin->pio ;
|
pio = pPin->pio ;
|
||||||
assert( _dwNumSources < MAX_INTERRUPT_SOURCES ) ;
|
assert( _dwNumSources < MAX_INTERRUPT_SOURCES ) ;
|
||||||
|
|
||||||
/* Define new source */
|
/* Define new source */
|
||||||
TRACE_DEBUG( "PIO_ConfigureIt: Defining new source #%" PRIu32 ".\n\r", _dwNumSources ) ;
|
TRACE_DEBUG( "PIO_ConfigureIt: Defining new source #%" PRIu32 ".\n\r", _dwNumSources ) ;
|
||||||
|
|
||||||
pSource = &(_aIntSources[_dwNumSources]) ;
|
pSource = &(_aIntSources[_dwNumSources]) ;
|
||||||
pSource->pPin = pPin ;
|
pSource->pPin = pPin ;
|
||||||
pSource->handler = handler ;
|
pSource->handler = handler ;
|
||||||
_dwNumSources++ ;
|
_dwNumSources++ ;
|
||||||
|
|
||||||
/* PIO3 with additional interrupt support
|
/* PIO3 with additional interrupt support
|
||||||
* Configure additional interrupt mode registers */
|
* Configure additional interrupt mode registers */
|
||||||
if ( pPin->attribute & PIO_IT_AIME )
|
if ( pPin->attribute & PIO_IT_AIME )
|
||||||
{
|
{
|
||||||
// enable additional interrupt mode
|
// enable additional interrupt mode
|
||||||
pio->PIO_AIMER = pPin->mask ;
|
pio->PIO_AIMER = pPin->mask ;
|
||||||
|
|
||||||
// if bit field of selected pin is 1, set as Rising Edge/High level detection event
|
// if bit field of selected pin is 1, set as Rising Edge/High level detection event
|
||||||
if ( pPin->attribute & PIO_IT_RE_OR_HL )
|
if ( pPin->attribute & PIO_IT_RE_OR_HL )
|
||||||
{
|
{
|
||||||
pio->PIO_REHLSR = pPin->mask ;
|
pio->PIO_REHLSR = pPin->mask ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
pio->PIO_FELLSR = pPin->mask;
|
pio->PIO_FELLSR = pPin->mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* if bit field of selected pin is 1, set as edge detection source */
|
/* if bit field of selected pin is 1, set as edge detection source */
|
||||||
if (pPin->attribute & PIO_IT_EDGE)
|
if (pPin->attribute & PIO_IT_EDGE)
|
||||||
pio->PIO_ESR = pPin->mask;
|
pio->PIO_ESR = pPin->mask;
|
||||||
else
|
else
|
||||||
pio->PIO_LSR = pPin->mask;
|
pio->PIO_LSR = pPin->mask;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* disable additional interrupt mode */
|
/* disable additional interrupt mode */
|
||||||
pio->PIO_AIMDR = pPin->mask;
|
pio->PIO_AIMDR = pPin->mask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Enables the given interrupt source if it has been configured. The status
|
* Enables the given interrupt source if it has been configured. The status
|
||||||
* register of the corresponding PIO controller is cleared prior to enabling
|
* register of the corresponding PIO controller is cleared prior to enabling
|
||||||
* the interrupt.
|
* the interrupt.
|
||||||
* \param pPin Interrupt source to enable.
|
* \param pPin Interrupt source to enable.
|
||||||
*/
|
*/
|
||||||
extern void PIO_EnableIt( const Pin *pPin )
|
extern void PIO_EnableIt( const Pin *pPin )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "PIO_EnableIt()\n\r" ) ;
|
TRACE_DEBUG( "PIO_EnableIt()\n\r" ) ;
|
||||||
|
|
||||||
assert( pPin != NULL ) ;
|
assert( pPin != NULL ) ;
|
||||||
|
|
||||||
#ifndef NOASSERT
|
#ifndef NOASSERT
|
||||||
uint32_t i = 0;
|
uint32_t i = 0;
|
||||||
uint32_t dwFound = 0;
|
uint32_t dwFound = 0;
|
||||||
|
|
||||||
while ( (i < _dwNumSources) && !dwFound )
|
while ( (i < _dwNumSources) && !dwFound )
|
||||||
{
|
{
|
||||||
if ( _aIntSources[i].pPin == pPin )
|
if ( _aIntSources[i].pPin == pPin )
|
||||||
{
|
{
|
||||||
dwFound = 1 ;
|
dwFound = 1 ;
|
||||||
}
|
}
|
||||||
i++ ;
|
i++ ;
|
||||||
}
|
}
|
||||||
assert( dwFound != 0 ) ;
|
assert( dwFound != 0 ) ;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
pPin->pio->PIO_ISR;
|
pPin->pio->PIO_ISR;
|
||||||
pPin->pio->PIO_IER = pPin->mask ;
|
pPin->pio->PIO_IER = pPin->mask ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Disables a given interrupt source, with no added side effects.
|
* Disables a given interrupt source, with no added side effects.
|
||||||
*
|
*
|
||||||
* \param pPin Interrupt source to disable.
|
* \param pPin Interrupt source to disable.
|
||||||
*/
|
*/
|
||||||
extern void PIO_DisableIt( const Pin *pPin )
|
extern void PIO_DisableIt( const Pin *pPin )
|
||||||
{
|
{
|
||||||
assert( pPin != NULL ) ;
|
assert( pPin != NULL ) ;
|
||||||
|
|
||||||
TRACE_DEBUG( "PIO_DisableIt()\n\r" ) ;
|
TRACE_DEBUG( "PIO_DisableIt()\n\r" ) ;
|
||||||
|
|
||||||
pPin->pio->PIO_IDR = pPin->mask;
|
pPin->pio->PIO_IDR = pPin->mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,168 +1,168 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
#include "trace.h"
|
#include "trace.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local definitions
|
* Local definitions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
#define MASK_STATUS0 0xFFFFFFFC
|
#define MASK_STATUS0 0xFFFFFFFC
|
||||||
#define MASK_STATUS1 0xFFFFFFFF
|
#define MASK_STATUS1 0xFFFFFFFF
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enables the clock of a peripheral. The peripheral ID is used
|
* \brief Enables the clock of a peripheral. The peripheral ID is used
|
||||||
* to identify which peripheral is targetted.
|
* to identify which peripheral is targetted.
|
||||||
*
|
*
|
||||||
* \note The ID must NOT be shifted (i.e. 1 << ID_xxx).
|
* \note The ID must NOT be shifted (i.e. 1 << ID_xxx).
|
||||||
*
|
*
|
||||||
* \param id Peripheral ID (ID_xxx).
|
* \param id Peripheral ID (ID_xxx).
|
||||||
*/
|
*/
|
||||||
extern void PMC_EnablePeripheral( uint32_t dwId )
|
extern void PMC_EnablePeripheral( uint32_t dwId )
|
||||||
{
|
{
|
||||||
assert( dwId < 35 ) ;
|
assert( dwId < 35 ) ;
|
||||||
|
|
||||||
if ( dwId < 32 )
|
if ( dwId < 32 )
|
||||||
{
|
{
|
||||||
if ( (PMC->PMC_PCSR0 & ((uint32_t)1 << dwId)) == ((uint32_t)1 << dwId) )
|
if ( (PMC->PMC_PCSR0 & ((uint32_t)1 << dwId)) == ((uint32_t)1 << dwId) )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "PMC_EnablePeripheral: clock of peripheral" " %" PRIu32 " is already enabled\n\r", dwId ) ;
|
TRACE_DEBUG( "PMC_EnablePeripheral: clock of peripheral" " %" PRIu32 " is already enabled\n\r", dwId ) ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
PMC->PMC_PCER0 = 1 << dwId ;
|
PMC->PMC_PCER0 = 1 << dwId ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
dwId -= 32;
|
dwId -= 32;
|
||||||
if ((PMC->PMC_PCSR1 & ((uint32_t)1 << dwId)) == ((uint32_t)1 << dwId))
|
if ((PMC->PMC_PCSR1 & ((uint32_t)1 << dwId)) == ((uint32_t)1 << dwId))
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "PMC_EnablePeripheral: clock of peripheral" " %" PRIu32 " is already enabled\n\r", dwId + 32 ) ;
|
TRACE_DEBUG( "PMC_EnablePeripheral: clock of peripheral" " %" PRIu32 " is already enabled\n\r", dwId + 32 ) ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
PMC->PMC_PCER1 = 1 << dwId ;
|
PMC->PMC_PCER1 = 1 << dwId ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disables the clock of a peripheral. The peripheral ID is used
|
* \brief Disables the clock of a peripheral. The peripheral ID is used
|
||||||
* to identify which peripheral is targetted.
|
* to identify which peripheral is targetted.
|
||||||
*
|
*
|
||||||
* \note The ID must NOT be shifted (i.e. 1 << ID_xxx).
|
* \note The ID must NOT be shifted (i.e. 1 << ID_xxx).
|
||||||
*
|
*
|
||||||
* \param id Peripheral ID (ID_xxx).
|
* \param id Peripheral ID (ID_xxx).
|
||||||
*/
|
*/
|
||||||
extern void PMC_DisablePeripheral( uint32_t dwId )
|
extern void PMC_DisablePeripheral( uint32_t dwId )
|
||||||
{
|
{
|
||||||
assert( dwId < 35 ) ;
|
assert( dwId < 35 ) ;
|
||||||
|
|
||||||
if ( dwId < 32 )
|
if ( dwId < 32 )
|
||||||
{
|
{
|
||||||
if ( (PMC->PMC_PCSR0 & ((uint32_t)1 << dwId)) != ((uint32_t)1 << dwId) )
|
if ( (PMC->PMC_PCSR0 & ((uint32_t)1 << dwId)) != ((uint32_t)1 << dwId) )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG("PMC_DisablePeripheral: clock of peripheral" " %" PRIu32 " is not enabled\n\r", dwId ) ;
|
TRACE_DEBUG("PMC_DisablePeripheral: clock of peripheral" " %" PRIu32 " is not enabled\n\r", dwId ) ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
PMC->PMC_PCDR0 = 1 << dwId ;
|
PMC->PMC_PCDR0 = 1 << dwId ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
dwId -= 32 ;
|
dwId -= 32 ;
|
||||||
if ( (PMC->PMC_PCSR1 & ((uint32_t)1 << dwId)) != ((uint32_t)1 << dwId) )
|
if ( (PMC->PMC_PCSR1 & ((uint32_t)1 << dwId)) != ((uint32_t)1 << dwId) )
|
||||||
{
|
{
|
||||||
TRACE_DEBUG( "PMC_DisablePeripheral: clock of peripheral" " %" PRIu32 " is not enabled\n\r", dwId + 32 ) ;
|
TRACE_DEBUG( "PMC_DisablePeripheral: clock of peripheral" " %" PRIu32 " is not enabled\n\r", dwId + 32 ) ;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
PMC->PMC_PCDR1 = 1 << dwId ;
|
PMC->PMC_PCDR1 = 1 << dwId ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enable all the periph clock via PMC.
|
* \brief Enable all the periph clock via PMC.
|
||||||
*/
|
*/
|
||||||
extern void PMC_EnableAllPeripherals( void )
|
extern void PMC_EnableAllPeripherals( void )
|
||||||
{
|
{
|
||||||
PMC->PMC_PCER0 = MASK_STATUS0 ;
|
PMC->PMC_PCER0 = MASK_STATUS0 ;
|
||||||
while ( (PMC->PMC_PCSR0 & MASK_STATUS0) != MASK_STATUS0 ) ;
|
while ( (PMC->PMC_PCSR0 & MASK_STATUS0) != MASK_STATUS0 ) ;
|
||||||
|
|
||||||
PMC->PMC_PCER1 = MASK_STATUS1 ;
|
PMC->PMC_PCER1 = MASK_STATUS1 ;
|
||||||
while ( (PMC->PMC_PCSR1 & MASK_STATUS1) != MASK_STATUS1 ) ;
|
while ( (PMC->PMC_PCSR1 & MASK_STATUS1) != MASK_STATUS1 ) ;
|
||||||
|
|
||||||
TRACE_DEBUG( "Enable all periph clocks\n\r" ) ;
|
TRACE_DEBUG( "Enable all periph clocks\n\r" ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disable all the periph clock via PMC.
|
* \brief Disable all the periph clock via PMC.
|
||||||
*/
|
*/
|
||||||
extern void PMC_DisableAllPeripherals( void )
|
extern void PMC_DisableAllPeripherals( void )
|
||||||
{
|
{
|
||||||
PMC->PMC_PCDR0 = MASK_STATUS0 ;
|
PMC->PMC_PCDR0 = MASK_STATUS0 ;
|
||||||
while ( (PMC->PMC_PCSR0 & MASK_STATUS0) != 0 ) ;
|
while ( (PMC->PMC_PCSR0 & MASK_STATUS0) != 0 ) ;
|
||||||
|
|
||||||
PMC->PMC_PCDR1 = MASK_STATUS1 ;
|
PMC->PMC_PCDR1 = MASK_STATUS1 ;
|
||||||
while ( (PMC->PMC_PCSR1 & MASK_STATUS1) != 0 ) ;
|
while ( (PMC->PMC_PCSR1 & MASK_STATUS1) != 0 ) ;
|
||||||
|
|
||||||
TRACE_DEBUG( "Disable all periph clocks\n\r" ) ;
|
TRACE_DEBUG( "Disable all periph clocks\n\r" ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Get Periph Status for the given peripheral ID.
|
* \brief Get Periph Status for the given peripheral ID.
|
||||||
*
|
*
|
||||||
* \param id Peripheral ID (ID_xxx).
|
* \param id Peripheral ID (ID_xxx).
|
||||||
*/
|
*/
|
||||||
extern uint32_t PMC_IsPeriphEnabled( uint32_t dwId )
|
extern uint32_t PMC_IsPeriphEnabled( uint32_t dwId )
|
||||||
{
|
{
|
||||||
assert( dwId < 35 ) ;
|
assert( dwId < 35 ) ;
|
||||||
|
|
||||||
if ( dwId < 32 )
|
if ( dwId < 32 )
|
||||||
{
|
{
|
||||||
return ( PMC->PMC_PCSR0 & (1 << dwId) ) ;
|
return ( PMC->PMC_PCSR0 & (1 << dwId) ) ;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return ( PMC->PMC_PCSR1 & (1 << (dwId - 32)) ) ;
|
return ( PMC->PMC_PCSR1 & (1 << (dwId - 32)) ) ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,352 +1,352 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** \addtogroup spi_module Working with SPI
|
/** \addtogroup spi_module Working with SPI
|
||||||
* The SPI driver provides the interface to configure and use the SPI
|
* The SPI driver provides the interface to configure and use the SPI
|
||||||
* peripheral.
|
* peripheral.
|
||||||
*
|
*
|
||||||
* The Serial Peripheral Interface (SPI) circuit is a synchronous serial
|
* The Serial Peripheral Interface (SPI) circuit is a synchronous serial
|
||||||
* data link that provides communication with external devices in Master
|
* data link that provides communication with external devices in Master
|
||||||
* or Slave Mode.
|
* or Slave Mode.
|
||||||
*
|
*
|
||||||
* To use the SPI, the user has to follow these few steps:
|
* To use the SPI, the user has to follow these few steps:
|
||||||
* -# Enable the SPI pins required by the application (see pio.h).
|
* -# Enable the SPI pins required by the application (see pio.h).
|
||||||
* -# Configure the SPI using the \ref SPI_Configure(). This enables the
|
* -# Configure the SPI using the \ref SPI_Configure(). This enables the
|
||||||
* peripheral clock. The mode register is loaded with the given value.
|
* peripheral clock. The mode register is loaded with the given value.
|
||||||
* -# Configure all the necessary chip selects with \ref SPI_ConfigureNPCS().
|
* -# Configure all the necessary chip selects with \ref SPI_ConfigureNPCS().
|
||||||
* -# Enable the SPI by calling \ref SPI_Enable().
|
* -# Enable the SPI by calling \ref SPI_Enable().
|
||||||
* -# Send/receive data using \ref SPI_Write() and \ref SPI_Read(). Note that \ref SPI_Read()
|
* -# Send/receive data using \ref SPI_Write() and \ref SPI_Read(). Note that \ref SPI_Read()
|
||||||
* must be called after \ref SPI_Write() to retrieve the last value read.
|
* must be called after \ref SPI_Write() to retrieve the last value read.
|
||||||
* -# Send/receive data using the PDC with the \ref SPI_WriteBuffer() and
|
* -# Send/receive data using the PDC with the \ref SPI_WriteBuffer() and
|
||||||
* \ref SPI_ReadBuffer() functions.
|
* \ref SPI_ReadBuffer() functions.
|
||||||
* -# Disable the SPI by calling \ref SPI_Disable().
|
* -# Disable the SPI by calling \ref SPI_Disable().
|
||||||
*
|
*
|
||||||
* For more accurate information, please look at the SPI section of the
|
* For more accurate information, please look at the SPI section of the
|
||||||
* Datasheet.
|
* Datasheet.
|
||||||
*
|
*
|
||||||
* Related files :\n
|
* Related files :\n
|
||||||
* \ref spi.c\n
|
* \ref spi.c\n
|
||||||
* \ref spi.h.\n
|
* \ref spi.h.\n
|
||||||
*/
|
*/
|
||||||
/*@{*/
|
/*@{*/
|
||||||
/*@}*/
|
/*@}*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* Implementation of Serial Peripheral Interface (SPI) controller.
|
* Implementation of Serial Peripheral Interface (SPI) controller.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
#include "pmc.h"
|
#include "pmc.h"
|
||||||
#include "spi.h"
|
#include "spi.h"
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enables a SPI peripheral.
|
* \brief Enables a SPI peripheral.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*/
|
*/
|
||||||
extern void SPI_Enable( Spi* spi )
|
extern void SPI_Enable( Spi* spi )
|
||||||
{
|
{
|
||||||
spi->SPI_CR = SPI_CR_SPIEN ;
|
spi->SPI_CR = SPI_CR_SPIEN ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disables a SPI peripheral.
|
* \brief Disables a SPI peripheral.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*/
|
*/
|
||||||
extern void SPI_Disable( Spi* spi )
|
extern void SPI_Disable( Spi* spi )
|
||||||
{
|
{
|
||||||
spi->SPI_CR = SPI_CR_SPIDIS ;
|
spi->SPI_CR = SPI_CR_SPIDIS ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enables one or more interrupt sources of a SPI peripheral.
|
* \brief Enables one or more interrupt sources of a SPI peripheral.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param sources Bitwise OR of selected interrupt sources.
|
* \param sources Bitwise OR of selected interrupt sources.
|
||||||
*/
|
*/
|
||||||
extern void SPI_EnableIt( Spi* spi, uint32_t dwSources )
|
extern void SPI_EnableIt( Spi* spi, uint32_t dwSources )
|
||||||
{
|
{
|
||||||
spi->SPI_IER = dwSources ;
|
spi->SPI_IER = dwSources ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disables one or more interrupt sources of a SPI peripheral.
|
* \brief Disables one or more interrupt sources of a SPI peripheral.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param sources Bitwise OR of selected interrupt sources.
|
* \param sources Bitwise OR of selected interrupt sources.
|
||||||
*/
|
*/
|
||||||
extern void SPI_DisableIt( Spi* spi, uint32_t dwSources )
|
extern void SPI_DisableIt( Spi* spi, uint32_t dwSources )
|
||||||
{
|
{
|
||||||
spi->SPI_IDR = dwSources ;
|
spi->SPI_IDR = dwSources ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures a SPI peripheral as specified. The configuration can be computed
|
* \brief Configures a SPI peripheral as specified. The configuration can be computed
|
||||||
* using several macros (see \ref spi_configuration_macros).
|
* using several macros (see \ref spi_configuration_macros).
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param id Peripheral ID of the SPI.
|
* \param id Peripheral ID of the SPI.
|
||||||
* \param configuration Value of the SPI configuration register.
|
* \param configuration Value of the SPI configuration register.
|
||||||
*/
|
*/
|
||||||
extern void SPI_Configure( Spi* spi, uint32_t dwId, uint32_t dwConfiguration )
|
extern void SPI_Configure( Spi* spi, uint32_t dwId, uint32_t dwConfiguration )
|
||||||
{
|
{
|
||||||
PMC_EnablePeripheral( dwId ) ;
|
PMC_EnablePeripheral( dwId ) ;
|
||||||
spi->SPI_CR = SPI_CR_SPIDIS ;
|
spi->SPI_CR = SPI_CR_SPIDIS ;
|
||||||
|
|
||||||
/* Execute a software reset of the SPI twice */
|
/* Execute a software reset of the SPI twice */
|
||||||
spi->SPI_CR = SPI_CR_SWRST ;
|
spi->SPI_CR = SPI_CR_SWRST ;
|
||||||
spi->SPI_CR = SPI_CR_SWRST ;
|
spi->SPI_CR = SPI_CR_SWRST ;
|
||||||
spi->SPI_MR = dwConfiguration ;
|
spi->SPI_MR = dwConfiguration ;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures a chip select of a SPI peripheral. The chip select configuration
|
* \brief Configures a chip select of a SPI peripheral. The chip select configuration
|
||||||
* is computed using several macros (see \ref spi_configuration_macros).
|
* is computed using several macros (see \ref spi_configuration_macros).
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param npcs Chip select to configure (0, 1, 2 or 3).
|
* \param npcs Chip select to configure (0, 1, 2 or 3).
|
||||||
* \param configuration Desired chip select configuration.
|
* \param configuration Desired chip select configuration.
|
||||||
*/
|
*/
|
||||||
void SPI_ConfigureNPCS( Spi* spi, uint32_t dwNpcs, uint32_t dwConfiguration )
|
void SPI_ConfigureNPCS( Spi* spi, uint32_t dwNpcs, uint32_t dwConfiguration )
|
||||||
{
|
{
|
||||||
spi->SPI_CSR[dwNpcs] = dwConfiguration ;
|
spi->SPI_CSR[dwNpcs] = dwConfiguration ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Get the current status register of the given SPI peripheral.
|
* \brief Get the current status register of the given SPI peripheral.
|
||||||
* \note This resets the internal value of the status register, so further
|
* \note This resets the internal value of the status register, so further
|
||||||
* read may yield different values.
|
* read may yield different values.
|
||||||
* \param spi Pointer to a Spi instance.
|
* \param spi Pointer to a Spi instance.
|
||||||
* \return SPI status register.
|
* \return SPI status register.
|
||||||
*/
|
*/
|
||||||
extern uint32_t SPI_GetStatus( Spi* spi )
|
extern uint32_t SPI_GetStatus( Spi* spi )
|
||||||
{
|
{
|
||||||
return spi->SPI_SR ;
|
return spi->SPI_SR ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Reads and returns the last word of data received by a SPI peripheral. This
|
* \brief Reads and returns the last word of data received by a SPI peripheral. This
|
||||||
* method must be called after a successful SPI_Write call.
|
* method must be called after a successful SPI_Write call.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*
|
*
|
||||||
* \return readed data.
|
* \return readed data.
|
||||||
*/
|
*/
|
||||||
extern uint32_t SPI_Read( Spi* spi )
|
extern uint32_t SPI_Read( Spi* spi )
|
||||||
{
|
{
|
||||||
while ( (spi->SPI_SR & SPI_SR_RDRF) == 0 ) ;
|
while ( (spi->SPI_SR & SPI_SR_RDRF) == 0 ) ;
|
||||||
|
|
||||||
return spi->SPI_RDR & 0xFFFF ;
|
return spi->SPI_RDR & 0xFFFF ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sends data through a SPI peripheral. If the SPI is configured to use a fixed
|
* \brief Sends data through a SPI peripheral. If the SPI is configured to use a fixed
|
||||||
* peripheral select, the npcs value is meaningless. Otherwise, it identifies
|
* peripheral select, the npcs value is meaningless. Otherwise, it identifies
|
||||||
* the component which shall be addressed.
|
* the component which shall be addressed.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param npcs Chip select of the component to address (0, 1, 2 or 3).
|
* \param npcs Chip select of the component to address (0, 1, 2 or 3).
|
||||||
* \param data Word of data to send.
|
* \param data Word of data to send.
|
||||||
*/
|
*/
|
||||||
extern void SPI_Write( Spi* spi, uint32_t dwNpcs, uint16_t wData )
|
extern void SPI_Write( Spi* spi, uint32_t dwNpcs, uint16_t wData )
|
||||||
{
|
{
|
||||||
/* Send data */
|
/* Send data */
|
||||||
while ( (spi->SPI_SR & SPI_SR_TXEMPTY) == 0 ) ;
|
while ( (spi->SPI_SR & SPI_SR_TXEMPTY) == 0 ) ;
|
||||||
spi->SPI_TDR = wData | SPI_PCS( dwNpcs ) ;
|
spi->SPI_TDR = wData | SPI_PCS( dwNpcs ) ;
|
||||||
while ( (spi->SPI_SR & SPI_SR_TDRE) == 0 ) ;
|
while ( (spi->SPI_SR & SPI_SR_TDRE) == 0 ) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Check if SPI transfer finish.
|
* \brief Check if SPI transfer finish.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*
|
*
|
||||||
* \return Returns 1 if there is no pending write operation on the SPI; otherwise
|
* \return Returns 1 if there is no pending write operation on the SPI; otherwise
|
||||||
* returns 0.
|
* returns 0.
|
||||||
*/
|
*/
|
||||||
extern uint32_t SPI_IsFinished( Spi* spi )
|
extern uint32_t SPI_IsFinished( Spi* spi )
|
||||||
{
|
{
|
||||||
return ((spi->SPI_SR & SPI_SR_TXEMPTY) != 0) ;
|
return ((spi->SPI_SR & SPI_SR_TXEMPTY) != 0) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enable Spi PDC transmit
|
* \brief Enable Spi PDC transmit
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*/
|
*/
|
||||||
extern void SPI_PdcEnableTx( Spi* spi )
|
extern void SPI_PdcEnableTx( Spi* spi )
|
||||||
{
|
{
|
||||||
spi->SPI_PTCR = SPI_PTCR_TXTEN ;
|
spi->SPI_PTCR = SPI_PTCR_TXTEN ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disable Spi PDC transmit
|
* \brief Disable Spi PDC transmit
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*/
|
*/
|
||||||
extern void SPI_PdcDisableTx( Spi* spi )
|
extern void SPI_PdcDisableTx( Spi* spi )
|
||||||
{
|
{
|
||||||
spi->SPI_PTCR = SPI_PTCR_TXTDIS ;
|
spi->SPI_PTCR = SPI_PTCR_TXTDIS ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enable Spi PDC receive
|
* \brief Enable Spi PDC receive
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*/
|
*/
|
||||||
extern void SPI_PdcEnableRx( Spi* spi )
|
extern void SPI_PdcEnableRx( Spi* spi )
|
||||||
{
|
{
|
||||||
spi->SPI_PTCR = SPI_PTCR_RXTEN ;
|
spi->SPI_PTCR = SPI_PTCR_RXTEN ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disable Spi PDC receive
|
* \brief Disable Spi PDC receive
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
*/
|
*/
|
||||||
extern void SPI_PdcDisableRx( Spi* spi )
|
extern void SPI_PdcDisableRx( Spi* spi )
|
||||||
{
|
{
|
||||||
spi->SPI_PTCR = SPI_PTCR_RXTDIS ;
|
spi->SPI_PTCR = SPI_PTCR_RXTDIS ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Set PDC transmit and next transmit buffer address and size.
|
* \brief Set PDC transmit and next transmit buffer address and size.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param txBuf PDC transmit buffer address.
|
* \param txBuf PDC transmit buffer address.
|
||||||
* \param txCount Length in bytes of the transmit buffer.
|
* \param txCount Length in bytes of the transmit buffer.
|
||||||
* \param txNextBuf PDC next transmit buffer address.
|
* \param txNextBuf PDC next transmit buffer address.
|
||||||
* \param txNextCount Length in bytes of the next transmit buffer.
|
* \param txNextCount Length in bytes of the next transmit buffer.
|
||||||
*/
|
*/
|
||||||
extern void SPI_PdcSetTx( Spi* spi, void* pvTxBuf, uint32_t dwTxCount, void* pvTxNextBuf, uint32_t dwTxNextCount )
|
extern void SPI_PdcSetTx( Spi* spi, void* pvTxBuf, uint32_t dwTxCount, void* pvTxNextBuf, uint32_t dwTxNextCount )
|
||||||
{
|
{
|
||||||
spi->SPI_TPR = (uint32_t)pvTxBuf ;
|
spi->SPI_TPR = (uint32_t)pvTxBuf ;
|
||||||
spi->SPI_TCR = dwTxCount ;
|
spi->SPI_TCR = dwTxCount ;
|
||||||
spi->SPI_TNPR = (uint32_t)pvTxNextBuf ;
|
spi->SPI_TNPR = (uint32_t)pvTxNextBuf ;
|
||||||
spi->SPI_TNCR = dwTxNextCount ;
|
spi->SPI_TNCR = dwTxNextCount ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Set PDC receive and next receive buffer address and size.
|
* \brief Set PDC receive and next receive buffer address and size.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param rxBuf PDC receive buffer address.
|
* \param rxBuf PDC receive buffer address.
|
||||||
* \param rxCount Length in bytes of the receive buffer.
|
* \param rxCount Length in bytes of the receive buffer.
|
||||||
* \param rxNextBuf PDC next receive buffer address.
|
* \param rxNextBuf PDC next receive buffer address.
|
||||||
* \param rxNextCount Length in bytes of the next receive buffer.
|
* \param rxNextCount Length in bytes of the next receive buffer.
|
||||||
*/
|
*/
|
||||||
extern void SPI_PdcSetRx( Spi* spi, void* pvRxBuf, uint32_t dwRxCount, void* pvRxNextBuf, uint32_t dwRxNextCount )
|
extern void SPI_PdcSetRx( Spi* spi, void* pvRxBuf, uint32_t dwRxCount, void* pvRxNextBuf, uint32_t dwRxNextCount )
|
||||||
{
|
{
|
||||||
spi->SPI_RPR = (uint32_t)pvRxBuf ;
|
spi->SPI_RPR = (uint32_t)pvRxBuf ;
|
||||||
spi->SPI_RCR = dwRxCount ;
|
spi->SPI_RCR = dwRxCount ;
|
||||||
spi->SPI_RNPR = (uint32_t)pvRxNextBuf ;
|
spi->SPI_RNPR = (uint32_t)pvRxNextBuf ;
|
||||||
spi->SPI_RNCR = dwRxNextCount ;
|
spi->SPI_RNCR = dwRxNextCount ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sends the contents of buffer through a SPI peripheral, using the PDC to
|
* \brief Sends the contents of buffer through a SPI peripheral, using the PDC to
|
||||||
* take care of the transfer.
|
* take care of the transfer.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param buffer Data buffer to send.
|
* \param buffer Data buffer to send.
|
||||||
* \param length Length of the data buffer.
|
* \param length Length of the data buffer.
|
||||||
*/
|
*/
|
||||||
extern uint32_t SPI_WriteBuffer( Spi* spi, void* pvBuffer, uint32_t dwLength )
|
extern uint32_t SPI_WriteBuffer( Spi* spi, void* pvBuffer, uint32_t dwLength )
|
||||||
{
|
{
|
||||||
/* Check if first bank is free */
|
/* Check if first bank is free */
|
||||||
if ( spi->SPI_TCR == 0 )
|
if ( spi->SPI_TCR == 0 )
|
||||||
{
|
{
|
||||||
spi->SPI_TPR = (uint32_t)pvBuffer ;
|
spi->SPI_TPR = (uint32_t)pvBuffer ;
|
||||||
spi->SPI_TCR = dwLength ;
|
spi->SPI_TCR = dwLength ;
|
||||||
spi->SPI_PTCR = PERIPH_PTCR_TXTEN ;
|
spi->SPI_PTCR = PERIPH_PTCR_TXTEN ;
|
||||||
|
|
||||||
return 1 ;
|
return 1 ;
|
||||||
}
|
}
|
||||||
/* Check if second bank is free */
|
/* Check if second bank is free */
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ( spi->SPI_TNCR == 0 )
|
if ( spi->SPI_TNCR == 0 )
|
||||||
{
|
{
|
||||||
spi->SPI_TNPR = (uint32_t)pvBuffer ;
|
spi->SPI_TNPR = (uint32_t)pvBuffer ;
|
||||||
spi->SPI_TNCR = dwLength ;
|
spi->SPI_TNCR = dwLength ;
|
||||||
|
|
||||||
return 1 ;
|
return 1 ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* No free banks */
|
/* No free banks */
|
||||||
return 0 ;
|
return 0 ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Reads data from a SPI peripheral until the provided buffer is filled. This
|
* \brief Reads data from a SPI peripheral until the provided buffer is filled. This
|
||||||
* method does NOT need to be called after SPI_Write or SPI_WriteBuffer.
|
* method does NOT need to be called after SPI_Write or SPI_WriteBuffer.
|
||||||
*
|
*
|
||||||
* \param spi Pointer to an Spi instance.
|
* \param spi Pointer to an Spi instance.
|
||||||
* \param buffer Data buffer to store incoming bytes.
|
* \param buffer Data buffer to store incoming bytes.
|
||||||
* \param length Length in bytes of the data buffer.
|
* \param length Length in bytes of the data buffer.
|
||||||
*/
|
*/
|
||||||
extern uint32_t SPI_ReadBuffer( Spi* spi, void *pvBuffer, uint32_t dwLength )
|
extern uint32_t SPI_ReadBuffer( Spi* spi, void *pvBuffer, uint32_t dwLength )
|
||||||
{
|
{
|
||||||
/* Check if the first bank is free */
|
/* Check if the first bank is free */
|
||||||
if ( spi->SPI_RCR == 0 )
|
if ( spi->SPI_RCR == 0 )
|
||||||
{
|
{
|
||||||
spi->SPI_RPR = (uint32_t)pvBuffer ;
|
spi->SPI_RPR = (uint32_t)pvBuffer ;
|
||||||
spi->SPI_RCR = dwLength ;
|
spi->SPI_RCR = dwLength ;
|
||||||
spi->SPI_PTCR = PERIPH_PTCR_RXTEN ;
|
spi->SPI_PTCR = PERIPH_PTCR_RXTEN ;
|
||||||
|
|
||||||
return 1 ;
|
return 1 ;
|
||||||
}
|
}
|
||||||
/* Check if second bank is free */
|
/* Check if second bank is free */
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ( spi->SPI_RNCR == 0 )
|
if ( spi->SPI_RNCR == 0 )
|
||||||
{
|
{
|
||||||
spi->SPI_RNPR = (uint32_t)pvBuffer ;
|
spi->SPI_RNPR = (uint32_t)pvBuffer ;
|
||||||
spi->SPI_RNCR = dwLength ;
|
spi->SPI_RNCR = dwLength ;
|
||||||
return 1 ;
|
return 1 ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* No free bank */
|
/* No free bank */
|
||||||
return 0 ;
|
return 0 ;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,175 +1,175 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* Implementation of Timer Counter (TC).
|
* Implementation of Timer Counter (TC).
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Global functions
|
* Global functions
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures a Timer Counter Channel
|
* \brief Configures a Timer Counter Channel
|
||||||
*
|
*
|
||||||
* Configures a Timer Counter to operate in the given mode. Timer is stopped
|
* Configures a Timer Counter to operate in the given mode. Timer is stopped
|
||||||
* after configuration and must be restarted with TC_Start(). All the
|
* after configuration and must be restarted with TC_Start(). All the
|
||||||
* interrupts of the timer are also disabled.
|
* interrupts of the timer are also disabled.
|
||||||
*
|
*
|
||||||
* \param pTc Pointer to a Tc instance.
|
* \param pTc Pointer to a Tc instance.
|
||||||
* \param channel Channel number.
|
* \param channel Channel number.
|
||||||
* \param mode Operating mode (TC_CMR value).
|
* \param mode Operating mode (TC_CMR value).
|
||||||
*/
|
*/
|
||||||
extern void TC_Configure( Tc *pTc, uint32_t dwChannel, uint32_t dwMode )
|
extern void TC_Configure( Tc *pTc, uint32_t dwChannel, uint32_t dwMode )
|
||||||
{
|
{
|
||||||
TcChannel* pTcCh ;
|
TcChannel* pTcCh ;
|
||||||
|
|
||||||
assert( dwChannel < (sizeof( pTc->TC_CHANNEL )/sizeof( pTc->TC_CHANNEL[0] )) ) ;
|
assert( dwChannel < (sizeof( pTc->TC_CHANNEL )/sizeof( pTc->TC_CHANNEL[0] )) ) ;
|
||||||
pTcCh = pTc->TC_CHANNEL+dwChannel ;
|
pTcCh = pTc->TC_CHANNEL+dwChannel ;
|
||||||
|
|
||||||
/* Disable TC clock */
|
/* Disable TC clock */
|
||||||
pTcCh->TC_CCR = TC_CCR_CLKDIS ;
|
pTcCh->TC_CCR = TC_CCR_CLKDIS ;
|
||||||
|
|
||||||
/* Disable interrupts */
|
/* Disable interrupts */
|
||||||
pTcCh->TC_IDR = 0xFFFFFFFF ;
|
pTcCh->TC_IDR = 0xFFFFFFFF ;
|
||||||
|
|
||||||
/* Clear status register */
|
/* Clear status register */
|
||||||
pTcCh->TC_SR ;
|
pTcCh->TC_SR ;
|
||||||
|
|
||||||
/* Set mode */
|
/* Set mode */
|
||||||
pTcCh->TC_CMR = dwMode ;
|
pTcCh->TC_CMR = dwMode ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Reset and Start the TC Channel
|
* \brief Reset and Start the TC Channel
|
||||||
*
|
*
|
||||||
* Enables the timer clock and performs a software reset to start the counting.
|
* Enables the timer clock and performs a software reset to start the counting.
|
||||||
*
|
*
|
||||||
* \param pTc Pointer to a Tc instance.
|
* \param pTc Pointer to a Tc instance.
|
||||||
* \param dwChannel Channel number.
|
* \param dwChannel Channel number.
|
||||||
*/
|
*/
|
||||||
extern void TC_Start( Tc *pTc, uint32_t dwChannel )
|
extern void TC_Start( Tc *pTc, uint32_t dwChannel )
|
||||||
{
|
{
|
||||||
TcChannel* pTcCh ;
|
TcChannel* pTcCh ;
|
||||||
|
|
||||||
assert( dwChannel < (sizeof( pTc->TC_CHANNEL )/sizeof( pTc->TC_CHANNEL[0] )) ) ;
|
assert( dwChannel < (sizeof( pTc->TC_CHANNEL )/sizeof( pTc->TC_CHANNEL[0] )) ) ;
|
||||||
|
|
||||||
pTcCh = pTc->TC_CHANNEL+dwChannel ;
|
pTcCh = pTc->TC_CHANNEL+dwChannel ;
|
||||||
pTcCh->TC_CCR = TC_CCR_CLKEN | TC_CCR_SWTRG ;
|
pTcCh->TC_CCR = TC_CCR_CLKEN | TC_CCR_SWTRG ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Stop TC Channel
|
* \brief Stop TC Channel
|
||||||
*
|
*
|
||||||
* Disables the timer clock, stopping the counting.
|
* Disables the timer clock, stopping the counting.
|
||||||
*
|
*
|
||||||
* \param pTc Pointer to a Tc instance.
|
* \param pTc Pointer to a Tc instance.
|
||||||
* \param dwChannel Channel number.
|
* \param dwChannel Channel number.
|
||||||
*/
|
*/
|
||||||
extern void TC_Stop(Tc *pTc, uint32_t dwChannel )
|
extern void TC_Stop(Tc *pTc, uint32_t dwChannel )
|
||||||
{
|
{
|
||||||
TcChannel* pTcCh ;
|
TcChannel* pTcCh ;
|
||||||
|
|
||||||
assert( dwChannel < (sizeof( pTc->TC_CHANNEL )/sizeof( pTc->TC_CHANNEL[0] )) ) ;
|
assert( dwChannel < (sizeof( pTc->TC_CHANNEL )/sizeof( pTc->TC_CHANNEL[0] )) ) ;
|
||||||
|
|
||||||
pTcCh = pTc->TC_CHANNEL+dwChannel ;
|
pTcCh = pTc->TC_CHANNEL+dwChannel ;
|
||||||
pTcCh->TC_CCR = TC_CCR_CLKDIS ;
|
pTcCh->TC_CCR = TC_CCR_CLKDIS ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Find best MCK divisor
|
* \brief Find best MCK divisor
|
||||||
*
|
*
|
||||||
* Finds the best MCK divisor given the timer frequency and MCK. The result
|
* Finds the best MCK divisor given the timer frequency and MCK. The result
|
||||||
* is guaranteed to satisfy the following equation:
|
* is guaranteed to satisfy the following equation:
|
||||||
* \code
|
* \code
|
||||||
* (MCK / (DIV * 65536)) <= freq <= (MCK / DIV)
|
* (MCK / (DIV * 65536)) <= freq <= (MCK / DIV)
|
||||||
* \endcode
|
* \endcode
|
||||||
* with DIV being the highest possible value.
|
* with DIV being the highest possible value.
|
||||||
*
|
*
|
||||||
* \param dwFreq Desired timer frequency.
|
* \param dwFreq Desired timer frequency.
|
||||||
* \param dwMCk Master clock frequency.
|
* \param dwMCk Master clock frequency.
|
||||||
* \param dwDiv Divisor value.
|
* \param dwDiv Divisor value.
|
||||||
* \param dwTcClks TCCLKS field value for divisor.
|
* \param dwTcClks TCCLKS field value for divisor.
|
||||||
* \param dwBoardMCK Board clock frequency.
|
* \param dwBoardMCK Board clock frequency.
|
||||||
*
|
*
|
||||||
* \return 1 if a proper divisor has been found, otherwise 0.
|
* \return 1 if a proper divisor has been found, otherwise 0.
|
||||||
*/
|
*/
|
||||||
extern uint32_t TC_FindMckDivisor( uint32_t dwFreq, uint32_t dwMCk, uint32_t *dwDiv, uint32_t *dwTcClks, uint32_t dwBoardMCK )
|
extern uint32_t TC_FindMckDivisor( uint32_t dwFreq, uint32_t dwMCk, uint32_t *dwDiv, uint32_t *dwTcClks, uint32_t dwBoardMCK )
|
||||||
{
|
{
|
||||||
const uint32_t adwDivisors[5] = { 2, 8, 32, 128, dwBoardMCK / 32768 } ;
|
const uint32_t adwDivisors[5] = { 2, 8, 32, 128, dwBoardMCK / 32768 } ;
|
||||||
|
|
||||||
uint32_t dwIndex = 0 ;
|
uint32_t dwIndex = 0 ;
|
||||||
|
|
||||||
/* Satisfy lower bound */
|
/* Satisfy lower bound */
|
||||||
while ( dwFreq < ((dwMCk / adwDivisors[dwIndex]) / 65536) )
|
while ( dwFreq < ((dwMCk / adwDivisors[dwIndex]) / 65536) )
|
||||||
{
|
{
|
||||||
dwIndex++ ;
|
dwIndex++ ;
|
||||||
|
|
||||||
/* If no divisor can be found, return 0 */
|
/* If no divisor can be found, return 0 */
|
||||||
if ( dwIndex == (sizeof( adwDivisors )/sizeof( adwDivisors[0] )) )
|
if ( dwIndex == (sizeof( adwDivisors )/sizeof( adwDivisors[0] )) )
|
||||||
{
|
{
|
||||||
return 0 ;
|
return 0 ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Try to maximize DIV while satisfying upper bound */
|
/* Try to maximize DIV while satisfying upper bound */
|
||||||
while ( dwIndex < 4 )
|
while ( dwIndex < 4 )
|
||||||
{
|
{
|
||||||
|
|
||||||
if ( dwFreq > (dwMCk / adwDivisors[dwIndex + 1]) )
|
if ( dwFreq > (dwMCk / adwDivisors[dwIndex + 1]) )
|
||||||
{
|
{
|
||||||
break ;
|
break ;
|
||||||
}
|
}
|
||||||
dwIndex++ ;
|
dwIndex++ ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Store results */
|
/* Store results */
|
||||||
if ( dwDiv )
|
if ( dwDiv )
|
||||||
{
|
{
|
||||||
*dwDiv = adwDivisors[dwIndex] ;
|
*dwDiv = adwDivisors[dwIndex] ;
|
||||||
}
|
}
|
||||||
if ( dwTcClks )
|
if ( dwTcClks )
|
||||||
{
|
{
|
||||||
*dwTcClks = dwIndex ;
|
*dwTcClks = dwIndex ;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1 ;
|
return 1 ;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,410 +1,410 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** \addtogroup usart_module Working with USART
|
/** \addtogroup usart_module Working with USART
|
||||||
* The USART driver provides the interface to configure and use the USART peripheral.\n
|
* The USART driver provides the interface to configure and use the USART peripheral.\n
|
||||||
*
|
*
|
||||||
* The USART supports several kinds of comminication modes such as full-duplex asynchronous/
|
* The USART supports several kinds of comminication modes such as full-duplex asynchronous/
|
||||||
* synchronous serial commnunication,RS485 with driver control signal,ISO7816,SPI and Test modes.
|
* synchronous serial commnunication,RS485 with driver control signal,ISO7816,SPI and Test modes.
|
||||||
*
|
*
|
||||||
* To start a USART transfer with \ref AT91SAM3S_PDC "PDC" support, the user could follow these steps:
|
* To start a USART transfer with \ref AT91SAM3S_PDC "PDC" support, the user could follow these steps:
|
||||||
* <ul>
|
* <ul>
|
||||||
* <li> Configure USART with expected mode and baudrate(see \ref USART_Configure), which could be done by:
|
* <li> Configure USART with expected mode and baudrate(see \ref USART_Configure), which could be done by:
|
||||||
* -# Resetting and disabling transmitter and receiver by setting US_CR(Control Register). </li>
|
* -# Resetting and disabling transmitter and receiver by setting US_CR(Control Register). </li>
|
||||||
* -# Conifguring the USART in a specific mode by setting USART_MODE bits in US_MR(Mode Register) </li>
|
* -# Conifguring the USART in a specific mode by setting USART_MODE bits in US_MR(Mode Register) </li>
|
||||||
* -# Setting baudrate which is different from mode to mode.
|
* -# Setting baudrate which is different from mode to mode.
|
||||||
</li>
|
</li>
|
||||||
* <li> Enable transmitter or receiver respectively by set US_CR_TXEN or US_CR_RXEN in US_CR.</li>
|
* <li> Enable transmitter or receiver respectively by set US_CR_TXEN or US_CR_RXEN in US_CR.</li>
|
||||||
* <li> Read from or write to the peripheral with \ref USART_ReadBuffer or \ref USART_WriteBuffer.
|
* <li> Read from or write to the peripheral with \ref USART_ReadBuffer or \ref USART_WriteBuffer.
|
||||||
These operations could be done by polling or interruption. </li>
|
These operations could be done by polling or interruption. </li>
|
||||||
* <li> For polling, check the status bit US_CSR_ENDRX/US_CSR_RXBUFF (READ) or US_CSR_ENDTX/
|
* <li> For polling, check the status bit US_CSR_ENDRX/US_CSR_RXBUFF (READ) or US_CSR_ENDTX/
|
||||||
US_CSR_TXBUFE (WRITE). </li>
|
US_CSR_TXBUFE (WRITE). </li>
|
||||||
* <li> For interruption,"enable" the status bit through US_IER and
|
* <li> For interruption,"enable" the status bit through US_IER and
|
||||||
realize the hanler with USARTx_IrqHandler according to IRQ vector
|
realize the hanler with USARTx_IrqHandler according to IRQ vector
|
||||||
table which is defined in board_cstartup_<toolchain>.c
|
table which is defined in board_cstartup_<toolchain>.c
|
||||||
To enable the interruption of USART,it should be configured with priority and enabled first through
|
To enable the interruption of USART,it should be configured with priority and enabled first through
|
||||||
NVIC .</li>
|
NVIC .</li>
|
||||||
* </ul>
|
* </ul>
|
||||||
*
|
*
|
||||||
* For more accurate information, please look at the USART section of the
|
* For more accurate information, please look at the USART section of the
|
||||||
* Datasheet.
|
* Datasheet.
|
||||||
*
|
*
|
||||||
* Related files :\n
|
* Related files :\n
|
||||||
* \ref usart.c\n
|
* \ref usart.c\n
|
||||||
* \ref usart.h\n
|
* \ref usart.h\n
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* Implementation of USART (Universal Synchronous Asynchronous Receiver Transmitter)
|
* Implementation of USART (Universal Synchronous Asynchronous Receiver Transmitter)
|
||||||
* controller.
|
* controller.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
#include "trace.h"
|
#include "trace.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Local definitions
|
* Local definitions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Configures an USART peripheral with the specified parameters.
|
* \brief Configures an USART peripheral with the specified parameters.
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* \param usart Pointer to the USART peripheral to configure.
|
* \param usart Pointer to the USART peripheral to configure.
|
||||||
* \param mode Desired value for the USART mode register (see the datasheet).
|
* \param mode Desired value for the USART mode register (see the datasheet).
|
||||||
* \param baudrate Baudrate at which the USART should operate (in Hz).
|
* \param baudrate Baudrate at which the USART should operate (in Hz).
|
||||||
* \param masterClock Frequency of the system master clock (in Hz).
|
* \param masterClock Frequency of the system master clock (in Hz).
|
||||||
*/
|
*/
|
||||||
void USART_Configure(Usart *usart,
|
void USART_Configure(Usart *usart,
|
||||||
uint32_t mode,
|
uint32_t mode,
|
||||||
uint32_t baudrate,
|
uint32_t baudrate,
|
||||||
uint32_t masterClock)
|
uint32_t masterClock)
|
||||||
{
|
{
|
||||||
/* Reset and disable receiver & transmitter*/
|
/* Reset and disable receiver & transmitter*/
|
||||||
usart->US_CR = US_CR_RSTRX | US_CR_RSTTX
|
usart->US_CR = US_CR_RSTRX | US_CR_RSTTX
|
||||||
| US_CR_RXDIS | US_CR_TXDIS;
|
| US_CR_RXDIS | US_CR_TXDIS;
|
||||||
|
|
||||||
/* Configure mode*/
|
/* Configure mode*/
|
||||||
usart->US_MR = mode;
|
usart->US_MR = mode;
|
||||||
|
|
||||||
/* Configure baudrate*/
|
/* Configure baudrate*/
|
||||||
/* Asynchronous, no oversampling*/
|
/* Asynchronous, no oversampling*/
|
||||||
if ( ((mode & US_MR_SYNC) == 0) && ((mode & US_MR_OVER) == 0) )
|
if ( ((mode & US_MR_SYNC) == 0) && ((mode & US_MR_OVER) == 0) )
|
||||||
{
|
{
|
||||||
usart->US_BRGR = (masterClock / baudrate) / 16;
|
usart->US_BRGR = (masterClock / baudrate) / 16;
|
||||||
}
|
}
|
||||||
|
|
||||||
if( ((mode & US_MR_USART_MODE_SPI_MASTER) == US_MR_USART_MODE_SPI_MASTER)
|
if( ((mode & US_MR_USART_MODE_SPI_MASTER) == US_MR_USART_MODE_SPI_MASTER)
|
||||||
|| ((mode & US_MR_SYNC) == US_MR_SYNC))
|
|| ((mode & US_MR_SYNC) == US_MR_SYNC))
|
||||||
{
|
{
|
||||||
if( (mode & US_MR_USCLKS_Msk) == US_MR_USCLKS_MCK)
|
if( (mode & US_MR_USCLKS_Msk) == US_MR_USCLKS_MCK)
|
||||||
{
|
{
|
||||||
usart->US_BRGR = masterClock / baudrate;
|
usart->US_BRGR = masterClock / baudrate;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ( (mode & US_MR_USCLKS_DIV) == US_MR_USCLKS_DIV)
|
if ( (mode & US_MR_USCLKS_DIV) == US_MR_USCLKS_DIV)
|
||||||
{
|
{
|
||||||
usart->US_BRGR = masterClock / baudrate / 8;
|
usart->US_BRGR = masterClock / baudrate / 8;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* TODO other modes*/
|
/* TODO other modes*/
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* \brief Enables or disables the transmitter of an USART peripheral.
|
* \brief Enables or disables the transmitter of an USART peripheral.
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral
|
* \param usart Pointer to an USART peripheral
|
||||||
* \param enabled If true, the transmitter is enabled; otherwise it is
|
* \param enabled If true, the transmitter is enabled; otherwise it is
|
||||||
* disabled.
|
* disabled.
|
||||||
*/
|
*/
|
||||||
void USART_SetTransmitterEnabled(Usart *usart, uint8_t enabled)
|
void USART_SetTransmitterEnabled(Usart *usart, uint8_t enabled)
|
||||||
{
|
{
|
||||||
if (enabled) {
|
if (enabled) {
|
||||||
|
|
||||||
usart->US_CR = US_CR_TXEN;
|
usart->US_CR = US_CR_TXEN;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
usart->US_CR = US_CR_TXDIS;
|
usart->US_CR = US_CR_TXDIS;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enables or disables the receiver of an USART peripheral
|
* \brief Enables or disables the receiver of an USART peripheral
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral
|
* \param usart Pointer to an USART peripheral
|
||||||
* \param enabled If true, the receiver is enabled; otherwise it is disabled.
|
* \param enabled If true, the receiver is enabled; otherwise it is disabled.
|
||||||
*/
|
*/
|
||||||
void USART_SetReceiverEnabled(Usart *usart,
|
void USART_SetReceiverEnabled(Usart *usart,
|
||||||
uint8_t enabled)
|
uint8_t enabled)
|
||||||
{
|
{
|
||||||
if (enabled) {
|
if (enabled) {
|
||||||
|
|
||||||
usart->US_CR = US_CR_RXEN;
|
usart->US_CR = US_CR_RXEN;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
usart->US_CR = US_CR_RXDIS;
|
usart->US_CR = US_CR_RXDIS;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sends one packet of data through the specified USART peripheral. This
|
* \brief Sends one packet of data through the specified USART peripheral. This
|
||||||
* function operates synchronously, so it only returns when the data has been
|
* function operates synchronously, so it only returns when the data has been
|
||||||
* actually sent.
|
* actually sent.
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral.
|
* \param usart Pointer to an USART peripheral.
|
||||||
* \param data Data to send including 9nth bit and sync field if necessary (in
|
* \param data Data to send including 9nth bit and sync field if necessary (in
|
||||||
* the same format as the US_THR register in the datasheet).
|
* the same format as the US_THR register in the datasheet).
|
||||||
* \param timeOut Time out value (0 = no timeout).
|
* \param timeOut Time out value (0 = no timeout).
|
||||||
*/
|
*/
|
||||||
void USART_Write(
|
void USART_Write(
|
||||||
Usart *usart,
|
Usart *usart,
|
||||||
uint16_t data,
|
uint16_t data,
|
||||||
volatile uint32_t timeOut)
|
volatile uint32_t timeOut)
|
||||||
{
|
{
|
||||||
if (timeOut == 0) {
|
if (timeOut == 0) {
|
||||||
|
|
||||||
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0);
|
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0) {
|
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0) {
|
||||||
|
|
||||||
if (timeOut == 0) {
|
if (timeOut == 0) {
|
||||||
|
|
||||||
TRACE_ERROR("USART_Write: Timed out.\n\r");
|
TRACE_ERROR("USART_Write: Timed out.\n\r");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
timeOut--;
|
timeOut--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
usart->US_THR = data;
|
usart->US_THR = data;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sends the contents of a data buffer through the specified USART peripheral.
|
* \brief Sends the contents of a data buffer through the specified USART peripheral.
|
||||||
* This function returns immediately (1 if the buffer has been queued, 0
|
* This function returns immediately (1 if the buffer has been queued, 0
|
||||||
* otherwise); poll the ENDTX and TXBUFE bits of the USART status register
|
* otherwise); poll the ENDTX and TXBUFE bits of the USART status register
|
||||||
* to check for the transfer completion.
|
* to check for the transfer completion.
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral.
|
* \param usart Pointer to an USART peripheral.
|
||||||
* \param buffer Pointer to the data buffer to send.
|
* \param buffer Pointer to the data buffer to send.
|
||||||
* \param size Size of the data buffer (in bytes).
|
* \param size Size of the data buffer (in bytes).
|
||||||
*/
|
*/
|
||||||
uint8_t USART_WriteBuffer(
|
uint8_t USART_WriteBuffer(
|
||||||
Usart *usart,
|
Usart *usart,
|
||||||
void *buffer,
|
void *buffer,
|
||||||
uint32_t size)
|
uint32_t size)
|
||||||
{
|
{
|
||||||
/* Check if the first PDC bank is free*/
|
/* Check if the first PDC bank is free*/
|
||||||
if ((usart->US_TCR == 0) && (usart->US_TNCR == 0)) {
|
if ((usart->US_TCR == 0) && (usart->US_TNCR == 0)) {
|
||||||
|
|
||||||
usart->US_TPR = (uint32_t) buffer;
|
usart->US_TPR = (uint32_t) buffer;
|
||||||
usart->US_TCR = size;
|
usart->US_TCR = size;
|
||||||
usart->US_PTCR = US_PTCR_TXTEN;
|
usart->US_PTCR = US_PTCR_TXTEN;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
/* Check if the second PDC bank is free*/
|
/* Check if the second PDC bank is free*/
|
||||||
else if (usart->US_TNCR == 0) {
|
else if (usart->US_TNCR == 0) {
|
||||||
|
|
||||||
usart->US_TNPR = (uint32_t) buffer;
|
usart->US_TNPR = (uint32_t) buffer;
|
||||||
usart->US_TNCR = size;
|
usart->US_TNCR = size;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Reads and return a packet of data on the specified USART peripheral. This
|
* \brief Reads and return a packet of data on the specified USART peripheral. This
|
||||||
* function operates asynchronously, so it waits until some data has been
|
* function operates asynchronously, so it waits until some data has been
|
||||||
* received.
|
* received.
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral.
|
* \param usart Pointer to an USART peripheral.
|
||||||
* \param timeOut Time out value (0 -> no timeout).
|
* \param timeOut Time out value (0 -> no timeout).
|
||||||
*/
|
*/
|
||||||
uint16_t USART_Read(
|
uint16_t USART_Read(
|
||||||
Usart *usart,
|
Usart *usart,
|
||||||
volatile uint32_t timeOut)
|
volatile uint32_t timeOut)
|
||||||
{
|
{
|
||||||
if (timeOut == 0) {
|
if (timeOut == 0) {
|
||||||
|
|
||||||
while ((usart->US_CSR & US_CSR_RXRDY) == 0);
|
while ((usart->US_CSR & US_CSR_RXRDY) == 0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
while ((usart->US_CSR & US_CSR_RXRDY) == 0) {
|
while ((usart->US_CSR & US_CSR_RXRDY) == 0) {
|
||||||
|
|
||||||
if (timeOut == 0) {
|
if (timeOut == 0) {
|
||||||
|
|
||||||
TRACE_ERROR( "USART_Read: Timed out.\n\r" ) ;
|
TRACE_ERROR( "USART_Read: Timed out.\n\r" ) ;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
timeOut--;
|
timeOut--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return usart->US_RHR;
|
return usart->US_RHR;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Reads data from an USART peripheral, filling the provided buffer until it
|
* \brief Reads data from an USART peripheral, filling the provided buffer until it
|
||||||
* becomes full. This function returns immediately with 1 if the buffer has
|
* becomes full. This function returns immediately with 1 if the buffer has
|
||||||
* been queued for transmission; otherwise 0.
|
* been queued for transmission; otherwise 0.
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral.
|
* \param usart Pointer to an USART peripheral.
|
||||||
* \param buffer Pointer to the buffer where the received data will be stored.
|
* \param buffer Pointer to the buffer where the received data will be stored.
|
||||||
* \param size Size of the data buffer (in bytes).
|
* \param size Size of the data buffer (in bytes).
|
||||||
*/
|
*/
|
||||||
uint8_t USART_ReadBuffer(Usart *usart,
|
uint8_t USART_ReadBuffer(Usart *usart,
|
||||||
void *buffer,
|
void *buffer,
|
||||||
uint32_t size)
|
uint32_t size)
|
||||||
{
|
{
|
||||||
/* Check if the first PDC bank is free*/
|
/* Check if the first PDC bank is free*/
|
||||||
if ((usart->US_RCR == 0) && (usart->US_RNCR == 0)) {
|
if ((usart->US_RCR == 0) && (usart->US_RNCR == 0)) {
|
||||||
|
|
||||||
usart->US_RPR = (uint32_t) buffer;
|
usart->US_RPR = (uint32_t) buffer;
|
||||||
usart->US_RCR = size;
|
usart->US_RCR = size;
|
||||||
usart->US_PTCR = US_PTCR_RXTEN;
|
usart->US_PTCR = US_PTCR_RXTEN;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
/* Check if the second PDC bank is free*/
|
/* Check if the second PDC bank is free*/
|
||||||
else if (usart->US_RNCR == 0) {
|
else if (usart->US_RNCR == 0) {
|
||||||
|
|
||||||
usart->US_RNPR = (uint32_t) buffer;
|
usart->US_RNPR = (uint32_t) buffer;
|
||||||
usart->US_RNCR = size;
|
usart->US_RNCR = size;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Returns 1 if some data has been received and can be read from an USART;
|
* \brief Returns 1 if some data has been received and can be read from an USART;
|
||||||
* otherwise returns 0.
|
* otherwise returns 0.
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an Usart instance.
|
* \param usart Pointer to an Usart instance.
|
||||||
*/
|
*/
|
||||||
uint8_t USART_IsDataAvailable(Usart *usart)
|
uint8_t USART_IsDataAvailable(Usart *usart)
|
||||||
{
|
{
|
||||||
if ((usart->US_CSR & US_CSR_RXRDY) != 0) {
|
if ((usart->US_CSR & US_CSR_RXRDY) != 0) {
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sets the filter value for the IRDA demodulator.
|
* \brief Sets the filter value for the IRDA demodulator.
|
||||||
*
|
*
|
||||||
* \param pUsart Pointer to an Usart instance.
|
* \param pUsart Pointer to an Usart instance.
|
||||||
* \param filter Filter value.
|
* \param filter Filter value.
|
||||||
*/
|
*/
|
||||||
void USART_SetIrdaFilter(Usart *pUsart, uint8_t filter)
|
void USART_SetIrdaFilter(Usart *pUsart, uint8_t filter)
|
||||||
{
|
{
|
||||||
assert( pUsart != NULL ) ;
|
assert( pUsart != NULL ) ;
|
||||||
|
|
||||||
pUsart->US_IF = filter;
|
pUsart->US_IF = filter;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Sends one packet of data through the specified USART peripheral. This
|
* \brief Sends one packet of data through the specified USART peripheral. This
|
||||||
* function operates synchronously, so it only returns when the data has been
|
* function operates synchronously, so it only returns when the data has been
|
||||||
* actually sent.
|
* actually sent.
|
||||||
*
|
*
|
||||||
* \param usart Pointer to an USART peripheral.
|
* \param usart Pointer to an USART peripheral.
|
||||||
* \param c Character to send
|
* \param c Character to send
|
||||||
*/
|
*/
|
||||||
void USART_PutChar(
|
void USART_PutChar(
|
||||||
Usart *usart,
|
Usart *usart,
|
||||||
uint8_t c)
|
uint8_t c)
|
||||||
{
|
{
|
||||||
/* Wait for the transmitter to be ready*/
|
/* Wait for the transmitter to be ready*/
|
||||||
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0);
|
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0);
|
||||||
|
|
||||||
/* Send character*/
|
/* Send character*/
|
||||||
usart->US_THR = c;
|
usart->US_THR = c;
|
||||||
|
|
||||||
/* Wait for the transfer to complete*/
|
/* Wait for the transfer to complete*/
|
||||||
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0);
|
while ((usart->US_CSR & US_CSR_TXEMPTY) == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Return 1 if a character can be read in USART
|
* \brief Return 1 if a character can be read in USART
|
||||||
*/
|
*/
|
||||||
uint32_t USART_IsRxReady(Usart *usart)
|
uint32_t USART_IsRxReady(Usart *usart)
|
||||||
{
|
{
|
||||||
return (usart->US_CSR & US_CSR_RXRDY);
|
return (usart->US_CSR & US_CSR_RXRDY);
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* \brief Get present status
|
* \brief Get present status
|
||||||
*/
|
*/
|
||||||
uint32_t USART_GetStatus(Usart *usart)
|
uint32_t USART_GetStatus(Usart *usart)
|
||||||
{
|
{
|
||||||
return usart->US_CSR;
|
return usart->US_CSR;
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* \brief Enable interrupt
|
* \brief Enable interrupt
|
||||||
*/
|
*/
|
||||||
void USART_EnableIt(Usart *usart,uint32_t mode)
|
void USART_EnableIt(Usart *usart,uint32_t mode)
|
||||||
{
|
{
|
||||||
usart->US_IER = mode;
|
usart->US_IER = mode;
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* \brief Disable interrupt
|
* \brief Disable interrupt
|
||||||
*/
|
*/
|
||||||
void USART_DisableIt(Usart *usart,uint32_t mode)
|
void USART_DisableIt(Usart *usart,uint32_t mode)
|
||||||
{
|
{
|
||||||
usart->US_IDR = mode;
|
usart->US_IDR = mode;
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* \brief Reads and returns a character from the USART.
|
* \brief Reads and returns a character from the USART.
|
||||||
*
|
*
|
||||||
* \note This function is synchronous (i.e. uses polling).
|
* \note This function is synchronous (i.e. uses polling).
|
||||||
* \param usart Pointer to an USART peripheral.
|
* \param usart Pointer to an USART peripheral.
|
||||||
* \return Character received.
|
* \return Character received.
|
||||||
*/
|
*/
|
||||||
uint8_t USART_GetChar(Usart *usart)
|
uint8_t USART_GetChar(Usart *usart)
|
||||||
{
|
{
|
||||||
while ((usart->US_CSR & US_CSR_RXRDY) == 0);
|
while ((usart->US_CSR & US_CSR_RXRDY) == 0);
|
||||||
return usart->US_RHR;
|
return usart->US_RHR;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,132 +1,132 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* modification, are permitted provided that the following conditions are met:
|
||||||
*
|
*
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* this list of conditions and the disclaimer below.
|
* this list of conditions and the disclaimer below.
|
||||||
*
|
*
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* this software without specific prior written permission.
|
* this software without specific prior written permission.
|
||||||
*
|
*
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
* Implementation of Watchdog Timer (WDT) controller.
|
* Implementation of Watchdog Timer (WDT) controller.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** \addtogroup wdt_module Working with WDT
|
/** \addtogroup wdt_module Working with WDT
|
||||||
* The WDT driver provides the interface to configure and use the WDT
|
* The WDT driver provides the interface to configure and use the WDT
|
||||||
* peripheral.
|
* peripheral.
|
||||||
*
|
*
|
||||||
* The WDT can be used to prevent system lock-up if the software becomes
|
* The WDT can be used to prevent system lock-up if the software becomes
|
||||||
* trapped in a deadlock. It can generate a general reset or a processor
|
* trapped in a deadlock. It can generate a general reset or a processor
|
||||||
* reset only. It is clocked by slow clock divided by 128.
|
* reset only. It is clocked by slow clock divided by 128.
|
||||||
*
|
*
|
||||||
* The WDT is running at reset with 16 seconds watchdog period (slow clock at 32.768 kHz)
|
* The WDT is running at reset with 16 seconds watchdog period (slow clock at 32.768 kHz)
|
||||||
* and external reset generation enabled. The user must either disable it or
|
* and external reset generation enabled. The user must either disable it or
|
||||||
* reprogram it to meet the application requires.
|
* reprogram it to meet the application requires.
|
||||||
*
|
*
|
||||||
* To use the WDT, the user could follow these few steps:
|
* To use the WDT, the user could follow these few steps:
|
||||||
* <ul>
|
* <ul>
|
||||||
* <li>Enable watchdog with given mode using \ref WDT_Enable().
|
* <li>Enable watchdog with given mode using \ref WDT_Enable().
|
||||||
* <li>Restart the watchdog using \ref WDT_Restart() within the watchdog period.
|
* <li>Restart the watchdog using \ref WDT_Restart() within the watchdog period.
|
||||||
* </ul>
|
* </ul>
|
||||||
*
|
*
|
||||||
* For more accurate information, please look at the WDT section of the
|
* For more accurate information, please look at the WDT section of the
|
||||||
* Datasheet.
|
* Datasheet.
|
||||||
*
|
*
|
||||||
* \note
|
* \note
|
||||||
* The Watchdog Mode Register (WDT_MR) can be written only once.\n
|
* The Watchdog Mode Register (WDT_MR) can be written only once.\n
|
||||||
*
|
*
|
||||||
* Related files :\n
|
* Related files :\n
|
||||||
* \ref wdt.c\n
|
* \ref wdt.c\n
|
||||||
* \ref wdt.h.\n
|
* \ref wdt.h.\n
|
||||||
*/
|
*/
|
||||||
/*@{*/
|
/*@{*/
|
||||||
/*@}*/
|
/*@}*/
|
||||||
|
|
||||||
/*---------------------------------------------------------------------------
|
/*---------------------------------------------------------------------------
|
||||||
* Headers
|
* Headers
|
||||||
*---------------------------------------------------------------------------*/
|
*---------------------------------------------------------------------------*/
|
||||||
|
|
||||||
#include "chip.h"
|
#include "chip.h"
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
* Exported functions
|
* Exported functions
|
||||||
*----------------------------------------------------------------------------*/
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Enable watchdog with given mode.
|
* \brief Enable watchdog with given mode.
|
||||||
*
|
*
|
||||||
* \note The Watchdog Mode Register (WDT_MR) can be written only once.
|
* \note The Watchdog Mode Register (WDT_MR) can be written only once.
|
||||||
* Only a processor reset resets it.
|
* Only a processor reset resets it.
|
||||||
*
|
*
|
||||||
* \param dwMode WDT mode to be set
|
* \param dwMode WDT mode to be set
|
||||||
*/
|
*/
|
||||||
extern void WDT_Enable( Wdt* pWDT, uint32_t dwMode )
|
extern void WDT_Enable( Wdt* pWDT, uint32_t dwMode )
|
||||||
{
|
{
|
||||||
pWDT->WDT_MR = dwMode ;
|
pWDT->WDT_MR = dwMode ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Disable watchdog.
|
* \brief Disable watchdog.
|
||||||
*
|
*
|
||||||
* \note The Watchdog Mode Register (WDT_MR) can be written only once.
|
* \note The Watchdog Mode Register (WDT_MR) can be written only once.
|
||||||
* Only a processor reset resets it.
|
* Only a processor reset resets it.
|
||||||
*/
|
*/
|
||||||
extern void WDT_Disable( Wdt* pWDT )
|
extern void WDT_Disable( Wdt* pWDT )
|
||||||
{
|
{
|
||||||
pWDT->WDT_MR = WDT_MR_WDDIS;
|
pWDT->WDT_MR = WDT_MR_WDDIS;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Watchdog restart.
|
* \brief Watchdog restart.
|
||||||
*/
|
*/
|
||||||
extern void WDT_Restart( Wdt* pWDT )
|
extern void WDT_Restart( Wdt* pWDT )
|
||||||
{
|
{
|
||||||
pWDT->WDT_CR = 0xA5000001;
|
pWDT->WDT_CR = 0xA5000001;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Watchdog get status.
|
* \brief Watchdog get status.
|
||||||
*/
|
*/
|
||||||
extern uint32_t WDT_GetStatus( Wdt* pWDT )
|
extern uint32_t WDT_GetStatus( Wdt* pWDT )
|
||||||
{
|
{
|
||||||
return (pWDT->WDT_SR & 0x3) ;
|
return (pWDT->WDT_SR & 0x3) ;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief Watchdog get period.
|
* \brief Watchdog get period.
|
||||||
*
|
*
|
||||||
* \param dwMs desired watchdog period in millisecond.
|
* \param dwMs desired watchdog period in millisecond.
|
||||||
*/
|
*/
|
||||||
extern uint32_t WDT_GetPeriod( uint32_t dwMs )
|
extern uint32_t WDT_GetPeriod( uint32_t dwMs )
|
||||||
{
|
{
|
||||||
if ( (dwMs < 4) || (dwMs > 16000) )
|
if ( (dwMs < 4) || (dwMs > 16000) )
|
||||||
{
|
{
|
||||||
return 0 ;
|
return 0 ;
|
||||||
}
|
}
|
||||||
return ((dwMs << 8) / 1000) ;
|
return ((dwMs << 8) / 1000) ;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -47,6 +47,8 @@
|
|||||||
#include "USBD.h"
|
#include "USBD.h"
|
||||||
#include "USBD_HAL.h"
|
#include "USBD_HAL.h"
|
||||||
|
|
||||||
|
extern void USBDFU_Runtime_RequestHandler(const USBGenericRequest *request);
|
||||||
|
|
||||||
/*---------------------------------------------------------------------------
|
/*---------------------------------------------------------------------------
|
||||||
* Definitions
|
* Definitions
|
||||||
*---------------------------------------------------------------------------*/
|
*---------------------------------------------------------------------------*/
|
||||||
@@ -144,7 +146,11 @@ void USBD_RequestHandler(uint8_t bEndpoint,
|
|||||||
bEndpoint);
|
bEndpoint);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
#if defined(BOARD_USB_DFU) && !defined(APPLICATION_dfu)
|
||||||
|
USBDFU_Runtime_RequestHandler(pRequest);
|
||||||
|
#else
|
||||||
USBDCallbacks_RequestReceived(pRequest);
|
USBDCallbacks_RequestReceived(pRequest);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2008, Atmel Corporation
|
* Copyright (c) 2008, Atmel Corporation
|
||||||
|
* Copyright (c) 2018, Kevin Redon <kredon@sysmocom.de>
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
@@ -331,8 +332,13 @@ static void GetDescriptor(
|
|||||||
/* Check if descriptor exists */
|
/* Check if descriptor exists */
|
||||||
|
|
||||||
if (indexRDesc >= numStrings) {
|
if (indexRDesc >= numStrings) {
|
||||||
|
/* Sometimes descriptor string 0xee is requested.
|
||||||
USBD_Stall(0);
|
* This is a mechanism used by Microsoft Windows to further identify the USB device.
|
||||||
|
* Instead of stalling, as is the original code, leading to an USB reset, we send an empty packet.
|
||||||
|
* I am not sure if sending an empty string would be better, but an empty packet seems sufficient.
|
||||||
|
*/
|
||||||
|
//USBD_Stall(0);
|
||||||
|
USBD_Write(0, NULL, 0, 0, 0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
|
||||||
|
|||||||
@@ -78,8 +78,8 @@ extern const USBDDriverDescriptors dfu_descriptors;
|
|||||||
|
|
||||||
/* no DFU bootloader is being used */
|
/* no DFU bootloader is being used */
|
||||||
#define DFURT_NUM_IF 0
|
#define DFURT_NUM_IF 0
|
||||||
#define DFURT_IF_DESCRIPTOR_STRUCT(a, b)
|
#define DFURT_IF_DESCRIPTOR_STRUCT
|
||||||
#define DFURT_IF_DESCRIPTOR
|
#define DFURT_IF_DESCRIPTOR(a, b)
|
||||||
|
|
||||||
#endif /* BOARD_USB_DFU */
|
#endif /* BOARD_USB_DFU */
|
||||||
|
|
||||||
@@ -101,7 +101,7 @@ struct dfudata {
|
|||||||
extern struct dfudata _g_dfu;
|
extern struct dfudata _g_dfu;
|
||||||
extern struct dfudata *g_dfu;
|
extern struct dfudata *g_dfu;
|
||||||
|
|
||||||
void set_usb_serial_str(const uint8_t *serial_usbstr);
|
void set_usb_serial_str(void);
|
||||||
|
|
||||||
void DFURT_SwitchToDFU(void);
|
void DFURT_SwitchToDFU(void);
|
||||||
|
|
||||||
|
|||||||
@@ -13,14 +13,87 @@
|
|||||||
#include <usb/common/dfu/usb_dfu.h>
|
#include <usb/common/dfu/usb_dfu.h>
|
||||||
#include <usb/device/dfu/dfu.h>
|
#include <usb/device/dfu/dfu.h>
|
||||||
|
|
||||||
|
#include "usb_strings_generated.h"
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
STR_MANUF = 1,
|
STR_MANUF = 1,
|
||||||
STR_PROD,
|
STR_PROD,
|
||||||
STR_CONFIG,
|
STR_CONFIG,
|
||||||
_STR_FIRST_ALT,
|
_STR_FIRST_ALT,
|
||||||
|
// serial string
|
||||||
STR_SERIAL = (_STR_FIRST_ALT+BOARD_DFU_NUM_IF),
|
STR_SERIAL = (_STR_FIRST_ALT+BOARD_DFU_NUM_IF),
|
||||||
|
// version string (on additional interface)
|
||||||
|
VERSION_CONF_STR,
|
||||||
|
VERSION_STR,
|
||||||
|
// count
|
||||||
|
STRING_DESC_CNT,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* USB string for the serial (using 128-bit device ID) */
|
||||||
|
static unsigned char usb_string_serial[] = {
|
||||||
|
USBStringDescriptor_LENGTH(32),
|
||||||
|
USBGenericDescriptor_STRING,
|
||||||
|
USBStringDescriptor_UNICODE('0'),
|
||||||
|
USBStringDescriptor_UNICODE('0'),
|
||||||
|
USBStringDescriptor_UNICODE('1'),
|
||||||
|
USBStringDescriptor_UNICODE('1'),
|
||||||
|
USBStringDescriptor_UNICODE('2'),
|
||||||
|
USBStringDescriptor_UNICODE('2'),
|
||||||
|
USBStringDescriptor_UNICODE('3'),
|
||||||
|
USBStringDescriptor_UNICODE('3'),
|
||||||
|
USBStringDescriptor_UNICODE('4'),
|
||||||
|
USBStringDescriptor_UNICODE('4'),
|
||||||
|
USBStringDescriptor_UNICODE('5'),
|
||||||
|
USBStringDescriptor_UNICODE('5'),
|
||||||
|
USBStringDescriptor_UNICODE('6'),
|
||||||
|
USBStringDescriptor_UNICODE('6'),
|
||||||
|
USBStringDescriptor_UNICODE('7'),
|
||||||
|
USBStringDescriptor_UNICODE('7'),
|
||||||
|
USBStringDescriptor_UNICODE('8'),
|
||||||
|
USBStringDescriptor_UNICODE('8'),
|
||||||
|
USBStringDescriptor_UNICODE('9'),
|
||||||
|
USBStringDescriptor_UNICODE('9'),
|
||||||
|
USBStringDescriptor_UNICODE('a'),
|
||||||
|
USBStringDescriptor_UNICODE('a'),
|
||||||
|
USBStringDescriptor_UNICODE('b'),
|
||||||
|
USBStringDescriptor_UNICODE('b'),
|
||||||
|
USBStringDescriptor_UNICODE('c'),
|
||||||
|
USBStringDescriptor_UNICODE('c'),
|
||||||
|
USBStringDescriptor_UNICODE('d'),
|
||||||
|
USBStringDescriptor_UNICODE('d'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE('f'),
|
||||||
|
USBStringDescriptor_UNICODE('f'),
|
||||||
|
};
|
||||||
|
|
||||||
|
/* USB string for the version */
|
||||||
|
static const unsigned char usb_string_version_conf[] = {
|
||||||
|
USBStringDescriptor_LENGTH(16),
|
||||||
|
USBGenericDescriptor_STRING,
|
||||||
|
USBStringDescriptor_UNICODE('f'),
|
||||||
|
USBStringDescriptor_UNICODE('i'),
|
||||||
|
USBStringDescriptor_UNICODE('r'),
|
||||||
|
USBStringDescriptor_UNICODE('m'),
|
||||||
|
USBStringDescriptor_UNICODE('w'),
|
||||||
|
USBStringDescriptor_UNICODE('a'),
|
||||||
|
USBStringDescriptor_UNICODE('r'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE(' '),
|
||||||
|
USBStringDescriptor_UNICODE('v'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE('r'),
|
||||||
|
USBStringDescriptor_UNICODE('s'),
|
||||||
|
USBStringDescriptor_UNICODE('i'),
|
||||||
|
USBStringDescriptor_UNICODE('o'),
|
||||||
|
USBStringDescriptor_UNICODE('n'),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const char git_version[] = GIT_VERSION;
|
||||||
|
static unsigned char usb_string_version[2 + ARRAY_SIZE(git_version) * 2 - 2];
|
||||||
|
/** array of static (from usb_strings) and runtime (serial, version) USB strings */
|
||||||
|
static const unsigned char *usb_strings_extended[STRING_DESC_CNT];
|
||||||
|
|
||||||
static const USBDeviceDescriptor fsDevice = {
|
static const USBDeviceDescriptor fsDevice = {
|
||||||
.bLength = sizeof(USBDeviceDescriptor),
|
.bLength = sizeof(USBDeviceDescriptor),
|
||||||
.bDescriptorType = USBGenericDescriptor_DEVICE,
|
.bDescriptorType = USBGenericDescriptor_DEVICE,
|
||||||
@@ -28,18 +101,14 @@ static const USBDeviceDescriptor fsDevice = {
|
|||||||
.bDeviceClass = 0,
|
.bDeviceClass = 0,
|
||||||
.bDeviceSubClass = 0,
|
.bDeviceSubClass = 0,
|
||||||
.bDeviceProtocol = 0,
|
.bDeviceProtocol = 0,
|
||||||
.bMaxPacketSize0 = BOARD_USB_ENDPOINTS_MAXPACKETSIZE(0),
|
.bMaxPacketSize0 = USBEndpointDescriptor_MAXCTRLSIZE_FS,
|
||||||
.idVendor = BOARD_USB_VENDOR_ID,
|
.idVendor = BOARD_USB_VENDOR_ID,
|
||||||
.idProduct = BOARD_DFU_USB_PRODUCT_ID,
|
.idProduct = BOARD_DFU_USB_PRODUCT_ID,
|
||||||
.bcdDevice = BOARD_USB_RELEASE,
|
.bcdDevice = BOARD_USB_RELEASE,
|
||||||
.iManufacturer = STR_MANUF,
|
.iManufacturer = STR_MANUF,
|
||||||
.iProduct = STR_PROD,
|
.iProduct = STR_PROD,
|
||||||
#ifdef BOARD_USB_SERIAL
|
|
||||||
.iSerialNumber = STR_SERIAL,
|
.iSerialNumber = STR_SERIAL,
|
||||||
#else
|
.bNumConfigurations = 2, // DFU + version configurations
|
||||||
.iSerialNumber = 0,
|
|
||||||
#endif
|
|
||||||
.bNumConfigurations = 1,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Alternate Interface Descriptor, we use one per partition/memory type */
|
/* Alternate Interface Descriptor, we use one per partition/memory type */
|
||||||
@@ -85,17 +154,74 @@ const struct dfu_desc dfu_cfg_descriptor = {
|
|||||||
.func_dfu = DFU_FUNC_DESC
|
.func_dfu = DFU_FUNC_DESC
|
||||||
};
|
};
|
||||||
|
|
||||||
#include "usb_strings_generated.h"
|
void set_usb_serial_str(void)
|
||||||
|
|
||||||
#if 0
|
|
||||||
void set_usb_serial_str(const uint8_t *serial_usbstr)
|
|
||||||
{
|
{
|
||||||
usb_strings[STR_SERIAL] = serial_usbstr;
|
unsigned int i;
|
||||||
|
|
||||||
|
// put device ID into USB serial number description
|
||||||
|
unsigned int device_id[4];
|
||||||
|
EEFC_ReadUniqueID(device_id);
|
||||||
|
char device_id_string[32 + 1];
|
||||||
|
snprintf(device_id_string, ARRAY_SIZE(device_id_string), "%08x%08x%08x%08x",
|
||||||
|
device_id[0], device_id[1], device_id[2], device_id[3]);
|
||||||
|
for (i = 0; i < ARRAY_SIZE(device_id_string) - 1; i++) {
|
||||||
|
usb_string_serial[2 + 2 * i] = device_id_string[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// put version into USB string
|
||||||
|
usb_string_version[0] = USBStringDescriptor_LENGTH(ARRAY_SIZE(git_version) - 1);
|
||||||
|
usb_string_version[1] = USBGenericDescriptor_STRING;
|
||||||
|
for (i = 0; i < ARRAY_SIZE(git_version) - 1; i++) {
|
||||||
|
usb_string_version[2 + i * 2 + 0] = git_version[i];
|
||||||
|
usb_string_version[2 + i * 2 + 1] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// fill extended USB strings
|
||||||
|
for (i = 0; i < ARRAY_SIZE(usb_strings) && i < ARRAY_SIZE(usb_strings_extended); i++) {
|
||||||
|
usb_strings_extended[i] = usb_strings[i];
|
||||||
|
}
|
||||||
|
usb_strings_extended[STR_SERIAL] = usb_string_serial;
|
||||||
|
usb_strings_extended[VERSION_CONF_STR] = usb_string_version_conf;
|
||||||
|
usb_strings_extended[VERSION_STR] = usb_string_version;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
/* USB descriptor just to show the version */
|
||||||
|
typedef struct _SIMTraceDriverConfigurationDescriptorVersion {
|
||||||
|
/** Standard configuration descriptor. */
|
||||||
|
USBConfigurationDescriptor configuration;
|
||||||
|
USBInterfaceDescriptor version;
|
||||||
|
} __attribute__ ((packed)) SIMTraceDriverConfigurationDescriptorVersion;
|
||||||
|
|
||||||
|
static const SIMTraceDriverConfigurationDescriptorVersion
|
||||||
|
configurationDescriptorVersion = {
|
||||||
|
/* Standard configuration descriptor for the interface descriptor*/
|
||||||
|
.configuration = {
|
||||||
|
.bLength = sizeof(USBConfigurationDescriptor),
|
||||||
|
.bDescriptorType = USBGenericDescriptor_CONFIGURATION,
|
||||||
|
.wTotalLength = sizeof(SIMTraceDriverConfigurationDescriptorVersion),
|
||||||
|
.bNumInterfaces = 1,
|
||||||
|
.bConfigurationValue = 2,
|
||||||
|
.iConfiguration = VERSION_CONF_STR,
|
||||||
|
.bmAttributes = USBD_BMATTRIBUTES,
|
||||||
|
.bMaxPower = USBConfigurationDescriptor_POWER(100),
|
||||||
|
},
|
||||||
|
/* Interface standard descriptor just holding the version information */
|
||||||
|
.version = {
|
||||||
|
.bLength = sizeof(USBInterfaceDescriptor),
|
||||||
|
.bDescriptorType = USBGenericDescriptor_INTERFACE,
|
||||||
|
.bInterfaceNumber = 0,
|
||||||
|
.bAlternateSetting = 0,
|
||||||
|
.bNumEndpoints = 0,
|
||||||
|
.bInterfaceClass = USB_CLASS_PROPRIETARY,
|
||||||
|
.bInterfaceSubClass = 0xff,
|
||||||
|
.bInterfaceProtocol = 0,
|
||||||
|
.iInterface = VERSION_STR,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
static const USBConfigurationDescriptor *conf_desc_arr[] = {
|
static const USBConfigurationDescriptor *conf_desc_arr[] = {
|
||||||
&dfu_cfg_descriptor.ucfg,
|
&dfu_cfg_descriptor.ucfg,
|
||||||
|
&configurationDescriptorVersion.configuration,
|
||||||
};
|
};
|
||||||
|
|
||||||
const USBDDriverDescriptors dfu_descriptors = {
|
const USBDDriverDescriptors dfu_descriptors = {
|
||||||
@@ -108,6 +234,6 @@ const USBDDriverDescriptors dfu_descriptors = {
|
|||||||
.pHsConfiguration = NULL,
|
.pHsConfiguration = NULL,
|
||||||
.pHsQualifier = NULL,
|
.pHsQualifier = NULL,
|
||||||
.pHsOtherSpeed = NULL,
|
.pHsOtherSpeed = NULL,
|
||||||
.pStrings = usb_strings,
|
.pStrings = usb_strings_extended,
|
||||||
.numStrings = ARRAY_SIZE(usb_strings),
|
.numStrings = ARRAY_SIZE(usb_strings_extended),
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -217,7 +217,7 @@ void USBDFU_DFU_RequestHandler(const USBGenericRequest *request)
|
|||||||
uint8_t req = USBGenericRequest_GetRequest(request);
|
uint8_t req = USBGenericRequest_GetRequest(request);
|
||||||
uint16_t len = USBGenericRequest_GetLength(request);
|
uint16_t len = USBGenericRequest_GetLength(request);
|
||||||
uint16_t val = USBGenericRequest_GetValue(request);
|
uint16_t val = USBGenericRequest_GetValue(request);
|
||||||
int rc, ret;
|
int rc, ret = DFU_RET_NOTHING;
|
||||||
|
|
||||||
TRACE_DEBUG("type=0x%x, recipient=0x%x val=0x%x len=%u\n\r",
|
TRACE_DEBUG("type=0x%x, recipient=0x%x val=0x%x len=%u\n\r",
|
||||||
USBGenericRequest_GetType(request),
|
USBGenericRequest_GetType(request),
|
||||||
@@ -447,6 +447,7 @@ void USBDFU_Initialize(const USBDDriverDescriptors *pDescriptors)
|
|||||||
/* We already start in DFU idle mode */
|
/* We already start in DFU idle mode */
|
||||||
g_dfu->state = DFU_STATE_dfuIDLE;
|
g_dfu->state = DFU_STATE_dfuIDLE;
|
||||||
|
|
||||||
|
set_usb_serial_str();
|
||||||
USBDDriver_Initialize(&usbdDriver, pDescriptors, if_altsettings);
|
USBDDriver_Initialize(&usbdDriver, pDescriptors, if_altsettings);
|
||||||
USBD_Init();
|
USBD_Init();
|
||||||
USBD_Connect();
|
USBD_Connect();
|
||||||
|
|||||||
@@ -107,7 +107,7 @@ void USBDFU_Runtime_RequestHandler(const USBGenericRequest *request)
|
|||||||
uint8_t req = USBGenericRequest_GetRequest(request);
|
uint8_t req = USBGenericRequest_GetRequest(request);
|
||||||
uint16_t len = USBGenericRequest_GetLength(request);
|
uint16_t len = USBGenericRequest_GetLength(request);
|
||||||
uint16_t val = USBGenericRequest_GetValue(request);
|
uint16_t val = USBGenericRequest_GetValue(request);
|
||||||
int rc, ret;
|
int rc, ret = DFU_RET_NOTHING;
|
||||||
|
|
||||||
TRACE_DEBUG("type=0x%x, recipient=0x%x val=0x%x len=%u\n\r",
|
TRACE_DEBUG("type=0x%x, recipient=0x%x val=0x%x len=%u\n\r",
|
||||||
USBGenericRequest_GetType(request),
|
USBGenericRequest_GetType(request),
|
||||||
@@ -141,7 +141,7 @@ void USBDFU_Runtime_RequestHandler(const USBGenericRequest *request)
|
|||||||
if (USBGenericRequest_GetType(request) != USBGenericRequest_CLASS ||
|
if (USBGenericRequest_GetType(request) != USBGenericRequest_CLASS ||
|
||||||
USBGenericRequest_GetRecipient(request) != USBGenericRequest_INTERFACE) {
|
USBGenericRequest_GetRecipient(request) != USBGenericRequest_INTERFACE) {
|
||||||
TRACE_DEBUG("std_ho_usbd ");
|
TRACE_DEBUG("std_ho_usbd ");
|
||||||
USBDDriver_RequestHandler(usbdDriver, request);
|
USBDCallbacks_RequestReceived(request);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -216,9 +216,3 @@ void DFURT_SwitchToDFU(void)
|
|||||||
* ResetVector of the bootloader */
|
* ResetVector of the bootloader */
|
||||||
NVIC_SystemReset();
|
NVIC_SystemReset();
|
||||||
}
|
}
|
||||||
|
|
||||||
void USBDCallbacks_RequestReceived(const USBGenericRequest *request)
|
|
||||||
{
|
|
||||||
/* FIXME: integration with CCID control point reqeusts */
|
|
||||||
USBDFU_Runtime_RequestHandler(request);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,3 +1,22 @@
|
|||||||
|
/* SIMtrace 2 common board pin definitions
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#ifndef _BOARD_
|
#ifndef _BOARD_
|
||||||
#define _BOARD_
|
#define _BOARD_
|
||||||
|
|
||||||
@@ -37,18 +56,12 @@
|
|||||||
/** Core definition */
|
/** Core definition */
|
||||||
#define cortexm3
|
#define cortexm3
|
||||||
|
|
||||||
#define BOARD_MCK 48000000
|
/* LEDs are used to indicate the status
|
||||||
|
* the LED definition is board specific
|
||||||
#define PIO_LED_RED PIO_PA17
|
* most boards have two LEDs, one green and one red
|
||||||
#define PIO_LED_GREEN PIO_PA17
|
* the red LED indicates of the main firmware is ready (on) or if there is an error (blinking)
|
||||||
|
* the green LED indicates if the firmware is idling (on) or if there is activity (blinking)
|
||||||
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
*/
|
||||||
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
|
||||||
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
|
||||||
|
|
||||||
#define LED_NUM_RED 0
|
|
||||||
#define LED_NUM_GREEN 1
|
|
||||||
|
|
||||||
/** USART0 pin RX */
|
/** USART0 pin RX */
|
||||||
#define PIN_USART0_RXD {PIO_PA9A_URXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
#define PIN_USART0_RXD {PIO_PA9A_URXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
/** USART0 pin TX */
|
/** USART0 pin TX */
|
||||||
@@ -63,12 +76,16 @@
|
|||||||
#define PINS_UART { PIO_PA9A_URXD0|PIO_PA10A_UTXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
#define PINS_UART { PIO_PA9A_URXD0|PIO_PA10A_UTXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
|
||||||
/** UART0 */
|
/** UART0 */
|
||||||
/** Console baudrate always using 115200. */
|
/** Console baud rate in bps */
|
||||||
#define CONSOLE_BAUDRATE 115200
|
#define CONSOLE_BAUDRATE 921600
|
||||||
/** Usart Hw interface used by the console (UART0). */
|
/** UART peripheral used by the console (UART0). */
|
||||||
#define CONSOLE_USART UART0
|
#define CONSOLE_UART UART0
|
||||||
/** Usart Hw ID used by the console (UART0). */
|
/** UART peripheral ID used by the console (UART0). */
|
||||||
#define CONSOLE_ID ID_UART0
|
#define CONSOLE_ID ID_UART0
|
||||||
|
/** UART ISR used by the console (UART0). */
|
||||||
|
#define CONSOLE_ISR UART0_IrqHandler
|
||||||
|
/** UART IRQ used by the console (UART0). */
|
||||||
|
#define CONSOLE_IRQ UART0_IRQn
|
||||||
/** Pins description corresponding to Rxd,Txd, (UART pins) */
|
/** Pins description corresponding to Rxd,Txd, (UART pins) */
|
||||||
#define CONSOLE_PINS {PINS_UART}
|
#define CONSOLE_PINS {PINS_UART}
|
||||||
|
|
||||||
@@ -77,39 +94,22 @@
|
|||||||
#define BOARD_ISO7816_BASE_USART USART0
|
#define BOARD_ISO7816_BASE_USART USART0
|
||||||
#define BOARD_ISO7816_ID_USART ID_USART0
|
#define BOARD_ISO7816_ID_USART ID_USART0
|
||||||
|
|
||||||
|
/* USART peripherals for a phone and SIM card setup */
|
||||||
|
/* USART peripheral connected to the SIM card */
|
||||||
#define USART_SIM USART0
|
#define USART_SIM USART0
|
||||||
|
/* ID of USART peripheral connected to the SIM card */
|
||||||
#define ID_USART_SIM ID_USART0
|
#define ID_USART_SIM ID_USART0
|
||||||
#define USART_PHONE USART1
|
/* Interrupt request ID of USART peripheral connected to the SIM card */
|
||||||
#define ID_USART_PHONE ID_USART1
|
#define IRQ_USART_SIM USART0_IRQn
|
||||||
|
/* USART peripheral connected to the phone */
|
||||||
#define SIM_PWEN PIO_PA5
|
#define USART_PHONE USART1
|
||||||
#define VCC_FWD PIO_PA26
|
/* ID of USART peripheral connected to the phone */
|
||||||
|
#define ID_USART_PHONE ID_USART1
|
||||||
|
/* Interrupt request ID of USART peripheral connected to the phone */
|
||||||
//** USB **/
|
#define IRQ_USART_PHONE USART1_IRQn
|
||||||
// USB pull-up control pin definition (PA16).
|
|
||||||
// Default: 1 (USB Pullup deactivated)
|
|
||||||
#define PIN_USB_PULLUP {1 << 16, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
|
||||||
|
|
||||||
// Board has UDP controller
|
// Board has UDP controller
|
||||||
#define BOARD_USB_UDP
|
#define BOARD_USB_UDP
|
||||||
// D+ has external pull-up
|
|
||||||
#define BOARD_USB_PULLUP_EXTERNAL
|
|
||||||
|
|
||||||
#define BOARD_USB_NUMENDPOINTS 8
|
|
||||||
|
|
||||||
// FIXME: in all other cases return 0?
|
|
||||||
#define BOARD_USB_ENDPOINTS_MAXPACKETSIZE(i) (((i == 4) || (i == 5))? 512 : 64)
|
|
||||||
#define BOARD_USB_ENDPOINTS_BANKS(i) (((i == 0) || (i == 3)) ? 1 : 2)
|
|
||||||
|
|
||||||
#define USB_VENDOR_OPENMOKO 0x1d50
|
|
||||||
#define USB_PRODUCT_OWHW_SAM3_DFU 0x4001 /* was 0x4000 */
|
|
||||||
#define USB_PRODUCT_OWHW_SAM3 0x4001
|
|
||||||
#define USB_PRODUCT_QMOD_HUB 0x4002
|
|
||||||
#define USB_PRODUCT_QMOD_SAM3_DFU 0x4004 /* was 0x4003 */
|
|
||||||
#define USB_PRODUCT_QMOD_SAM3 0x4004
|
|
||||||
#define USB_PRODUCT_SIMTRACE2_DFU 0x60e3 /* was 0x60e2 */
|
|
||||||
#define USB_PRODUCT_SIMTRACE2 0x60e3
|
|
||||||
|
|
||||||
#define BOARD_USB_DFU
|
#define BOARD_USB_DFU
|
||||||
#define BOARD_DFU_BOOT_SIZE (16 * 1024)
|
#define BOARD_DFU_BOOT_SIZE (16 * 1024)
|
||||||
@@ -119,4 +119,5 @@
|
|||||||
|
|
||||||
extern void board_exec_dbg_cmd(int ch);
|
extern void board_exec_dbg_cmd(int ch);
|
||||||
extern void board_main_top(void);
|
extern void board_main_top(void);
|
||||||
|
extern int board_override_enter_dfu(void);
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -26,7 +26,6 @@
|
|||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file
|
* \file
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,3 +1,18 @@
|
|||||||
|
/* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
int get_board_version_adc(void);
|
int get_board_version_adc(void);
|
||||||
|
uint32_t adc2uv(uint16_t adc);
|
||||||
|
|||||||
@@ -1,3 +1,17 @@
|
|||||||
|
/* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
enum led {
|
enum led {
|
||||||
@@ -13,9 +27,11 @@ enum led_pattern {
|
|||||||
BLINK_3O_30F = 3,
|
BLINK_3O_30F = 3,
|
||||||
BLINK_3O_1F_3O_30F = 4,
|
BLINK_3O_1F_3O_30F = 4,
|
||||||
BLINK_3O_1F_3O_1F_3O_30F= 5,
|
BLINK_3O_1F_3O_1F_3O_30F= 5,
|
||||||
BLINK_200O_F = 6,
|
BLINK_2O_F = 6,
|
||||||
BLINK_600O_F = 7,
|
BLINK_200O_F = 7,
|
||||||
BLINK_CUSTOM = 8,
|
BLINK_600O_F = 8,
|
||||||
|
BLINK_CUSTOM = 9,
|
||||||
|
BLINK_2F_O,
|
||||||
_NUM_LED_BLINK
|
_NUM_LED_BLINK
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,17 @@
|
|||||||
|
/* 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 _MANIFEST_H
|
#ifndef _MANIFEST_H
|
||||||
#define _MANIFEST_H
|
#define _MANIFEST_H
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,17 @@
|
|||||||
|
/* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
int sim_switch_use_physical(unsigned int nr, int physical);
|
int sim_switch_use_physical(unsigned int nr, int physical);
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
|
* Copyright (c) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
@@ -26,19 +27,17 @@
|
|||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#ifndef _UART_CONSOLE_
|
#ifndef _UART_CONSOLE_
|
||||||
#define _UART_CONSOLE_
|
#define _UART_CONSOLE_
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
extern void UART_Configure( uint32_t dwBaudrate, uint32_t dwMasterClock ) ;
|
extern void UART_Configure( uint32_t dwBaudrate, uint32_t dwMasterClock ) ;
|
||||||
|
extern void UART_Exit( void ) ;
|
||||||
extern void UART_PutChar( uint8_t uc ) ;
|
extern void UART_PutChar( uint8_t uc ) ;
|
||||||
|
extern void UART_PutChar_Sync( uint8_t uc ) ;
|
||||||
extern uint32_t UART_GetChar( void ) ;
|
extern uint32_t UART_GetChar( void ) ;
|
||||||
extern uint32_t UART_IsRxReady( void ) ;
|
extern uint32_t UART_IsRxReady( void ) ;
|
||||||
|
|
||||||
|
|
||||||
extern void UART_DumpFrame( uint8_t* pucFrame, uint32_t dwSize ) ;
|
extern void UART_DumpFrame( uint8_t* pucFrame, uint32_t dwSize ) ;
|
||||||
extern void UART_DumpMemory( uint8_t* pucBuffer, uint32_t dwSize, uint32_t dwAddress ) ;
|
extern void UART_DumpMemory( uint8_t* pucBuffer, uint32_t dwSize, uint32_t dwAddress ) ;
|
||||||
extern uint32_t UART_GetInteger( uint32_t* pdwValue ) ;
|
extern uint32_t UART_GetInteger( uint32_t* pdwValue ) ;
|
||||||
|
|||||||
@@ -1,203 +1,208 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2010, Atmel Corporation
|
* Copyright (c) 2010, Atmel Corporation
|
||||||
*
|
* Copyright (c) 2017, Harald Welte <laforge@gnumonks.org>
|
||||||
* All rights reserved.
|
* Copyright (c) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* All rights reserved.
|
||||||
* modification, are permitted provided that the following conditions are met:
|
*
|
||||||
*
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
* modification, are permitted provided that the following conditions are met:
|
||||||
* this list of conditions and the disclaimer below.
|
*
|
||||||
*
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
* this list of conditions and the disclaimer below.
|
||||||
* this software without specific prior written permission.
|
*
|
||||||
*
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
* this software without specific prior written permission.
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
*
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* ----------------------------------------------------------------------------
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
*/
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
/*----------------------------------------------------------------------------
|
*/
|
||||||
* Headers
|
|
||||||
*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------
|
||||||
|
* Headers
|
||||||
#include "board.h"
|
*----------------------------------------------------------------------------*/
|
||||||
#include "board_lowlevel.h"
|
|
||||||
|
#include "board.h"
|
||||||
/*----------------------------------------------------------------------------
|
#include "board_lowlevel.h"
|
||||||
* Exported variables
|
|
||||||
*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------
|
||||||
|
* Exported variables
|
||||||
/* Stack Configuration */
|
*----------------------------------------------------------------------------*/
|
||||||
#define STACK_SIZE 0x900 /** Stack size (in DWords) */
|
|
||||||
__attribute__ ((aligned(8),section(".stack")))
|
/* Stack Configuration */
|
||||||
uint32_t pdwStack[STACK_SIZE] ;
|
#define STACK_SIZE 0x900 /** Stack size (in DWords) */
|
||||||
|
__attribute__ ((aligned(8),section(".stack")))
|
||||||
/* Initialize segments */
|
uint32_t pdwStack[STACK_SIZE] ;
|
||||||
extern uint32_t _sfixed;
|
|
||||||
extern uint32_t _efixed;
|
/* Initialize segments */
|
||||||
extern uint32_t _etext;
|
extern uint32_t _sfixed;
|
||||||
extern uint32_t _srelocate;
|
extern uint32_t _efixed;
|
||||||
extern uint32_t _erelocate;
|
extern uint32_t _etext;
|
||||||
extern uint32_t _szero;
|
extern uint32_t _srelocate;
|
||||||
extern uint32_t _ezero;
|
extern uint32_t _erelocate;
|
||||||
|
extern uint32_t _szero;
|
||||||
|
extern uint32_t _ezero;
|
||||||
/*----------------------------------------------------------------------------
|
|
||||||
* ProtoTypes
|
|
||||||
*----------------------------------------------------------------------------*/
|
/*----------------------------------------------------------------------------
|
||||||
|
* ProtoTypes
|
||||||
/** \cond DOXYGEN_SHOULD_SKIP_THIS */
|
*----------------------------------------------------------------------------*/
|
||||||
extern int main( void ) ;
|
|
||||||
/** \endcond */
|
/** \cond DOXYGEN_SHOULD_SKIP_THIS */
|
||||||
void ResetException( void ) ;
|
extern int main( void ) ;
|
||||||
|
/** \endcond */
|
||||||
/*------------------------------------------------------------------------------
|
void ResetException( void ) ;
|
||||||
* Exception Table
|
|
||||||
*------------------------------------------------------------------------------*/
|
/*------------------------------------------------------------------------------
|
||||||
|
* Exception Table
|
||||||
__attribute__((section(".vectors")))
|
*------------------------------------------------------------------------------*/
|
||||||
IntFunc exception_table[] = {
|
|
||||||
|
__attribute__((section(".vectors")))
|
||||||
/* Configure Initial Stack Pointer, using linker-generated symbols */
|
IntFunc exception_table[] = {
|
||||||
(IntFunc)(&pdwStack[STACK_SIZE-1]),
|
|
||||||
ResetException,
|
/* Configure Initial Stack Pointer, using linker-generated symbols */
|
||||||
|
(IntFunc)(&pdwStack[STACK_SIZE-1]),
|
||||||
NMI_Handler,
|
ResetException,
|
||||||
HardFault_Handler,
|
|
||||||
MemManage_Handler,
|
NMI_Handler,
|
||||||
BusFault_Handler,
|
HardFault_Handler,
|
||||||
UsageFault_Handler,
|
MemManage_Handler,
|
||||||
0, 0, 0, 0, /* Reserved */
|
BusFault_Handler,
|
||||||
SVC_Handler,
|
UsageFault_Handler,
|
||||||
DebugMon_Handler,
|
0, 0, 0, 0, /* Reserved */
|
||||||
0, /* Reserved */
|
SVC_Handler,
|
||||||
PendSV_Handler,
|
DebugMon_Handler,
|
||||||
SysTick_Handler,
|
0, /* Reserved */
|
||||||
|
PendSV_Handler,
|
||||||
/* Configurable interrupts */
|
SysTick_Handler,
|
||||||
SUPC_IrqHandler, /* 0 Supply Controller */
|
|
||||||
RSTC_IrqHandler, /* 1 Reset Controller */
|
/* Configurable interrupts */
|
||||||
RTC_IrqHandler, /* 2 Real Time Clock */
|
SUPC_IrqHandler, /* 0 Supply Controller */
|
||||||
RTT_IrqHandler, /* 3 Real Time Timer */
|
RSTC_IrqHandler, /* 1 Reset Controller */
|
||||||
WDT_IrqHandler, /* 4 Watchdog Timer */
|
RTC_IrqHandler, /* 2 Real Time Clock */
|
||||||
PMC_IrqHandler, /* 5 PMC */
|
RTT_IrqHandler, /* 3 Real Time Timer */
|
||||||
EEFC_IrqHandler, /* 6 EEFC */
|
WDT_IrqHandler, /* 4 Watchdog Timer */
|
||||||
IrqHandlerNotUsed, /* 7 Reserved */
|
PMC_IrqHandler, /* 5 PMC */
|
||||||
UART0_IrqHandler, /* 8 UART0 */
|
EEFC_IrqHandler, /* 6 EEFC */
|
||||||
UART1_IrqHandler, /* 9 UART1 */
|
IrqHandlerNotUsed, /* 7 Reserved */
|
||||||
SMC_IrqHandler, /* 10 SMC */
|
UART0_IrqHandler, /* 8 UART0 */
|
||||||
PIOA_IrqHandler, /* 11 Parallel IO Controller A */
|
UART1_IrqHandler, /* 9 UART1 */
|
||||||
PIOB_IrqHandler, /* 12 Parallel IO Controller B */
|
SMC_IrqHandler, /* 10 SMC */
|
||||||
PIOC_IrqHandler, /* 13 Parallel IO Controller C */
|
PIOA_IrqHandler, /* 11 Parallel IO Controller A */
|
||||||
USART0_IrqHandler, /* 14 USART 0 */
|
PIOB_IrqHandler, /* 12 Parallel IO Controller B */
|
||||||
USART1_IrqHandler, /* 15 USART 1 */
|
PIOC_IrqHandler, /* 13 Parallel IO Controller C */
|
||||||
IrqHandlerNotUsed, /* 16 Reserved */
|
USART0_IrqHandler, /* 14 USART 0 */
|
||||||
IrqHandlerNotUsed, /* 17 Reserved */
|
USART1_IrqHandler, /* 15 USART 1 */
|
||||||
MCI_IrqHandler, /* 18 MCI */
|
IrqHandlerNotUsed, /* 16 Reserved */
|
||||||
TWI0_IrqHandler, /* 19 TWI 0 */
|
IrqHandlerNotUsed, /* 17 Reserved */
|
||||||
TWI1_IrqHandler, /* 20 TWI 1 */
|
MCI_IrqHandler, /* 18 MCI */
|
||||||
SPI_IrqHandler, /* 21 SPI */
|
TWI0_IrqHandler, /* 19 TWI 0 */
|
||||||
SSC_IrqHandler, /* 22 SSC */
|
TWI1_IrqHandler, /* 20 TWI 1 */
|
||||||
TC0_IrqHandler, /* 23 Timer Counter 0 */
|
SPI_IrqHandler, /* 21 SPI */
|
||||||
TC1_IrqHandler, /* 24 Timer Counter 1 */
|
SSC_IrqHandler, /* 22 SSC */
|
||||||
TC2_IrqHandler, /* 25 Timer Counter 2 */
|
TC0_IrqHandler, /* 23 Timer Counter 0 */
|
||||||
TC3_IrqHandler, /* 26 Timer Counter 3 */
|
TC1_IrqHandler, /* 24 Timer Counter 1 */
|
||||||
TC4_IrqHandler, /* 27 Timer Counter 4 */
|
TC2_IrqHandler, /* 25 Timer Counter 2 */
|
||||||
TC5_IrqHandler, /* 28 Timer Counter 5 */
|
TC3_IrqHandler, /* 26 Timer Counter 3 */
|
||||||
ADC_IrqHandler, /* 29 ADC controller */
|
TC4_IrqHandler, /* 27 Timer Counter 4 */
|
||||||
DAC_IrqHandler, /* 30 DAC controller */
|
TC5_IrqHandler, /* 28 Timer Counter 5 */
|
||||||
PWM_IrqHandler, /* 31 PWM */
|
ADC_IrqHandler, /* 29 ADC controller */
|
||||||
CRCCU_IrqHandler, /* 32 CRC Calculation Unit */
|
DAC_IrqHandler, /* 30 DAC controller */
|
||||||
ACC_IrqHandler, /* 33 Analog Comparator */
|
PWM_IrqHandler, /* 31 PWM */
|
||||||
USBD_IrqHandler, /* 34 USB Device Port */
|
CRCCU_IrqHandler, /* 32 CRC Calculation Unit */
|
||||||
IrqHandlerNotUsed /* 35 not used */
|
ACC_IrqHandler, /* 33 Analog Comparator */
|
||||||
};
|
USBD_IrqHandler, /* 34 USB Device Port */
|
||||||
|
IrqHandlerNotUsed /* 35 not used */
|
||||||
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu)
|
};
|
||||||
#include "usb/device/dfu/dfu.h"
|
|
||||||
static void BootIntoApp(void)
|
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu)
|
||||||
{
|
#include "usb/device/dfu/dfu.h"
|
||||||
unsigned int *pSrc;
|
static void BootIntoApp(void)
|
||||||
void (*appReset)(void);
|
{
|
||||||
|
unsigned int *pSrc;
|
||||||
pSrc = (unsigned int *) ((unsigned char *)IFLASH_ADDR + BOARD_DFU_BOOT_SIZE);
|
void (*appReset)(void);
|
||||||
SCB->VTOR = ((unsigned int)(pSrc)) | (0x0 << 7);
|
|
||||||
appReset = pSrc[1];
|
pSrc = (unsigned int *) ((unsigned char *)IFLASH_ADDR + BOARD_DFU_BOOT_SIZE);
|
||||||
|
/* set vector table to application vector table (store at the beginning of the application) */
|
||||||
g_dfu->state = DFU_STATE_appIDLE;
|
SCB->VTOR = (unsigned int)(pSrc);
|
||||||
|
/* set stack pointer to address provided in the beginning of the application (loaded into a register first) */
|
||||||
appReset();
|
__asm__ volatile ("MSR msp,%0" : :"r"(*pSrc));
|
||||||
}
|
/* start application (by jumping to the reset function which address is stored as second entry of the vector table) */
|
||||||
#endif
|
appReset = (void(*)(void))pSrc[1];
|
||||||
|
|
||||||
/**
|
g_dfu->state = DFU_STATE_appIDLE;
|
||||||
* \brief This is the code that gets called on processor reset.
|
|
||||||
* To initialize the device, and call the main() routine.
|
appReset();
|
||||||
*/
|
}
|
||||||
void ResetException( void )
|
#endif
|
||||||
{
|
|
||||||
uint32_t *pSrc, *pDest ;
|
/**
|
||||||
|
* \brief This is the code that gets called on processor reset.
|
||||||
/* Low level Initialize */
|
* To initialize the device, and call the main() routine.
|
||||||
LowLevelInit() ;
|
*/
|
||||||
|
void ResetException( void )
|
||||||
|
{
|
||||||
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu)
|
uint32_t *pSrc, *pDest ;
|
||||||
/* we are before the text segment has been relocated, so g_dfu is
|
|
||||||
* not initialized yet */
|
/* Low level Initialize */
|
||||||
g_dfu = &_g_dfu;
|
LowLevelInit() ;
|
||||||
if ((g_dfu->magic != USB_DFU_MAGIC) && !USBDFU_OverrideEnterDFU()) {
|
|
||||||
BootIntoApp();
|
|
||||||
/* Infinite loop */
|
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu)
|
||||||
while ( 1 ) ;
|
if (!USBDFU_OverrideEnterDFU()) {
|
||||||
}
|
UART_Exit();
|
||||||
#endif
|
__disable_irq();
|
||||||
|
BootIntoApp();
|
||||||
/* Initialize the relocate segment */
|
/* Infinite loop */
|
||||||
pSrc = &_etext ;
|
while ( 1 ) ;
|
||||||
pDest = &_srelocate ;
|
}
|
||||||
|
#endif
|
||||||
if ( pSrc != pDest )
|
|
||||||
{
|
/* Initialize the relocate segment */
|
||||||
for ( ; pDest < &_erelocate ; )
|
pSrc = &_etext ;
|
||||||
{
|
pDest = &_srelocate ;
|
||||||
*pDest++ = *pSrc++ ;
|
|
||||||
}
|
if ( pSrc != pDest )
|
||||||
}
|
{
|
||||||
|
for ( ; pDest < &_erelocate ; )
|
||||||
/* Clear the zero segment */
|
{
|
||||||
for ( pDest = &_szero ; pDest < &_ezero ; )
|
*pDest++ = *pSrc++ ;
|
||||||
{
|
}
|
||||||
*pDest++ = 0;
|
}
|
||||||
}
|
|
||||||
|
/* Clear the zero segment */
|
||||||
/* Set the vector table base address */
|
for ( pDest = &_szero ; pDest < &_ezero ; )
|
||||||
pSrc = (uint32_t *)&_sfixed;
|
{
|
||||||
SCB->VTOR = ( (uint32_t)pSrc & SCB_VTOR_TBLOFF_Msk ) ;
|
*pDest++ = 0;
|
||||||
|
}
|
||||||
if ( ((uint32_t)pSrc >= IRAM_ADDR) && ((uint32_t)pSrc < IRAM_ADDR+IRAM_SIZE) )
|
|
||||||
{
|
/* Set the vector table base address */
|
||||||
SCB->VTOR |= 1 << SCB_VTOR_TBLBASE_Pos ;
|
pSrc = (uint32_t *)&_sfixed;
|
||||||
}
|
SCB->VTOR = ( (uint32_t)pSrc & SCB_VTOR_TBLOFF_Msk ) ;
|
||||||
|
|
||||||
/* App should have disabled interrupts during the transition */
|
if ( ((uint32_t)pSrc >= IRAM_ADDR) && ((uint32_t)pSrc < IRAM_ADDR+IRAM_SIZE) )
|
||||||
__enable_irq();
|
{
|
||||||
|
SCB->VTOR |= 1 << SCB_VTOR_TBLBASE_Pos ;
|
||||||
/* Branch to main function */
|
}
|
||||||
main() ;
|
|
||||||
|
/* App should have disabled interrupts during the transition */
|
||||||
/* Infinite loop */
|
__enable_irq();
|
||||||
while ( 1 ) ;
|
|
||||||
}
|
/* Branch to main function */
|
||||||
|
main() ;
|
||||||
|
|
||||||
|
/* Infinite loop */
|
||||||
|
while ( 1 ) ;
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,216 +1,220 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
* Copyright (c) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
* All rights reserved.
|
*
|
||||||
*
|
* All rights reserved.
|
||||||
* Redistribution and use in source and binary forms, with or without
|
*
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* Redistribution and use in source and binary forms, with or without
|
||||||
*
|
* modification, are permitted provided that the following conditions are met:
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
*
|
||||||
* this list of conditions and the disclaimer below.
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
*
|
* this list of conditions and the disclaimer below.
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
*
|
||||||
* this software without specific prior written permission.
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
*
|
* this software without specific prior written permission.
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
*
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* ----------------------------------------------------------------------------
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
* ----------------------------------------------------------------------------
|
||||||
|
*/
|
||||||
/**
|
|
||||||
* \file
|
/**
|
||||||
*
|
* \file
|
||||||
* Provides the low-level initialization function that called on chip startup.
|
*
|
||||||
*/
|
* Provides the low-level initialization function that called on chip startup.
|
||||||
|
*/
|
||||||
/*----------------------------------------------------------------------------
|
|
||||||
* Headers
|
/*----------------------------------------------------------------------------
|
||||||
*----------------------------------------------------------------------------*/
|
* Headers
|
||||||
|
*----------------------------------------------------------------------------*/
|
||||||
#include "board.h"
|
|
||||||
|
#include "board.h"
|
||||||
/*----------------------------------------------------------------------------
|
|
||||||
* Local definitions
|
/*----------------------------------------------------------------------------
|
||||||
*----------------------------------------------------------------------------*/
|
* Local definitions
|
||||||
|
*----------------------------------------------------------------------------*/
|
||||||
#define BOARD_OSCOUNT (CKGR_MOR_MOSCXTST(0x8))
|
|
||||||
#define BOARD_MCKR (PMC_MCKR_PRES_CLK | PMC_MCKR_CSS_PLLA_CLK)
|
#define BOARD_OSCOUNT (CKGR_MOR_MOSCXTST(0x8))
|
||||||
|
#define BOARD_MCKR (PMC_MCKR_PRES_CLK | PMC_MCKR_CSS_PLLA_CLK)
|
||||||
#if (BOARD_MCK == 48000000)
|
|
||||||
#if (BOARD_MAINOSC == 18432000)
|
/** configure PLL to generate main clock based on main oscillator frequency */
|
||||||
/* Clock settings at 48MHz for 18 MHz crystal */
|
#if (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 48000000)
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
| CKGR_PLLAR_MULA(13-1) \
|
| CKGR_PLLAR_MULA(8-1) \
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_DIVA(5))
|
| CKGR_PLLAR_DIVA(2))
|
||||||
#elif (BOARD_MAINOSC == 12000000)
|
#elif (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 58000000)
|
||||||
/* QMod has 12 MHz clock, so multply by 8 (96 MHz) and divide by 2 */
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
| CKGR_PLLAR_MULA(29-1) \
|
||||||
| CKGR_PLLAR_MULA(8-1) \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
| CKGR_PLLAR_DIVA(6))
|
||||||
| CKGR_PLLAR_DIVA(2))
|
#elif (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 60000000)
|
||||||
#else
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
#error "Please define PLLA config for your MAINOSC frequency"
|
| CKGR_PLLAR_MULA(10-1) \
|
||||||
#endif /* MAINOSC */
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
#elif (BOARD_MCK == 64000000)
|
| CKGR_PLLAR_DIVA(2))
|
||||||
#if (BOARD_MAINOSC == 18432000)
|
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 47923200)
|
||||||
/* Clock settings at 64MHz for 18 MHz crystal: 64.512 MHz */
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
| CKGR_PLLAR_MULA(13-1) \
|
||||||
| CKGR_PLLAR_MULA(7-1) \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
| CKGR_PLLAR_DIVA(5))
|
||||||
| CKGR_PLLAR_DIVA(2))
|
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 58982400)
|
||||||
#elif (BOARD_MAINOSC == 12000000)
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
/* QMod has 12 MHz clock, so multply by 10 / div by 2: 60 MHz */
|
| CKGR_PLLAR_MULA(16-1) \
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_MULA(10-1) \
|
| CKGR_PLLAR_DIVA(5))
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 64512000)
|
||||||
| CKGR_PLLAR_DIVA(2))
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
#error "Please define PLLA config for your MAINOSC frequency"
|
| CKGR_PLLAR_MULA(7-1) \
|
||||||
#endif /* MAINOSC */
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
#else
|
| CKGR_PLLAR_DIVA(2))
|
||||||
#error "No PLL settings for current BOARD_MCK."
|
#else
|
||||||
#endif
|
#error "Please define PLLA config for your BOARD_MCK/MAINOSC frequency"
|
||||||
|
#endif
|
||||||
#if (BOARD_MAINOSC == 12000000)
|
|
||||||
#define PLLB_CFG (CKGR_PLLBR_DIVB(2)|CKGR_PLLBR_MULB(8-1)|CKGR_PLLBR_PLLBCOUNT_Msk)
|
#if (BOARD_MAINOSC == 12000000)
|
||||||
#elif (BOARD_MAINOSC == 18432000)
|
#define PLLB_CFG (CKGR_PLLBR_DIVB(2)|CKGR_PLLBR_MULB(8-1)|CKGR_PLLBR_PLLBCOUNT_Msk)
|
||||||
#define PLLB_CFG (CKGR_PLLBR_DIVB(5)|CKGR_PLLBR_MULB(13-1)|CKGR_PLLBR_PLLBCOUNT_Msk)
|
#elif (BOARD_MAINOSC == 18432000)
|
||||||
#else
|
#define PLLB_CFG (CKGR_PLLBR_DIVB(5)|CKGR_PLLBR_MULB(13-1)|CKGR_PLLBR_PLLBCOUNT_Msk)
|
||||||
#error "Please configure PLLB for your MAINOSC freq"
|
#else
|
||||||
#endif
|
#error "Please configure PLLB for your MAINOSC freq"
|
||||||
|
#endif
|
||||||
/* Define clock timeout */
|
|
||||||
#define CLOCK_TIMEOUT 0xFFFFFFFF
|
/* Define clock timeout */
|
||||||
|
#define CLOCK_TIMEOUT 0xFFFFFFFF
|
||||||
/**
|
|
||||||
* \brief Configure 48MHz Clock for USB
|
/**
|
||||||
*/
|
* \brief Configure 48MHz Clock for USB
|
||||||
static void _ConfigureUsbClock(void)
|
*/
|
||||||
{
|
static void _ConfigureUsbClock(void)
|
||||||
/* Enable PLLB for USB */
|
{
|
||||||
PMC->CKGR_PLLBR = PLLB_CFG;
|
/* Enable PLLB for USB */
|
||||||
while ((PMC->PMC_SR & PMC_SR_LOCKB) == 0) ;
|
PMC->CKGR_PLLBR = PLLB_CFG;
|
||||||
|
while ((PMC->PMC_SR & PMC_SR_LOCKB) == 0) ;
|
||||||
/* USB Clock uses PLLB */
|
|
||||||
PMC->PMC_USB = PMC_USB_USBDIV(0) /* /1 (no divider) */
|
/* USB Clock uses PLLB */
|
||||||
| PMC_USB_USBS; /* PLLB */
|
PMC->PMC_USB = PMC_USB_USBDIV(0) /* /1 (no divider) */
|
||||||
}
|
| PMC_USB_USBS; /* PLLB */
|
||||||
|
}
|
||||||
/*----------------------------------------------------------------------------
|
|
||||||
* Exported functions
|
/*----------------------------------------------------------------------------
|
||||||
*----------------------------------------------------------------------------*/
|
* Exported functions
|
||||||
|
*----------------------------------------------------------------------------*/
|
||||||
/**
|
|
||||||
* \brief Performs the low-level initialization of the chip.
|
/**
|
||||||
* This includes EFC and master clock configuration.
|
* \brief Performs the low-level initialization of the chip.
|
||||||
* It also enable a low level on the pin NRST triggers a user reset.
|
* This includes EFC and master clock configuration.
|
||||||
*/
|
* It also enable a low level on the pin NRST triggers a user reset.
|
||||||
extern WEAK void LowLevelInit( void )
|
*/
|
||||||
{
|
extern WEAK void LowLevelInit( void )
|
||||||
uint32_t timeout = 0;
|
{
|
||||||
|
uint32_t timeout = 0;
|
||||||
/* Configure the Supply Monitor to reset the CPU in case VDDIO is
|
|
||||||
* lower than 3.0V. As we run the board on 3.3V, any lower voltage
|
/* Configure the Supply Monitor to reset the CPU in case VDDIO is
|
||||||
* might be some kind of leakage that creeps in some way, but is not
|
* lower than 3.0V. As we run the board on 3.3V, any lower voltage
|
||||||
* the "official" power supply */
|
* might be some kind of leakage that creeps in some way, but is not
|
||||||
SUPC->SUPC_SMMR = SUPC_SMMR_SMTH_3_0V | SUPC_SMMR_SMSMPL_CSM |
|
* the "official" power supply */
|
||||||
SUPC_SMMR_SMRSTEN_ENABLE;
|
SUPC->SUPC_SMMR = SUPC_SMMR_SMTH_3_0V | SUPC_SMMR_SMSMPL_CSM |
|
||||||
|
SUPC_SMMR_SMRSTEN_ENABLE;
|
||||||
/* enable both LED and green LED */
|
|
||||||
PIOA->PIO_PER |= PIO_LED_RED | PIO_LED_GREEN;
|
/* disable ERASE pin to prevent accidental flash erase */
|
||||||
PIOA->PIO_OER |= PIO_LED_RED | PIO_LED_GREEN;
|
MATRIX->CCFG_SYSIO |= CCFG_SYSIO_SYSIO12;
|
||||||
PIOA->PIO_CODR |= PIO_LED_RED | PIO_LED_GREEN;
|
|
||||||
|
/* enable both LED and green LED */
|
||||||
/* Set 3 FWS for Embedded Flash Access */
|
PIOA->PIO_PER |= PIO_LED_RED | PIO_LED_GREEN;
|
||||||
EFC->EEFC_FMR = EEFC_FMR_FWS(3);
|
PIOA->PIO_OER |= PIO_LED_RED | PIO_LED_GREEN;
|
||||||
|
PIOA->PIO_CODR |= PIO_LED_RED | PIO_LED_GREEN;
|
||||||
/* Select external slow clock */
|
|
||||||
/* if ((SUPC->SUPC_SR & SUPC_SR_OSCSEL) != SUPC_SR_OSCSEL_CRYST)
|
/* Set 3 FWS for Embedded Flash Access */
|
||||||
{
|
EFC->EEFC_FMR = EEFC_FMR_FWS(3);
|
||||||
SUPC->SUPC_CR = (uint32_t)(SUPC_CR_XTALSEL_CRYSTAL_SEL | SUPC_CR_KEY(0xA5));
|
|
||||||
timeout = 0;
|
/* Select external slow clock */
|
||||||
while (!(SUPC->SUPC_SR & SUPC_SR_OSCSEL_CRYST) );
|
/* if ((SUPC->SUPC_SR & SUPC_SR_OSCSEL) != SUPC_SR_OSCSEL_CRYST)
|
||||||
}
|
{
|
||||||
*/
|
SUPC->SUPC_CR = (uint32_t)(SUPC_CR_XTALSEL_CRYSTAL_SEL | SUPC_CR_KEY(0xA5));
|
||||||
|
timeout = 0;
|
||||||
#ifndef qmod
|
while (!(SUPC->SUPC_SR & SUPC_SR_OSCSEL_CRYST) );
|
||||||
/* Initialize main oscillator */
|
}
|
||||||
if ( !(PMC->CKGR_MOR & CKGR_MOR_MOSCSEL) )
|
*/
|
||||||
{
|
|
||||||
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN;
|
#ifndef qmod
|
||||||
timeout = 0;
|
/* Initialize main oscillator */
|
||||||
while (!(PMC->PMC_SR & PMC_SR_MOSCXTS) && (timeout++ < CLOCK_TIMEOUT));
|
if ( !(PMC->CKGR_MOR & CKGR_MOR_MOSCSEL) )
|
||||||
}
|
{
|
||||||
|
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN;
|
||||||
/* Switch to 3-20MHz Xtal oscillator */
|
timeout = 0;
|
||||||
PIOB->PIO_PDR = (1 << 8) | (1 << 9);
|
while (!(PMC->PMC_SR & PMC_SR_MOSCXTS) && (timeout++ < CLOCK_TIMEOUT));
|
||||||
PIOB->PIO_PUDR = (1 << 8) | (1 << 9);
|
}
|
||||||
PIOB->PIO_PPDDR = (1 << 8) | (1 << 9);
|
|
||||||
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN | CKGR_MOR_MOSCSEL;
|
/* Switch to 3-20MHz Xtal oscillator */
|
||||||
/* wait for Main XTAL oscillator stabilization */
|
PIOB->PIO_PDR = (1 << 8) | (1 << 9);
|
||||||
timeout = 0;
|
PIOB->PIO_PUDR = (1 << 8) | (1 << 9);
|
||||||
while (!(PMC->PMC_SR & PMC_SR_MOSCSELS) && (timeout++ < CLOCK_TIMEOUT));
|
PIOB->PIO_PPDDR = (1 << 8) | (1 << 9);
|
||||||
#else
|
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN | CKGR_MOR_MOSCSEL;
|
||||||
/* QMOD has external 12MHz clock source */
|
/* wait for Main XTAL oscillator stabilization */
|
||||||
PIOB->PIO_PDR = (1 << 9);
|
timeout = 0;
|
||||||
PIOB->PIO_PUDR = (1 << 9);
|
while (!(PMC->PMC_SR & PMC_SR_MOSCSELS) && (timeout++ < CLOCK_TIMEOUT));
|
||||||
PIOB->PIO_PPDDR = (1 << 9);
|
#else
|
||||||
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTBY| CKGR_MOR_MOSCSEL;
|
/* QMOD has external 12MHz clock source */
|
||||||
#endif
|
PIOB->PIO_PDR = (1 << 9);
|
||||||
|
PIOB->PIO_PUDR = (1 << 9);
|
||||||
/* disable the red LED after main clock initialization */
|
PIOB->PIO_PPDDR = (1 << 9);
|
||||||
PIOA->PIO_SODR = PIO_LED_RED;
|
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTBY| CKGR_MOR_MOSCSEL;
|
||||||
|
#endif
|
||||||
/* "switch" to main clock as master clock source (should already be the case */
|
|
||||||
PMC->PMC_MCKR = (PMC->PMC_MCKR & ~(uint32_t)PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK;
|
/* disable the red LED after main clock initialization */
|
||||||
/* wait for master clock to be ready */
|
PIOA->PIO_SODR = PIO_LED_RED;
|
||||||
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
|
|
||||||
|
/* "switch" to main clock as master clock source (should already be the case */
|
||||||
/* Initialize PLLA */
|
PMC->PMC_MCKR = (PMC->PMC_MCKR & ~(uint32_t)PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK;
|
||||||
PMC->CKGR_PLLAR = BOARD_PLLAR;
|
/* wait for master clock to be ready */
|
||||||
/* Wait for PLLA to lock */
|
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
|
||||||
timeout = 0;
|
|
||||||
while (!(PMC->PMC_SR & PMC_SR_LOCKA) && (timeout++ < CLOCK_TIMEOUT));
|
/* Initialize PLLA */
|
||||||
|
PMC->CKGR_PLLAR = BOARD_PLLAR;
|
||||||
/* Switch to main clock (again ?!?) */
|
/* Wait for PLLA to lock */
|
||||||
PMC->PMC_MCKR = (BOARD_MCKR & ~PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK;
|
timeout = 0;
|
||||||
/* wait for master clock to be ready */
|
while (!(PMC->PMC_SR & PMC_SR_LOCKA) && (timeout++ < CLOCK_TIMEOUT));
|
||||||
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
|
|
||||||
|
/* Switch to main clock (again ?!?) */
|
||||||
/* switch to PLLA as master clock source */
|
PMC->PMC_MCKR = (BOARD_MCKR & ~PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK;
|
||||||
PMC->PMC_MCKR = BOARD_MCKR ;
|
/* wait for master clock to be ready */
|
||||||
/* wait for master clock to be ready */
|
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
|
||||||
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
|
|
||||||
|
/* switch to PLLA as master clock source */
|
||||||
/* Configure SysTick for 1ms */
|
PMC->PMC_MCKR = BOARD_MCKR ;
|
||||||
SysTick_Config(BOARD_MCK/1000);
|
/* wait for master clock to be ready */
|
||||||
|
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
|
||||||
_ConfigureUsbClock();
|
|
||||||
}
|
/* Configure SysTick for 1ms */
|
||||||
|
SysTick_Config(BOARD_MCK/1000);
|
||||||
/* SysTick based delay function */
|
|
||||||
|
_ConfigureUsbClock();
|
||||||
volatile uint32_t jiffies;
|
}
|
||||||
|
|
||||||
/* Interrupt handler for SysTick interrupt */
|
/* SysTick based delay function */
|
||||||
void SysTick_Handler(void)
|
|
||||||
{
|
volatile uint32_t jiffies;
|
||||||
jiffies++;
|
|
||||||
}
|
/* Interrupt handler for SysTick interrupt */
|
||||||
|
void SysTick_Handler(void)
|
||||||
void mdelay(unsigned int msecs)
|
{
|
||||||
{
|
jiffies++;
|
||||||
uint32_t jiffies_start = jiffies;
|
}
|
||||||
do {
|
|
||||||
} while ((jiffies - jiffies_start) < msecs);
|
void mdelay(unsigned int msecs)
|
||||||
}
|
{
|
||||||
|
uint32_t jiffies_start = jiffies;
|
||||||
|
do {
|
||||||
|
} while ((jiffies - jiffies_start) < msecs);
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,22 @@
|
|||||||
|
/* 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 "board.h"
|
#include "board.h"
|
||||||
#include "boardver_adc.h"
|
#include "boardver_adc.h"
|
||||||
|
|
||||||
/* FIXME: share this with mode_cardemu.c */
|
|
||||||
#define UV_PER_LSB ((3300 * 1000) / 4096)
|
#define UV_PER_LSB ((3300 * 1000) / 4096)
|
||||||
static uint32_t adc2uv(uint16_t adc)
|
uint32_t adc2uv(uint16_t adc)
|
||||||
{
|
{
|
||||||
uint32_t uv = (uint32_t) adc * UV_PER_LSB;
|
uint32_t uv = (uint32_t) adc * UV_PER_LSB;
|
||||||
return uv;
|
return uv;
|
||||||
@@ -73,7 +86,7 @@ int get_board_version_adc(void)
|
|||||||
/* convert to voltage */
|
/* convert to voltage */
|
||||||
sample = ADC->ADC_CDR[2];
|
sample = ADC->ADC_CDR[2];
|
||||||
uv = adc2uv(sample);
|
uv = adc2uv(sample);
|
||||||
TRACE_INFO("VERSION_DET ADC=%u => %u uV\r\n", sample, uv);
|
TRACE_INFO("VERSION_DET ADC=%u => %lu uV\r\n", sample, uv);
|
||||||
|
|
||||||
/* FIXME: convert to board version based on thresholds */
|
/* FIXME: convert to board version based on thresholds */
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,22 @@
|
|||||||
|
/* LED control
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
@@ -16,9 +35,9 @@ static void led_set(enum led led, int on)
|
|||||||
ASSERT(led < PIO_LISTSIZE(pinsLeds));
|
ASSERT(led < PIO_LISTSIZE(pinsLeds));
|
||||||
|
|
||||||
if (on)
|
if (on)
|
||||||
PIO_Set(&pinsLeds[led]);
|
PIO_Clear(&pinsLeds[led]);
|
||||||
else
|
else
|
||||||
PIO_Clear(&pinsLeds[led]);
|
PIO_Set(&pinsLeds[led]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* LED blinking code */
|
/* LED blinking code */
|
||||||
@@ -27,7 +46,7 @@ static void led_set(enum led led, int on)
|
|||||||
struct blink_state {
|
struct blink_state {
|
||||||
/* duration of the state in ms */
|
/* duration of the state in ms */
|
||||||
uint16_t duration;
|
uint16_t duration;
|
||||||
/* bringhtness of LED during the state */
|
/* brightness of LED during the state */
|
||||||
uint8_t on;
|
uint8_t on;
|
||||||
} __attribute__((packed));
|
} __attribute__((packed));
|
||||||
|
|
||||||
@@ -54,13 +73,23 @@ static const struct blink_state bs_3on_1off_3on_30off[] = {
|
|||||||
static const struct blink_state bs_3on_1off_3on_1off_3on_30off[] = {
|
static const struct blink_state bs_3on_1off_3on_1off_3on_30off[] = {
|
||||||
{ 300, 1 }, { 100, 0 }, { 300, 1 }, { 100, 0 }, { 300, 1 }, { 3000, 0 }
|
{ 300, 1 }, { 100, 0 }, { 300, 1 }, { 100, 0 }, { 300, 1 }, { 3000, 0 }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct blink_state bs_2on_off[] = {
|
||||||
|
{ 200, 1 }, { 0, 0 },
|
||||||
|
};
|
||||||
|
|
||||||
static const struct blink_state bs_200on_off[] = {
|
static const struct blink_state bs_200on_off[] = {
|
||||||
{ 20000, 1 }, { 0, 0 },
|
{ 20000, 1 }, { 0, 0 },
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct blink_state bs_600on_off[] = {
|
static const struct blink_state bs_600on_off[] = {
|
||||||
{ 60000, 1 }, { 0, 0 },
|
{ 60000, 1 }, { 0, 0 },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const struct blink_state bs_2off_on[] = {
|
||||||
|
{ 200, 0 }, { 0, 1 },
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/* a blink pattern is an array of blink_states */
|
/* a blink pattern is an array of blink_states */
|
||||||
struct blink_pattern {
|
struct blink_pattern {
|
||||||
@@ -94,6 +123,10 @@ static const struct blink_pattern patterns[] = {
|
|||||||
.states = bs_3on_1off_3on_1off_3on_30off,
|
.states = bs_3on_1off_3on_1off_3on_30off,
|
||||||
.size = ARRAY_SIZE(bs_3on_1off_3on_1off_3on_30off),
|
.size = ARRAY_SIZE(bs_3on_1off_3on_1off_3on_30off),
|
||||||
},
|
},
|
||||||
|
[BLINK_2O_F] = {
|
||||||
|
.states = bs_2on_off,
|
||||||
|
.size = ARRAY_SIZE(bs_2on_off),
|
||||||
|
},
|
||||||
[BLINK_200O_F] = {
|
[BLINK_200O_F] = {
|
||||||
.states = bs_200on_off,
|
.states = bs_200on_off,
|
||||||
.size = ARRAY_SIZE(bs_200on_off),
|
.size = ARRAY_SIZE(bs_200on_off),
|
||||||
@@ -102,6 +135,11 @@ static const struct blink_pattern patterns[] = {
|
|||||||
.states = bs_600on_off,
|
.states = bs_600on_off,
|
||||||
.size = ARRAY_SIZE(bs_600on_off),
|
.size = ARRAY_SIZE(bs_600on_off),
|
||||||
},
|
},
|
||||||
|
[BLINK_2F_O] = {
|
||||||
|
.states = bs_2off_on,
|
||||||
|
.size = ARRAY_SIZE(bs_2off_on),
|
||||||
|
},
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct led_state {
|
struct led_state {
|
||||||
@@ -158,16 +196,16 @@ static void blink_tmr_cb(void *data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static struct led_state led_state[] = {
|
static struct led_state led_state[] = {
|
||||||
[LED_GREEN] = {
|
|
||||||
.led = LED_GREEN,
|
|
||||||
.timer.cb = blink_tmr_cb,
|
|
||||||
.timer.data = &led_state[LED_GREEN],
|
|
||||||
},
|
|
||||||
[LED_RED] = {
|
[LED_RED] = {
|
||||||
.led = LED_RED,
|
.led = LED_RED,
|
||||||
.timer.cb = blink_tmr_cb,
|
.timer.cb = blink_tmr_cb,
|
||||||
.timer.data = &led_state[LED_RED],
|
.timer.data = &led_state[LED_RED],
|
||||||
},
|
},
|
||||||
|
[LED_GREEN] = {
|
||||||
|
.led = LED_GREEN,
|
||||||
|
.timer.cb = blink_tmr_cb,
|
||||||
|
.timer.data = &led_state[LED_GREEN],
|
||||||
|
},
|
||||||
};
|
};
|
||||||
#endif /* PINS_LEDS */
|
#endif /* PINS_LEDS */
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,17 @@
|
|||||||
|
/* 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 "manifest.h"
|
#include "manifest.h"
|
||||||
|
|
||||||
const char *manifest_application = APPLICATION;
|
const char *manifest_application = APPLICATION;
|
||||||
|
|||||||
@@ -1,7 +1,24 @@
|
|||||||
/* Code to switch between local (physical) and remote (emulated) SIM */
|
/* Code to switch between local (physical) and remote (emulated) SIM
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.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
|
||||||
|
*/
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "trace.h"
|
#include "trace.h"
|
||||||
|
#include "led.h"
|
||||||
#include "sim_switch.h"
|
#include "sim_switch.h"
|
||||||
|
|
||||||
#ifdef PIN_SIM_SWITCH1
|
#ifdef PIN_SIM_SWITCH1
|
||||||
@@ -16,6 +33,7 @@ static int initialized = 0;
|
|||||||
int sim_switch_use_physical(unsigned int nr, int physical)
|
int sim_switch_use_physical(unsigned int nr, int physical)
|
||||||
{
|
{
|
||||||
const Pin *pin;
|
const Pin *pin;
|
||||||
|
enum led led;
|
||||||
|
|
||||||
if (!initialized) {
|
if (!initialized) {
|
||||||
TRACE_ERROR("Somebody forgot to call sim_switch_init()\r\n");
|
TRACE_ERROR("Somebody forgot to call sim_switch_init()\r\n");
|
||||||
@@ -29,11 +47,13 @@ int sim_switch_use_physical(unsigned int nr, int physical)
|
|||||||
#ifdef PIN_SIM_SWITCH1
|
#ifdef PIN_SIM_SWITCH1
|
||||||
case 0:
|
case 0:
|
||||||
pin = &pin_conn_usim1;
|
pin = &pin_conn_usim1;
|
||||||
|
led = LED_USIM1;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
#ifdef PIN_SIM_SWITCH2
|
#ifdef PIN_SIM_SWITCH2
|
||||||
case 1:
|
case 1:
|
||||||
pin = &pin_conn_usim2;
|
pin = &pin_conn_usim2;
|
||||||
|
led = LED_USIM2;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
default:
|
default:
|
||||||
@@ -44,9 +64,11 @@ int sim_switch_use_physical(unsigned int nr, int physical)
|
|||||||
if (physical) {
|
if (physical) {
|
||||||
TRACE_INFO("%u: Use local/physical SIM\r\n", nr);
|
TRACE_INFO("%u: Use local/physical SIM\r\n", nr);
|
||||||
PIO_Clear(pin);
|
PIO_Clear(pin);
|
||||||
|
led_blink(led, BLINK_ALWAYS_ON);
|
||||||
} else {
|
} else {
|
||||||
TRACE_INFO("%u: Use remote/emulated SIM\r\n", nr);
|
TRACE_INFO("%u: Use remote/emulated SIM\r\n", nr);
|
||||||
PIO_Set(pin);
|
PIO_Set(pin);
|
||||||
|
led_blink(led, BLINK_ALWAYS_OFF);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@@ -63,5 +85,6 @@ int sim_switch_init(void)
|
|||||||
PIO_Configure(&pin_conn_usim2, 1);
|
PIO_Configure(&pin_conn_usim2, 1);
|
||||||
num_switch++;
|
num_switch++;
|
||||||
#endif
|
#endif
|
||||||
|
initialized = 1;
|
||||||
return num_switch;
|
return num_switch;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,380 +1,453 @@
|
|||||||
/* ----------------------------------------------------------------------------
|
/* ----------------------------------------------------------------------------
|
||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
*
|
* Copyright (c) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
* All rights reserved.
|
*
|
||||||
*
|
* All rights reserved.
|
||||||
* Redistribution and use in source and binary forms, with or without
|
*
|
||||||
* modification, are permitted provided that the following conditions are met:
|
* Redistribution and use in source and binary forms, with or without
|
||||||
*
|
* modification, are permitted provided that the following conditions are met:
|
||||||
* - Redistributions of source code must retain the above copyright notice,
|
*
|
||||||
* this list of conditions and the disclaimer below.
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
*
|
* this list of conditions and the disclaimer below.
|
||||||
* Atmel's name may not be used to endorse or promote products derived from
|
*
|
||||||
* this software without specific prior written permission.
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
*
|
* this software without specific prior written permission.
|
||||||
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
*
|
||||||
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
* ----------------------------------------------------------------------------
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
* ----------------------------------------------------------------------------
|
||||||
|
*/
|
||||||
/**
|
|
||||||
* \file
|
/**
|
||||||
*
|
* \file
|
||||||
* Implements UART console.
|
*
|
||||||
*
|
* Implements UART console.
|
||||||
*/
|
*
|
||||||
|
*/
|
||||||
/*----------------------------------------------------------------------------
|
|
||||||
* Headers
|
/*----------------------------------------------------------------------------
|
||||||
*----------------------------------------------------------------------------*/
|
* Headers
|
||||||
|
*----------------------------------------------------------------------------*/
|
||||||
#include "board.h"
|
|
||||||
|
#include "board.h"
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdint.h>
|
#include <stdio.h>
|
||||||
|
#include <stdint.h>
|
||||||
/*----------------------------------------------------------------------------
|
|
||||||
* Definitions
|
#include "ringbuffer.h"
|
||||||
*----------------------------------------------------------------------------*/
|
|
||||||
|
/*----------------------------------------------------------------------------
|
||||||
/*----------------------------------------------------------------------------
|
* Definitions
|
||||||
* Variables
|
*----------------------------------------------------------------------------*/
|
||||||
*----------------------------------------------------------------------------*/
|
|
||||||
|
/*----------------------------------------------------------------------------
|
||||||
/** Is Console Initialized. */
|
* Variables
|
||||||
static uint8_t _ucIsConsoleInitialized=0 ;
|
*----------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
/** Is Console Initialized. */
|
||||||
* \brief Configures an USART peripheral with the specified parameters.
|
static uint8_t _ucIsConsoleInitialized=0;
|
||||||
*
|
/** Ring buffer to queue data to be sent */
|
||||||
* \param baudrate Baudrate at which the USART should operate (in Hz).
|
static ringbuf uart_tx_buffer;
|
||||||
* \param masterClock Frequency of the system master clock (in Hz).
|
|
||||||
*/
|
/**
|
||||||
extern void UART_Configure( uint32_t baudrate, uint32_t masterClock)
|
* \brief Configures an USART peripheral with the specified parameters.
|
||||||
{
|
*
|
||||||
const Pin pPins[] = CONSOLE_PINS;
|
* \param baudrate Baudrate at which the USART should operate (in Hz).
|
||||||
Uart *pUart = CONSOLE_USART;
|
* \param masterClock Frequency of the system master clock (in Hz).
|
||||||
|
*/
|
||||||
/* Configure PIO */
|
extern void UART_Configure( uint32_t baudrate, uint32_t masterClock)
|
||||||
PIO_Configure(pPins, PIO_LISTSIZE(pPins));
|
{
|
||||||
|
const Pin pPins[] = CONSOLE_PINS;
|
||||||
/* Configure PMC */
|
Uart *pUart = CONSOLE_UART;
|
||||||
PMC->PMC_PCER0 = 1 << CONSOLE_ID;
|
|
||||||
|
/* Configure PIO */
|
||||||
/* Reset and disable receiver & transmitter */
|
PIO_Configure(pPins, PIO_LISTSIZE(pPins));
|
||||||
pUart->UART_CR = UART_CR_RSTRX | UART_CR_RSTTX
|
|
||||||
| UART_CR_RXDIS | UART_CR_TXDIS;
|
/* Configure PMC */
|
||||||
|
PMC->PMC_PCER0 = 1 << CONSOLE_ID;
|
||||||
/* Configure mode */
|
|
||||||
pUart->UART_MR = UART_MR_PAR_NO;
|
/* Reset and disable receiver & transmitter */
|
||||||
|
pUart->UART_CR = UART_CR_RSTRX | UART_CR_RSTTX
|
||||||
/* Configure baudrate */
|
| UART_CR_RXDIS | UART_CR_TXDIS;
|
||||||
/* Asynchronous, no oversampling */
|
|
||||||
pUart->UART_BRGR = (masterClock / baudrate) / 16;
|
/* Configure mode */
|
||||||
|
pUart->UART_MR = UART_MR_PAR_NO;
|
||||||
/* Disable PDC channel */
|
|
||||||
pUart->UART_PTCR = UART_PTCR_RXTDIS | UART_PTCR_TXTDIS;
|
/* Configure baudrate */
|
||||||
|
/* Asynchronous, no oversampling */
|
||||||
/* Enable receiver and transmitter */
|
//pUart->UART_BRGR = (masterClock / baudrate) / 16;
|
||||||
pUart->UART_CR = UART_CR_RXEN | UART_CR_TXEN;
|
if ((masterClock / baudrate) % 16 >= 7) {
|
||||||
|
pUart->UART_BRGR = ( masterClock / baudrate) / 16 + 1;
|
||||||
_ucIsConsoleInitialized=1 ;
|
} else {
|
||||||
}
|
pUart->UART_BRGR = ( masterClock / baudrate) / 16 + 0;
|
||||||
|
}
|
||||||
/**
|
|
||||||
* \brief Outputs a character on the UART line.
|
/* Disable PDC channel */
|
||||||
*
|
pUart->UART_PTCR = UART_PTCR_RXTDIS | UART_PTCR_TXTDIS;
|
||||||
* \note This function is synchronous (i.e. uses polling).
|
|
||||||
* \param c Character to send.
|
/* Reset transmit ring buffer */
|
||||||
*/
|
rbuf_reset(&uart_tx_buffer);
|
||||||
extern void UART_PutChar( uint8_t c )
|
|
||||||
{
|
/* Enable TX interrupts */
|
||||||
Uart *pUart=CONSOLE_USART ;
|
pUart->UART_IER = UART_IER_TXRDY;
|
||||||
|
NVIC_EnableIRQ(CONSOLE_IRQ);
|
||||||
if ( !_ucIsConsoleInitialized )
|
|
||||||
{
|
/* Enable receiver and transmitter */
|
||||||
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
|
pUart->UART_CR = UART_CR_RXEN | UART_CR_TXEN;
|
||||||
}
|
|
||||||
|
/* Remember the configuration is complete */
|
||||||
/* Wait for the transmitter to be ready */
|
_ucIsConsoleInitialized=1 ;
|
||||||
while ( (pUart->UART_SR & UART_SR_TXEMPTY) == 0 ) ;
|
}
|
||||||
|
|
||||||
/* Send character */
|
/**
|
||||||
pUart->UART_THR=c ;
|
* \brief Disables the USART peripheral and related IRQ
|
||||||
|
*/
|
||||||
}
|
void UART_Exit(void)
|
||||||
|
{
|
||||||
/**
|
if (!_ucIsConsoleInitialized) {
|
||||||
* \brief Input a character from the UART line.
|
return;
|
||||||
*
|
}
|
||||||
* \note This function is synchronous
|
|
||||||
* \return character received.
|
Uart *pUart = CONSOLE_UART;
|
||||||
*/
|
pUart->UART_IDR = UART_IDR_TXRDY;
|
||||||
extern uint32_t UART_GetChar( void )
|
pUart->UART_CR = UART_CR_RSTRX | UART_CR_RSTTX | UART_CR_RXDIS | UART_CR_TXDIS | UART_CR_RSTSTA;
|
||||||
{
|
PMC->PMC_PCDR0 = 1 << CONSOLE_ID;
|
||||||
Uart *pUart=CONSOLE_USART ;
|
NVIC_DisableIRQ(CONSOLE_IRQ);
|
||||||
|
}
|
||||||
if ( !_ucIsConsoleInitialized )
|
|
||||||
{
|
/** Interrupt Service routine to transmit queued data */
|
||||||
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
|
void CONSOLE_ISR(void)
|
||||||
}
|
{
|
||||||
|
Uart *uart = CONSOLE_UART;
|
||||||
while ( (pUart->UART_SR & UART_SR_RXRDY) == 0 ) ;
|
if (uart->UART_SR & UART_SR_TXRDY) {
|
||||||
|
if (!rbuf_is_empty(&uart_tx_buffer)) {
|
||||||
return pUart->UART_RHR ;
|
uart->UART_THR = rbuf_read(&uart_tx_buffer);
|
||||||
}
|
} else {
|
||||||
|
uart->UART_IDR = UART_IER_TXRDY;
|
||||||
/**
|
}
|
||||||
* \brief Check if there is Input from UART line.
|
}
|
||||||
*
|
}
|
||||||
* \return true if there is Input.
|
|
||||||
*/
|
/**
|
||||||
extern uint32_t UART_IsRxReady( void )
|
* \brief Outputs a character on the UART line.
|
||||||
{
|
*
|
||||||
Uart *pUart=CONSOLE_USART ;
|
* \note This function is asynchronous (i.e. uses a buffer and interrupt to complete the transfer).
|
||||||
|
* \param c Character to send.
|
||||||
if ( !_ucIsConsoleInitialized )
|
*/
|
||||||
{
|
void UART_PutChar( uint8_t uc )
|
||||||
UART_Configure( CONSOLE_BAUDRATE, BOARD_MCK ) ;
|
{
|
||||||
}
|
Uart *pUart = CONSOLE_UART ;
|
||||||
|
|
||||||
return (pUart->UART_SR & UART_SR_RXRDY) > 0 ;
|
/* Initialize console is not already done */
|
||||||
}
|
if ( !_ucIsConsoleInitialized )
|
||||||
|
{
|
||||||
/**
|
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
|
||||||
* Displays the content of the given frame on the UART0.
|
}
|
||||||
*
|
|
||||||
* \param pucFrame Pointer to the frame to dump.
|
if (!rbuf_is_full(&uart_tx_buffer)) {
|
||||||
* \param dwSize Buffer size in bytes.
|
rbuf_write(&uart_tx_buffer, uc);
|
||||||
*/
|
if (!(pUart->UART_IMR & UART_IMR_TXRDY)) {
|
||||||
extern void UART_DumpFrame( uint8_t* pucFrame, uint32_t dwSize )
|
pUart->UART_IER = UART_IER_TXRDY;
|
||||||
{
|
CONSOLE_ISR();
|
||||||
uint32_t dw ;
|
}
|
||||||
|
}
|
||||||
for ( dw=0 ; dw < dwSize ; dw++ )
|
}
|
||||||
{
|
|
||||||
printf( "%02X ", pucFrame[dw] ) ;
|
/**
|
||||||
}
|
* \brief Outputs a character on the UART line.
|
||||||
|
*
|
||||||
printf( "\n\r" ) ;
|
* \note This function is synchronous (i.e. uses polling and blocks until the transfer is complete).
|
||||||
}
|
* \param c Character to send.
|
||||||
|
*/
|
||||||
/**
|
void UART_PutChar_Sync( uint8_t uc )
|
||||||
* Displays the content of the given buffer on the UART0.
|
{
|
||||||
*
|
Uart *pUart = CONSOLE_UART ;
|
||||||
* \param pucBuffer Pointer to the buffer to dump.
|
|
||||||
* \param dwSize Buffer size in bytes.
|
/* Initialize console is not already done */
|
||||||
* \param dwAddress Start address to display
|
if ( !_ucIsConsoleInitialized )
|
||||||
*/
|
{
|
||||||
extern void UART_DumpMemory( uint8_t* pucBuffer, uint32_t dwSize, uint32_t dwAddress )
|
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
|
||||||
{
|
}
|
||||||
uint32_t i ;
|
|
||||||
uint32_t j ;
|
while (!(pUart->UART_SR & UART_SR_TXRDY)); /* Wait for transfer buffer to be empty */
|
||||||
uint32_t dwLastLineStart ;
|
pUart->UART_THR = uc; /* Send data to UART peripheral */
|
||||||
uint8_t* pucTmp ;
|
while (!(pUart->UART_SR & UART_SR_TXRDY)); /* Wait for transfer buffer to transferred to shift register */
|
||||||
|
while (!(pUart->UART_SR & UART_SR_TXEMPTY)); /* Wait for transfer shift register to be empty (i.e. transfer is complete) */
|
||||||
for ( i=0 ; i < (dwSize / 16) ; i++ )
|
}
|
||||||
{
|
|
||||||
printf( "0x%08X: ", (unsigned int)(dwAddress + (i*16)) ) ;
|
/**
|
||||||
pucTmp = (uint8_t*)&pucBuffer[i*16] ;
|
* \brief Input a character from the UART line.
|
||||||
|
*
|
||||||
for ( j=0 ; j < 4 ; j++ )
|
* \note This function is synchronous
|
||||||
{
|
* \return character received.
|
||||||
printf( "%02X%02X%02X%02X ", pucTmp[0], pucTmp[1], pucTmp[2], pucTmp[3] ) ;
|
*/
|
||||||
pucTmp += 4 ;
|
extern uint32_t UART_GetChar( void )
|
||||||
}
|
{
|
||||||
|
Uart *pUart = CONSOLE_UART ;
|
||||||
pucTmp=(uint8_t*)&pucBuffer[i*16] ;
|
|
||||||
|
if ( !_ucIsConsoleInitialized )
|
||||||
for ( j=0 ; j < 16 ; j++ )
|
{
|
||||||
{
|
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
|
||||||
UART_PutChar( *pucTmp++ ) ;
|
}
|
||||||
}
|
|
||||||
|
while ( (pUart->UART_SR & UART_SR_RXRDY) == 0 )
|
||||||
printf( "\n\r" ) ;
|
WDT_Restart(WDT);
|
||||||
}
|
|
||||||
|
return pUart->UART_RHR ;
|
||||||
if ( (dwSize%16) != 0 )
|
}
|
||||||
{
|
|
||||||
dwLastLineStart=dwSize - (dwSize%16) ;
|
/**
|
||||||
|
* \brief Check if there is Input from UART line.
|
||||||
printf( "0x%08X: ", (unsigned int)(dwAddress + dwLastLineStart) ) ;
|
*
|
||||||
for ( j=dwLastLineStart ; j < dwLastLineStart+16 ; j++ )
|
* \return true if there is Input.
|
||||||
{
|
*/
|
||||||
if ( (j!=dwLastLineStart) && (j%4 == 0) )
|
extern uint32_t UART_IsRxReady( void )
|
||||||
{
|
{
|
||||||
printf( " " ) ;
|
Uart *pUart = CONSOLE_UART;
|
||||||
}
|
|
||||||
|
if ( !_ucIsConsoleInitialized )
|
||||||
if ( j < dwSize )
|
{
|
||||||
{
|
UART_Configure( CONSOLE_BAUDRATE, BOARD_MCK ) ;
|
||||||
printf( "%02X", pucBuffer[j] ) ;
|
}
|
||||||
}
|
|
||||||
else
|
return (pUart->UART_SR & UART_SR_RXRDY) > 0 ;
|
||||||
{
|
}
|
||||||
printf(" ") ;
|
|
||||||
}
|
/**
|
||||||
}
|
* Displays the content of the given frame on the UART0.
|
||||||
|
*
|
||||||
printf( " " ) ;
|
* \param pucFrame Pointer to the frame to dump.
|
||||||
for ( j=dwLastLineStart ; j < dwSize ; j++ )
|
* \param dwSize Buffer size in bytes.
|
||||||
{
|
*/
|
||||||
UART_PutChar( pucBuffer[j] ) ;
|
extern void UART_DumpFrame( uint8_t* pucFrame, uint32_t dwSize )
|
||||||
}
|
{
|
||||||
|
uint32_t dw ;
|
||||||
printf( "\n\r" ) ;
|
|
||||||
}
|
for ( dw=0 ; dw < dwSize ; dw++ )
|
||||||
}
|
{
|
||||||
|
printf( "%02X ", pucFrame[dw] ) ;
|
||||||
/**
|
}
|
||||||
* Reads an integer
|
|
||||||
*
|
printf( "\n\r" ) ;
|
||||||
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
|
}
|
||||||
*/
|
|
||||||
extern uint32_t UART_GetInteger( uint32_t* pdwValue )
|
/**
|
||||||
{
|
* Displays the content of the given buffer on the UART0.
|
||||||
uint8_t ucKey ;
|
*
|
||||||
uint8_t ucNbNb=0 ;
|
* \param pucBuffer Pointer to the buffer to dump.
|
||||||
uint32_t dwValue=0 ;
|
* \param dwSize Buffer size in bytes.
|
||||||
|
* \param dwAddress Start address to display
|
||||||
while ( 1 )
|
*/
|
||||||
{
|
extern void UART_DumpMemory( uint8_t* pucBuffer, uint32_t dwSize, uint32_t dwAddress )
|
||||||
ucKey=UART_GetChar() ;
|
{
|
||||||
UART_PutChar( ucKey ) ;
|
uint32_t i ;
|
||||||
|
uint32_t j ;
|
||||||
if ( ucKey >= '0' && ucKey <= '9' )
|
uint32_t dwLastLineStart ;
|
||||||
{
|
uint8_t* pucTmp ;
|
||||||
dwValue = (dwValue * 10) + (ucKey - '0');
|
|
||||||
ucNbNb++ ;
|
for ( i=0 ; i < (dwSize / 16) ; i++ )
|
||||||
}
|
{
|
||||||
else
|
printf( "0x%08X: ", (unsigned int)(dwAddress + (i*16)) ) ;
|
||||||
{
|
pucTmp = (uint8_t*)&pucBuffer[i*16] ;
|
||||||
if ( ucKey == 0x0D || ucKey == ' ' )
|
|
||||||
{
|
for ( j=0 ; j < 4 ; j++ )
|
||||||
if ( ucNbNb == 0 )
|
{
|
||||||
{
|
printf( "%02X%02X%02X%02X ", pucTmp[0], pucTmp[1], pucTmp[2], pucTmp[3] ) ;
|
||||||
printf( "\n\rWrite a number and press ENTER or SPACE!\n\r" ) ;
|
pucTmp += 4 ;
|
||||||
return 0 ;
|
}
|
||||||
}
|
|
||||||
else
|
pucTmp=(uint8_t*)&pucBuffer[i*16] ;
|
||||||
{
|
|
||||||
printf( "\n\r" ) ;
|
for ( j=0 ; j < 16 ; j++ )
|
||||||
*pdwValue=dwValue ;
|
{
|
||||||
|
UART_PutChar( *pucTmp++ ) ;
|
||||||
return 1 ;
|
}
|
||||||
}
|
|
||||||
}
|
printf( "\n\r" ) ;
|
||||||
else
|
}
|
||||||
{
|
|
||||||
printf( "\n\r'%c' not a number!\n\r", ucKey ) ;
|
if ( (dwSize%16) != 0 )
|
||||||
|
{
|
||||||
return 0 ;
|
dwLastLineStart=dwSize - (dwSize%16) ;
|
||||||
}
|
|
||||||
}
|
printf( "0x%08X: ", (unsigned int)(dwAddress + dwLastLineStart) ) ;
|
||||||
}
|
for ( j=dwLastLineStart ; j < dwLastLineStart+16 ; j++ )
|
||||||
}
|
{
|
||||||
|
if ( (j!=dwLastLineStart) && (j%4 == 0) )
|
||||||
/**
|
{
|
||||||
* Reads an integer and check the value
|
printf( " " ) ;
|
||||||
*
|
}
|
||||||
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
|
|
||||||
* \param dwMin Minimum value
|
if ( j < dwSize )
|
||||||
* \param dwMax Maximum value
|
{
|
||||||
*/
|
printf( "%02X", pucBuffer[j] ) ;
|
||||||
extern uint32_t UART_GetIntegerMinMax( uint32_t* pdwValue, uint32_t dwMin, uint32_t dwMax )
|
}
|
||||||
{
|
else
|
||||||
uint32_t dwValue=0 ;
|
{
|
||||||
|
printf(" ") ;
|
||||||
if ( UART_GetInteger( &dwValue ) == 0 )
|
}
|
||||||
{
|
}
|
||||||
return 0 ;
|
|
||||||
}
|
printf( " " ) ;
|
||||||
|
for ( j=dwLastLineStart ; j < dwSize ; j++ )
|
||||||
if ( dwValue < dwMin || dwValue > dwMax )
|
{
|
||||||
{
|
UART_PutChar( pucBuffer[j] ) ;
|
||||||
printf( "\n\rThe number have to be between %d and %d\n\r", (int)dwMin, (int)dwMax ) ;
|
}
|
||||||
|
|
||||||
return 0 ;
|
printf( "\n\r" ) ;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
printf( "\n\r" ) ;
|
|
||||||
|
/**
|
||||||
*pdwValue = dwValue ;
|
* Reads an integer
|
||||||
|
*
|
||||||
return 1 ;
|
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
|
||||||
}
|
*/
|
||||||
|
extern uint32_t UART_GetInteger( uint32_t* pdwValue )
|
||||||
/**
|
{
|
||||||
* Reads an hexadecimal number
|
uint8_t ucKey ;
|
||||||
*
|
uint8_t ucNbNb=0 ;
|
||||||
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
|
uint32_t dwValue=0 ;
|
||||||
*/
|
|
||||||
extern uint32_t UART_GetHexa32( uint32_t* pdwValue )
|
while ( 1 )
|
||||||
{
|
{
|
||||||
uint8_t ucKey ;
|
ucKey=UART_GetChar() ;
|
||||||
uint32_t dw = 0 ;
|
UART_PutChar( ucKey ) ;
|
||||||
uint32_t dwValue = 0 ;
|
|
||||||
|
if ( ucKey >= '0' && ucKey <= '9' )
|
||||||
for ( dw=0 ; dw < 8 ; dw++ )
|
{
|
||||||
{
|
dwValue = (dwValue * 10) + (ucKey - '0');
|
||||||
ucKey = UART_GetChar() ;
|
ucNbNb++ ;
|
||||||
UART_PutChar( ucKey ) ;
|
}
|
||||||
|
else
|
||||||
if ( ucKey >= '0' && ucKey <= '9' )
|
{
|
||||||
{
|
if ( ucKey == 0x0D || ucKey == ' ' )
|
||||||
dwValue = (dwValue * 16) + (ucKey - '0') ;
|
{
|
||||||
}
|
if ( ucNbNb == 0 )
|
||||||
else
|
{
|
||||||
{
|
printf( "\n\rWrite a number and press ENTER or SPACE!\n\r" ) ;
|
||||||
if ( ucKey >= 'A' && ucKey <= 'F' )
|
return 0 ;
|
||||||
{
|
}
|
||||||
dwValue = (dwValue * 16) + (ucKey - 'A' + 10) ;
|
else
|
||||||
}
|
{
|
||||||
else
|
printf( "\n\r" ) ;
|
||||||
{
|
*pdwValue=dwValue ;
|
||||||
if ( ucKey >= 'a' && ucKey <= 'f' )
|
|
||||||
{
|
return 1 ;
|
||||||
dwValue = (dwValue * 16) + (ucKey - 'a' + 10) ;
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
printf( "\n\rIt is not a hexa character!\n\r" ) ;
|
printf( "\n\r'%c' not a number!\n\r", ucKey ) ;
|
||||||
|
|
||||||
return 0 ;
|
return 0 ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
WDT_Restart(WDT);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
printf("\n\r" ) ;
|
|
||||||
*pdwValue = dwValue ;
|
/**
|
||||||
|
* Reads an integer and check the value
|
||||||
return 1 ;
|
*
|
||||||
}
|
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
|
||||||
|
* \param dwMin Minimum value
|
||||||
#if defined __ICCARM__ /* IAR Ewarm 5.41+ */
|
* \param dwMax Maximum value
|
||||||
/**
|
*/
|
||||||
* \brief Outputs a character on the UART.
|
extern uint32_t UART_GetIntegerMinMax( uint32_t* pdwValue, uint32_t dwMin, uint32_t dwMax )
|
||||||
*
|
{
|
||||||
* \param c Character to output.
|
uint32_t dwValue=0 ;
|
||||||
*
|
|
||||||
* \return The character that was output.
|
if ( UART_GetInteger( &dwValue ) == 0 )
|
||||||
*/
|
{
|
||||||
extern WEAK signed int putchar( signed int c )
|
return 0 ;
|
||||||
{
|
}
|
||||||
UART_PutChar( c ) ;
|
|
||||||
|
if ( dwValue < dwMin || dwValue > dwMax )
|
||||||
return c ;
|
{
|
||||||
}
|
printf( "\n\rThe number have to be between %d and %d\n\r", (int)dwMin, (int)dwMax ) ;
|
||||||
#endif // defined __ICCARM__
|
|
||||||
|
return 0 ;
|
||||||
|
}
|
||||||
|
|
||||||
|
printf( "\n\r" ) ;
|
||||||
|
|
||||||
|
*pdwValue = dwValue ;
|
||||||
|
|
||||||
|
return 1 ;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads an hexadecimal number
|
||||||
|
*
|
||||||
|
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
|
||||||
|
*/
|
||||||
|
extern uint32_t UART_GetHexa32( uint32_t* pdwValue )
|
||||||
|
{
|
||||||
|
uint8_t ucKey ;
|
||||||
|
uint32_t dw = 0 ;
|
||||||
|
uint32_t dwValue = 0 ;
|
||||||
|
|
||||||
|
for ( dw=0 ; dw < 8 ; dw++ )
|
||||||
|
{
|
||||||
|
ucKey = UART_GetChar() ;
|
||||||
|
UART_PutChar( ucKey ) ;
|
||||||
|
|
||||||
|
if ( ucKey >= '0' && ucKey <= '9' )
|
||||||
|
{
|
||||||
|
dwValue = (dwValue * 16) + (ucKey - '0') ;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ( ucKey >= 'A' && ucKey <= 'F' )
|
||||||
|
{
|
||||||
|
dwValue = (dwValue * 16) + (ucKey - 'A' + 10) ;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ( ucKey >= 'a' && ucKey <= 'f' )
|
||||||
|
{
|
||||||
|
dwValue = (dwValue * 16) + (ucKey - 'a' + 10) ;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf( "\n\rIt is not a hexa character!\n\r" ) ;
|
||||||
|
|
||||||
|
return 0 ;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("\n\r" ) ;
|
||||||
|
*pdwValue = dwValue ;
|
||||||
|
|
||||||
|
return 1 ;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if defined __ICCARM__ /* IAR Ewarm 5.41+ */
|
||||||
|
/**
|
||||||
|
* \brief Outputs a character on the UART.
|
||||||
|
*
|
||||||
|
* \param c Character to output.
|
||||||
|
*
|
||||||
|
* \return The character that was output.
|
||||||
|
*/
|
||||||
|
extern WEAK signed int putchar( signed int c )
|
||||||
|
{
|
||||||
|
UART_PutChar( c ) ;
|
||||||
|
|
||||||
|
return c ;
|
||||||
|
}
|
||||||
|
#endif // defined __ICCARM__
|
||||||
|
|
||||||
|
|||||||
170
firmware/libboard/octsimtest/include/board.h
Normal file
170
firmware/libboard/octsimtest/include/board.h
Normal file
@@ -0,0 +1,170 @@
|
|||||||
|
/* octSIMtest with SAM3S board definition
|
||||||
|
*
|
||||||
|
* (C) 2019 by sysmocom -s.f.m.c. GmbH, Author:Joachim Steiger <jsteiger@sysmocom.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
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
|
/* Name of the board */
|
||||||
|
#define BOARD_NAME "OCTSIMTEST"
|
||||||
|
/* Board definition */
|
||||||
|
#define octsimtest
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
|
#define BOARD_MAINOSC 18432000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58982400 // 18.432 * 16 / 5
|
||||||
|
|
||||||
|
/** Pin configuration **/
|
||||||
|
|
||||||
|
/** there is no red LED, but the code needs this second LED, thus we provide an unused pin */
|
||||||
|
#define PIO_LED_RED PIO_PB13
|
||||||
|
/** MCU pin connected to green LED, which is actually amber, and the logic is inverted since it is connected to an NPN transistor (used as open drain) */
|
||||||
|
#define PIO_LED_GREEN PIO_PA4
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOB, ID_PIOB, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/* Button to force bootloader start (shorted to ground when pressed */
|
||||||
|
#define PIN_BOOTLOADER_SW {PIO_PA5, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP}
|
||||||
|
|
||||||
|
//FIXME SIM_PWEN_PIN collides with PA5/bootloader_sw on octsimtest
|
||||||
|
/* Enable powering the card using the second 3.3 V output of the LDO (active high) */
|
||||||
|
#define SIM_PWEN_PIN {PIO_PA12, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Enable powering the SIM card */
|
||||||
|
#define PWR_PINS SIM_PWEN_PIN
|
||||||
|
|
||||||
|
// FIXME PA8 is 32khz xtal on octsimtest
|
||||||
|
/* Card presence pin */
|
||||||
|
#define SW_SIM PIO_PA11
|
||||||
|
/* Pull card presence pin high (shorted to ground in card slot when card is present) */
|
||||||
|
#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_DEGLITCH | PIO_IT_EDGE }
|
||||||
|
|
||||||
|
/** Smart card connection **/
|
||||||
|
//FIXME
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_SIM_RST {PIO_PA13, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Card I/O data signal input/output (I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO {PIO_PA6A_TXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Card CLK clock input (CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK {PIO_PA2B_SCK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin to measure card I/O timing (to start measuring the ETU on I/O activity; connected I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO_INPUT {PIO_PA1B_TIOB0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
//FIXME PIO_PA4B_TCLK0 PA4 is LED on octsimtest
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK_INPUT {PIO_PA14, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pins used to measure ETU timing (using timer counter) */
|
||||||
|
#define PINS_TC PIN_SIM_IO_INPUT, PIN_SIM_CLK_INPUT
|
||||||
|
|
||||||
|
/** Phone connection **/
|
||||||
|
/* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_VCC {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
/* Phone USIM slot 1 RST pin (active low; RST_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_nRST {PIO_PA24, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_RISE_EDGE | PIO_DEGLITCH }
|
||||||
|
/* Phone I/O data signal input/output (I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO {PIO_PA22A_TXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Phone CLK clock input (CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK {PIO_PA23A_SCK1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used for phone USIM slot 1 communication */
|
||||||
|
#define PINS_USIM1 PIN_PHONE_IO, PIN_PHONE_CLK, PIN_PHONE_CLK_INPUT, PIN_USIM1_VCC, PIN_PHONE_IO_INPUT, PIN_USIM1_nRST
|
||||||
|
/* Phone I/O data signal input/output (unused USART RX input; connected to I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO_INPUT {PIO_PA21A_RXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK_INPUT {PIO_PA29B_TCLK2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Default pin configuration **/
|
||||||
|
/* Disconnect VPP, CLK, and RST lines between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_SC_SW_DEFAULT {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect I/O line between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_IO_SW_DEFAULT {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_DEFAULT PIN_SC_SW_DEFAULT, PIN_IO_SW_DEFAULT
|
||||||
|
|
||||||
|
/** Sniffer configuration **/
|
||||||
|
/* Connect VPP, CLK, and RST lines between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_SC_SW_SNIFF {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect I/O line between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_IO_SW_SNIFF {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_SNIFF PIN_SC_SW_SNIFF, PIN_IO_SW_SNIFF
|
||||||
|
/* Card RST reset signal input (use as input since the phone will drive it) */
|
||||||
|
#define PIN_SIM_RST_SNIFF {PIO_PA7, PIOA, ID_PIOA, PIO_INPUT, PIO_DEGLITCH | PIO_IT_EDGE}
|
||||||
|
/* Pins used to sniff phone-card communication */
|
||||||
|
#define PINS_SIM_SNIFF PIN_SIM_IO, PIN_SIM_CLK, PIN_SIM_RST_SNIFF
|
||||||
|
/* Disable power converter 4.5-6V to 3.3V (active high) */
|
||||||
|
#define PIN_SIM_PWEN_SNIFF {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Enable power switch to forward VCC_PHONE to VCC_SIM (active high) */
|
||||||
|
#define PIN_VCC_FWD_SNIFF {VCC_FWD, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Use phone VCC to power card */
|
||||||
|
#define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF
|
||||||
|
|
||||||
|
/** CCID configuration */
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_ISO7816_RSTMC {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* ISO7816-communication related pins */
|
||||||
|
#define PINS_ISO7816 PIN_SIM_IO, PIN_SIM_CLK, PIN_ISO7816_RSTMC // SIM_PWEN_PIN, PIN_SIM_IO2, PIN_SIM_CLK2
|
||||||
|
|
||||||
|
/** External SPI flash interface **/
|
||||||
|
/* SPI MISO pin definition */
|
||||||
|
#define PIN_SPI_MISO {PIO_PA12A_MISO, PIOA, PIOA, PIO_PERIPH_A, PIO_PULLUP}
|
||||||
|
/* SPI MOSI pin definition */
|
||||||
|
#define PIN_SPI_MOSI {PIO_PA13A_MOSI, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI SCK pin definition */
|
||||||
|
#define PIN_SPI_SCK {PIO_PA14A_SPCK, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI pins definition. Contains MISO, MOSI & SCK */
|
||||||
|
#define PINS_SPI PIN_SPI_MISO, PIN_SPI_MOSI, PIN_SPI_SCK
|
||||||
|
/* SPI chip select 0 pin definition */
|
||||||
|
#define PIN_SPI_NPCS0 {PIO_PA11A_NPCS0, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI flash write protect pin (active low, pulled low) */
|
||||||
|
#define PIN_SPI_WP {PA15, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Pin configuration to control USB pull-up on D+
|
||||||
|
* @details the USB pull-up on D+ is enable by default on the board but can be disabled by setting PA16 high
|
||||||
|
*/
|
||||||
|
#define PIN_USB_PULLUP {PIO_PA16, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** USB definitions */
|
||||||
|
/* OpenMoko SIMtrace 2 USB vendor ID */
|
||||||
|
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO
|
||||||
|
/* OpenMoko SIMtrace 2 USB product ID (main application/runtime mode) */
|
||||||
|
#define BOARD_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2
|
||||||
|
/* OpenMoko SIMtrace 2 DFU USB product ID (DFU bootloader/DFU mode) */
|
||||||
|
#define BOARD_DFU_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2_DFU
|
||||||
|
/* USB release number (bcdDevice, shown as 0.00) */
|
||||||
|
#define BOARD_USB_RELEASE 0x000
|
||||||
|
/* Indicate SIMtrace is bus power in USB attributes */
|
||||||
|
#define BOARD_USB_BMATTRIBUTES USBConfigurationDescriptor_BUSPOWERED_NORWAKEUP
|
||||||
|
|
||||||
|
/** Supported modes */
|
||||||
|
/* SIMtrace board supports sniffer mode */
|
||||||
|
//#define HAVE_SNIFFER
|
||||||
|
/* SIMtrace board supports CCID mode */
|
||||||
|
//#define HAVE_CCID
|
||||||
|
/* SIMtrace board supports card emulation mode */
|
||||||
|
//#define HAVE_CARDEM
|
||||||
|
/* SIMtrace board supports man-in-the-middle mode */
|
||||||
|
//#define HAVE_MITM
|
||||||
|
/* octsimtest board supports gpio_test mode */
|
||||||
|
#define HAVE_GPIO_TEST
|
||||||
28
firmware/libboard/octsimtest/include/i2c.h
Normal file
28
firmware/libboard/octsimtest/include/i2c.h
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
/* I2C EEPROM memory read and write utilities
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
void i2c_pin_init(void);
|
||||||
|
|
||||||
|
bool i2c_write_byte(bool send_start, bool send_stop, uint8_t byte);
|
||||||
|
uint8_t i2c_read_byte(bool nack, bool send_stop);
|
||||||
|
void i2c_stop_cond(void);
|
||||||
|
|
||||||
|
int eeprom_write_byte(uint8_t slave, uint8_t addr, uint8_t byte);
|
||||||
|
int eeprom_read_byte(uint8_t slave, uint8_t addr);
|
||||||
25
firmware/libboard/octsimtest/include/mcp23017.h
Normal file
25
firmware/libboard/octsimtest/include/mcp23017.h
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
/* mcp23017 i2c gpio expander read and write utilities
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#define MCP23017_ADDRESS 0x20
|
||||||
|
|
||||||
|
int mcp23017_init(uint8_t slave);
|
||||||
|
int mcp23017_test(uint8_t slave);
|
||||||
|
int mcp23017_toggle(uint8_t slave);
|
||||||
|
//int mcp23017_write_byte(uint8_t slave, uint8_t addr, uint8_t byte);
|
||||||
|
//int mcp23017_read_byte(uint8_t slave, uint8_t addr);
|
||||||
81
firmware/libboard/octsimtest/source/board_octsimtest.c
Normal file
81
firmware/libboard/octsimtest/source/board_octsimtest.c
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
/* SIMtrace with SAM3S specific application code
|
||||||
|
*
|
||||||
|
* (C) 2017 by Harald Welte <laforge@gnumonks.org>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
|
#include "board.h"
|
||||||
|
#include "simtrace.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "sim_switch.h"
|
||||||
|
#include <osmocom/core/timer.h>
|
||||||
|
#include "usb_buf.h"
|
||||||
|
#include "i2c.h"
|
||||||
|
#include "mcp23017.h"
|
||||||
|
|
||||||
|
void board_exec_dbg_cmd(int ch)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case '?':
|
||||||
|
printf("\t?\thelp\n\r");
|
||||||
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
printf("\tm\trun mcp23017 test\n\r");
|
||||||
|
printf("\tR\ttoggle MSB of gpio on mcp23017\n\r");
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
printf("Asking NVIC to reset us\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
break;
|
||||||
|
case 'm':
|
||||||
|
mcp23017_test(MCP23017_ADDRESS);
|
||||||
|
break;
|
||||||
|
case 't':
|
||||||
|
mcp23017_toggle(MCP23017_ADDRESS);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void board_main_top(void)
|
||||||
|
{
|
||||||
|
#ifndef APPLICATION_dfu
|
||||||
|
usb_buf_init();
|
||||||
|
|
||||||
|
i2c_pin_init();
|
||||||
|
if (!mcp23017_init(MCP23017_ADDRESS))
|
||||||
|
printf("mcp23017 not found!\n\r");
|
||||||
|
/* Initialize checking for card insert/remove events */
|
||||||
|
//card_present_init();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
const Pin bl_sw_pin = PIN_BOOTLOADER_SW;
|
||||||
|
|
||||||
|
PIO_Configure(&bl_sw_pin, 1);
|
||||||
|
|
||||||
|
/* Enter DFU bootloader in case the respective button is pressed */
|
||||||
|
if (PIO_Get(&bl_sw_pin) == 0) {
|
||||||
|
/* do not print to early since the console is not initialized yet */
|
||||||
|
//printf("BOOTLOADER switch pressed -> Force DFU\n\r");
|
||||||
|
return 1;
|
||||||
|
} else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
225
firmware/libboard/octsimtest/source/i2c.c
Normal file
225
firmware/libboard/octsimtest/source/i2c.c
Normal file
@@ -0,0 +1,225 @@
|
|||||||
|
/* I2C EEPROM memory read and write utilities
|
||||||
|
*
|
||||||
|
* 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 "board.h"
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
/* Low-Level I2C Routines */
|
||||||
|
|
||||||
|
static const Pin pin_sda = {PIO_PA30, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_OPENDRAIN };
|
||||||
|
static const Pin pin_sda_in = {PIO_PA30, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT };
|
||||||
|
static const Pin pin_scl = {PIO_PA31, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_OPENDRAIN };
|
||||||
|
|
||||||
|
static void i2c_delay()
|
||||||
|
{
|
||||||
|
volatile int v;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* 100 cycles results in SCL peak length of 44us, so it's about
|
||||||
|
* 440ns per cycle here */
|
||||||
|
for (i = 0; i < 14; i++) {
|
||||||
|
v = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void i2c_pin_init(void)
|
||||||
|
{
|
||||||
|
PIO_Configure(&pin_scl, PIO_LISTSIZE(pin_scl));
|
||||||
|
PIO_Configure(&pin_sda, PIO_LISTSIZE(pin_sda));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void set_scl(void)
|
||||||
|
{
|
||||||
|
PIO_Set(&pin_scl);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void set_sda(void)
|
||||||
|
{
|
||||||
|
PIO_Set(&pin_sda);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void clear_scl(void)
|
||||||
|
{
|
||||||
|
PIO_Clear(&pin_scl);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void clear_sda(void)
|
||||||
|
{
|
||||||
|
PIO_Clear(&pin_sda);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool read_sda(void)
|
||||||
|
{
|
||||||
|
bool ret;
|
||||||
|
|
||||||
|
PIO_Configure(&pin_sda_in, PIO_LISTSIZE(pin_sda_in));
|
||||||
|
if (PIO_Get(&pin_sda_in))
|
||||||
|
ret = true;
|
||||||
|
else
|
||||||
|
ret = false;
|
||||||
|
PIO_Configure(&pin_sda, PIO_LISTSIZE(pin_sda));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Core I2C Routines */
|
||||||
|
|
||||||
|
static bool i2c_started = false;
|
||||||
|
|
||||||
|
static void i2c_start_cond(void)
|
||||||
|
{
|
||||||
|
if (i2c_started) {
|
||||||
|
set_sda();
|
||||||
|
set_scl();
|
||||||
|
}
|
||||||
|
|
||||||
|
clear_sda();
|
||||||
|
i2c_delay();
|
||||||
|
clear_scl();
|
||||||
|
i2c_started = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void i2c_stop_cond(void)
|
||||||
|
{
|
||||||
|
clear_sda();
|
||||||
|
set_scl();
|
||||||
|
set_sda();
|
||||||
|
i2c_delay();
|
||||||
|
i2c_started = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void i2c_write_bit(bool bit)
|
||||||
|
{
|
||||||
|
if (bit)
|
||||||
|
set_sda();
|
||||||
|
else
|
||||||
|
clear_sda();
|
||||||
|
i2c_delay(); // ?
|
||||||
|
set_scl();
|
||||||
|
clear_scl();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool i2c_read_bit(void)
|
||||||
|
{
|
||||||
|
bool bit;
|
||||||
|
|
||||||
|
set_sda();
|
||||||
|
set_scl();
|
||||||
|
bit = read_sda();
|
||||||
|
clear_scl();
|
||||||
|
|
||||||
|
return bit;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool i2c_write_byte(bool send_start, bool send_stop, uint8_t byte)
|
||||||
|
{
|
||||||
|
uint8_t bit;
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
if (send_start)
|
||||||
|
i2c_start_cond();
|
||||||
|
|
||||||
|
for (bit = 0; bit < 8; bit++) {
|
||||||
|
i2c_write_bit((byte & 0x80) != 0);
|
||||||
|
byte <<= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
nack = i2c_read_bit();
|
||||||
|
|
||||||
|
if (send_stop)
|
||||||
|
i2c_stop_cond();
|
||||||
|
|
||||||
|
return nack;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t i2c_read_byte(bool nack, bool send_stop)
|
||||||
|
{
|
||||||
|
uint8_t byte = 0;
|
||||||
|
uint8_t bit;
|
||||||
|
|
||||||
|
for (bit = 0; bit < 8; bit++) {
|
||||||
|
byte = (byte << 1) | i2c_read_bit();
|
||||||
|
}
|
||||||
|
|
||||||
|
i2c_write_bit(nack);
|
||||||
|
|
||||||
|
if (send_stop)
|
||||||
|
i2c_stop_cond();
|
||||||
|
|
||||||
|
return byte;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* EEPROM related code */
|
||||||
|
|
||||||
|
int eeprom_write_byte(uint8_t slave, uint8_t addr, uint8_t byte)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
/* Write slave address */
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, true, byte);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
/* Wait tWR time to ensure EEPROM is writing correctly (tWR = 5 ms for AT24C02) */
|
||||||
|
mdelay(5);
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int eeprom_read_byte(uint8_t slave, uint8_t addr)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
/* dummy write cycle */
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
/* Re-start with read */
|
||||||
|
nack = i2c_write_byte(true, false, (slave << 1) | 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
|
||||||
|
return i2c_read_byte(true, true);
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
140
firmware/libboard/octsimtest/source/mcp23017.c
Normal file
140
firmware/libboard/octsimtest/source/mcp23017.c
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
#include "board.h"
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include "i2c.h"
|
||||||
|
#include "mcp23017.h"
|
||||||
|
|
||||||
|
|
||||||
|
//defines from https://github.com/adafruit/Adafruit-MCP23017-Arduino-Library/blob/master/Adafruit_MCP23017.h under BSD license
|
||||||
|
|
||||||
|
// registers
|
||||||
|
#define MCP23017_IODIRA 0x00
|
||||||
|
#define MCP23017_IPOLA 0x02
|
||||||
|
#define MCP23017_GPINTENA 0x04
|
||||||
|
#define MCP23017_DEFVALA 0x06
|
||||||
|
#define MCP23017_INTCONA 0x08
|
||||||
|
#define MCP23017_IOCONA 0x0A
|
||||||
|
#define MCP23017_GPPUA 0x0C
|
||||||
|
#define MCP23017_INTFA 0x0E
|
||||||
|
#define MCP23017_INTCAPA 0x10
|
||||||
|
#define MCP23017_GPIOA 0x12
|
||||||
|
#define MCP23017_OLATA 0x14
|
||||||
|
|
||||||
|
|
||||||
|
#define MCP23017_IODIRB 0x01
|
||||||
|
#define MCP23017_IPOLB 0x03
|
||||||
|
#define MCP23017_GPINTENB 0x05
|
||||||
|
#define MCP23017_DEFVALB 0x07
|
||||||
|
#define MCP23017_INTCONB 0x09
|
||||||
|
#define MCP23017_IOCONB 0x0B
|
||||||
|
#define MCP23017_GPPUB 0x0D
|
||||||
|
#define MCP23017_INTFB 0x0F
|
||||||
|
#define MCP23017_INTCAPB 0x11
|
||||||
|
#define MCP23017_GPIOB 0x13
|
||||||
|
#define MCP23017_OLATB 0x15
|
||||||
|
|
||||||
|
#define MCP23017_INT_ERR 255
|
||||||
|
|
||||||
|
|
||||||
|
//bool i2c_write_byte(bool send_start, bool send_stop, uint8_t byte)
|
||||||
|
//uint8_t i2c_read_byte(bool nack, bool send_stop)
|
||||||
|
//static void i2c_stop_cond(void)
|
||||||
|
|
||||||
|
int mcp23017_write_byte(uint8_t slave, uint8_t addr, uint8_t byte)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
// Write slave address
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, true, byte);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_read_byte(uint8_t slave, uint8_t addr)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
// dummy write cycle
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
// Re-start with read
|
||||||
|
nack = i2c_write_byte(true, false, (slave << 1) | 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
|
||||||
|
return i2c_read_byte(true, true);
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_init(uint8_t slave)
|
||||||
|
{
|
||||||
|
printf("mcp23017_init\n\r");
|
||||||
|
// all gpio input
|
||||||
|
if (mcp23017_write_byte(slave, MCP23017_IODIRA, 0xff))
|
||||||
|
return false;
|
||||||
|
// msb of portb output, rest input
|
||||||
|
if (mcp23017_write_byte(slave, MCP23017_IODIRB, 0x7f))
|
||||||
|
return false;
|
||||||
|
if (mcp23017_write_byte(slave, MCP23017_IOCONA, 0x20)) //disable SEQOP (autoinc addressing)
|
||||||
|
return false;
|
||||||
|
printf("mcp23017 found\n\r");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_test(uint8_t slave)
|
||||||
|
{
|
||||||
|
printf("mcp23017_test\n\r");
|
||||||
|
printf("GPIOA 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_GPIOA));
|
||||||
|
printf("GPIOB 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_GPIOB));
|
||||||
|
printf("IODIRA 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IODIRA));
|
||||||
|
printf("IODIRB 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IODIRB));
|
||||||
|
printf("IOCONA 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IOCONA));
|
||||||
|
printf("IOCONB 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IOCONB));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_toggle(uint8_t slave)
|
||||||
|
{
|
||||||
|
// example writing MSB of gpio
|
||||||
|
static bool foo=false;
|
||||||
|
if (foo)
|
||||||
|
{
|
||||||
|
printf("+\n\r");
|
||||||
|
mcp23017_write_byte(slave, MCP23017_OLATB, 0x80);
|
||||||
|
foo=false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf("-\n\r");
|
||||||
|
mcp23017_write_byte(slave, MCP23017_OLATB, 0x00);
|
||||||
|
foo=true;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -1,12 +1,53 @@
|
|||||||
|
/* OWHW board definition
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "board_common.h"
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
/** Name of the board */
|
/** Name of the board */
|
||||||
#define BOARD_NAME "OWHW"
|
#define BOARD_NAME "OWHW"
|
||||||
/** Board definition */
|
/** Board definition */
|
||||||
#define owhw
|
#define owhw
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
#define BOARD_MAINOSC 18432000
|
#define BOARD_MAINOSC 18432000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58982400 // 18.432 * 16 / 5
|
||||||
|
|
||||||
|
/** MCU pin connected to red LED */
|
||||||
|
#define PIO_LED_RED PIO_PA17
|
||||||
|
/** MCU pin connected to green LED */
|
||||||
|
#define PIO_LED_GREEN PIO_PA18
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/* pin connected to the SIMTRACE_BOOTLOADER signal. set high to force DFU bootloader start */
|
||||||
|
#define PIN_BOOTLOADER {PIO_PA31, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
|
||||||
/* USIM 2 interface (USART) */
|
/* USIM 2 interface (USART) */
|
||||||
#define PIN_USIM2_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
#define PIN_USIM2_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|||||||
@@ -1,28 +1,55 @@
|
|||||||
/* Card simulator specific functions */
|
/* Card simulator specific functions
|
||||||
/* (C) 2015 by Harald Welte <hwelte@hmw-consulting.de>
|
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
* it under the terms of the GNU General Public License as published by
|
* (C) 2018-2019, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
* 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,
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
* it under the terms of the GNU General Public License as published by
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
* GNU General Public License for more details.
|
* (at your option) any later version.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU General Public License
|
* This program is distributed in the hope that it will be useful,
|
||||||
* along with this program; if not, write to the Free Software
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
* 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 "chip.h"
|
#include "chip.h"
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
|
#include "usb_buf.h"
|
||||||
|
|
||||||
static const Pin pins_cardsim[] = PINS_CARDSIM;
|
static const Pin pins_cardsim[] = PINS_CARDSIM;
|
||||||
|
|
||||||
|
void board_exec_dbg_cmd(int ch)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case '?':
|
||||||
|
printf("\t?\thelp\n\r");
|
||||||
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
printf("Asking NVIC to reset us\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void board_main_top(void)
|
||||||
|
{
|
||||||
|
#ifndef APPLICATION_dfu
|
||||||
|
usb_buf_init();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
void cardsim_set_simpres(uint8_t slot, int present)
|
void cardsim_set_simpres(uint8_t slot, int present)
|
||||||
{
|
{
|
||||||
if (slot > 1)
|
if (slot > 1)
|
||||||
@@ -36,5 +63,18 @@ void cardsim_set_simpres(uint8_t slot, int present)
|
|||||||
|
|
||||||
void cardsim_gpio_init(void)
|
void cardsim_gpio_init(void)
|
||||||
{
|
{
|
||||||
PIO_Configure(&pins_cardsim, ARRAY_SIZE(pins_cardsim));
|
PIO_Configure(pins_cardsim, ARRAY_SIZE(pins_cardsim));
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
const Pin bl_pin = PIN_BOOTLOADER;
|
||||||
|
|
||||||
|
PIO_Configure(&bl_pin, 1);
|
||||||
|
|
||||||
|
if (PIO_Get(&bl_pin) == 0) { // signal low
|
||||||
|
return 0; // do not override enter DFU
|
||||||
|
} else {
|
||||||
|
return 1; // override enter DFU
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,53 @@
|
|||||||
|
/* sysmocom quad-modem sysmoQMOD board definition
|
||||||
|
*
|
||||||
|
* (C) 2016-2017 by Harald Welte <hwelte@hmw-consulting.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "board_common.h"
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
/** Name of the board */
|
/** Name of the board */
|
||||||
#define BOARD_NAME "QMOD"
|
#define BOARD_NAME "QMOD"
|
||||||
/** Board definition */
|
/** Board definition */
|
||||||
#define qmod
|
#define qmod
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
#define BOARD_MAINOSC 12000000
|
#define BOARD_MAINOSC 12000000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58000000 // 18.432 * 29 / 6
|
||||||
|
|
||||||
|
/** MCU pin connected to red LED */
|
||||||
|
#define PIO_LED_RED PIO_PA17
|
||||||
|
/** MCU pin connected to green LED */
|
||||||
|
#define PIO_LED_GREEN PIO_PA18
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
/** the green LED is actually red and used as indication for USIM1 */
|
||||||
|
#define LED_USIM1 LED_GREEN
|
||||||
|
/** the green LED is actually red and used as indication for USIM2 */
|
||||||
|
#define LED_USIM2 LED_RED
|
||||||
|
|
||||||
/* USIM 2 interface (USART) */
|
/* USIM 2 interface (USART) */
|
||||||
#define PIN_USIM2_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
#define PIN_USIM2_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|||||||
@@ -1,3 +1,21 @@
|
|||||||
|
/* card presence utilities
|
||||||
|
*
|
||||||
|
* (C) 2016-2017 by Harald Welte <hwelte@hmw-consulting.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
int is_card_present(int port);
|
int is_card_present(int port);
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* I2C EEPROM memory read and write utilities
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
void i2c_pin_init(void);
|
void i2c_pin_init(void);
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* Code to read/track the status of the WWAN LEDs of attached modems
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
int wwan_led_active(int wwan);
|
int wwan_led_active(int wwan);
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* Code to control the PERST lines of attached modems
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
int wwan_perst_set(int modem_nr, int active);
|
int wwan_perst_set(int modem_nr, int active);
|
||||||
|
|||||||
@@ -1,16 +1,34 @@
|
|||||||
/* Quad-modem speciic application code */
|
/* sysmocom quad-modem sysmoQMOD application code
|
||||||
/* (C) 2016-2016 by Harald Welte <laforge@gnumonks.org> */
|
*
|
||||||
|
* (C) 2016-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018-2019, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "simtrace.h"
|
#include "simtrace.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
|
#include "led.h"
|
||||||
#include "wwan_led.h"
|
#include "wwan_led.h"
|
||||||
#include "wwan_perst.h"
|
#include "wwan_perst.h"
|
||||||
#include "sim_switch.h"
|
#include "sim_switch.h"
|
||||||
#include "boardver_adc.h"
|
#include "boardver_adc.h"
|
||||||
#include "card_pres.h"
|
#include "card_pres.h"
|
||||||
#include "osmocom/core/timer.h"
|
#include <osmocom/core/timer.h>
|
||||||
#include "usb_buf.h"
|
#include "usb_buf.h"
|
||||||
|
#include "i2c.h"
|
||||||
|
|
||||||
static const Pin pin_hubpwr_override = PIN_PRTPWR_OVERRIDE;
|
static const Pin pin_hubpwr_override = PIN_PRTPWR_OVERRIDE;
|
||||||
static const Pin pin_hub_rst = {PIO_PA13, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT};
|
static const Pin pin_hub_rst = {PIO_PA13, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT};
|
||||||
@@ -29,6 +47,7 @@ static int qmod_sam3_is_12(void)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
const unsigned char __eeprom_bin[256] = {
|
const unsigned char __eeprom_bin[256] = {
|
||||||
USB_VENDOR_OPENMOKO & 0xff,
|
USB_VENDOR_OPENMOKO & 0xff,
|
||||||
USB_VENDOR_OPENMOKO >> 8,
|
USB_VENDOR_OPENMOKO >> 8,
|
||||||
@@ -52,11 +71,8 @@ const unsigned char __eeprom_bin[256] = {
|
|||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xA0, 0x56, 0x23, 0x71, 0x04, 0x00, /* 0xf0 - 0xff */
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xA0, 0x56, 0x23, 0x71, 0x04, 0x00, /* 0xf0 - 0xff */
|
||||||
};
|
};
|
||||||
|
|
||||||
#include "i2c.h"
|
|
||||||
static int write_hub_eeprom(void)
|
static int write_hub_eeprom(void)
|
||||||
{
|
{
|
||||||
const unsigned int __eeprom_bin_len = 256;
|
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* wait */
|
/* wait */
|
||||||
@@ -64,22 +80,25 @@ static int write_hub_eeprom(void)
|
|||||||
|
|
||||||
TRACE_INFO("Writing EEPROM...\n\r");
|
TRACE_INFO("Writing EEPROM...\n\r");
|
||||||
/* write the EEPROM once */
|
/* write the EEPROM once */
|
||||||
for (i = 0; i < 256; i++) {
|
for (i = 0; i < ARRAY_SIZE(__eeprom_bin); i++) {
|
||||||
int rc = eeprom_write_byte(0x50, i, __eeprom_bin[i]);
|
int rc = eeprom_write_byte(0x50, i, __eeprom_bin[i]);
|
||||||
/* if the result was negative, repeat that write */
|
if (rc < 0) {
|
||||||
if (rc < 0)
|
TRACE_ERROR("Writing EEPROM failed at byte %u: 0x%02x\n\r",
|
||||||
i--;
|
i, __eeprom_bin[i]);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* then pursue re-reading it again and again */
|
/* then pursue re-reading it again and again */
|
||||||
TRACE_INFO("Verifying EEPROM...\n\r");
|
TRACE_INFO("Verifying EEPROM...\n\r");
|
||||||
for (i = 0; i < 256; i++) {
|
for (i = 0; i < ARRAY_SIZE(__eeprom_bin); i++) {
|
||||||
int byte = eeprom_read_byte(0x50, i);
|
int byte = eeprom_read_byte(0x50, i);
|
||||||
TRACE_INFO("0x%02x: %02x\n\r", i, byte);
|
TRACE_DEBUG("0x%02x: %02x\n\r", i, byte);
|
||||||
if (byte != __eeprom_bin[i])
|
if (byte != __eeprom_bin[i])
|
||||||
TRACE_ERROR("Byte %u is wrong, expected 0x%02x, found 0x%02x\n\r",
|
TRACE_ERROR("Byte %u is wrong, expected 0x%02x, found 0x%02x\n\r",
|
||||||
i, __eeprom_bin[i], byte);
|
i, __eeprom_bin[i], byte);
|
||||||
}
|
}
|
||||||
|
TRACE_INFO("EEPROM written\n\r");
|
||||||
|
|
||||||
/* FIXME: Release PIN_PRTPWR_OVERRIDE after we know the hub is
|
/* FIXME: Release PIN_PRTPWR_OVERRIDE after we know the hub is
|
||||||
* again powering us up */
|
* again powering us up */
|
||||||
@@ -87,6 +106,29 @@ static int write_hub_eeprom(void)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int erase_hub_eeprom(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* wait */
|
||||||
|
mdelay(100);
|
||||||
|
|
||||||
|
TRACE_INFO("Erasing EEPROM...\n\r");
|
||||||
|
/* write the EEPROM once */
|
||||||
|
for (i = 0; i < 256; i++) {
|
||||||
|
int rc = eeprom_write_byte(0x50, i, 0xff);
|
||||||
|
if (rc < 0) {
|
||||||
|
TRACE_ERROR("Erasing EEPROM failed at byte %u: 0x%02x\n\r",
|
||||||
|
i, __eeprom_bin[i]);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
TRACE_INFO("EEPROM erased\n\r");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif /* ALLOW_PEER_ERASE */
|
||||||
|
|
||||||
static void board_exec_dbg_cmd_st12only(int ch)
|
static void board_exec_dbg_cmd_st12only(int ch)
|
||||||
{
|
{
|
||||||
uint32_t addr, val;
|
uint32_t addr, val;
|
||||||
@@ -96,9 +138,14 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
switch (ch) {
|
switch (ch) {
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
case 'E':
|
case 'E':
|
||||||
write_hub_eeprom();
|
write_hub_eeprom();
|
||||||
break;
|
break;
|
||||||
|
case 'e':
|
||||||
|
erase_hub_eeprom();
|
||||||
|
break;
|
||||||
|
#endif /* ALLOW_PEER_ERASE */
|
||||||
case 'O':
|
case 'O':
|
||||||
printf("Setting PRTPWR_OVERRIDE\n\r");
|
printf("Setting PRTPWR_OVERRIDE\n\r");
|
||||||
PIO_Set(&pin_hubpwr_override);
|
PIO_Set(&pin_hubpwr_override);
|
||||||
@@ -107,6 +154,7 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
printf("Clearing PRTPWR_OVERRIDE\n\r");
|
printf("Clearing PRTPWR_OVERRIDE\n\r");
|
||||||
PIO_Clear(&pin_hubpwr_override);
|
PIO_Clear(&pin_hubpwr_override);
|
||||||
break;
|
break;
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
case 'H':
|
case 'H':
|
||||||
printf("Clearing _HUB_RESET -> HUB_RESET high (inactive)\n\r");
|
printf("Clearing _HUB_RESET -> HUB_RESET high (inactive)\n\r");
|
||||||
PIO_Clear(&pin_hub_rst);
|
PIO_Clear(&pin_hub_rst);
|
||||||
@@ -123,13 +171,14 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
UART_GetIntegerMinMax(&addr, 0, 255);
|
UART_GetIntegerMinMax(&addr, 0, 255);
|
||||||
printf("Please enter EEPROM value:\n\r");
|
printf("Please enter EEPROM value:\n\r");
|
||||||
UART_GetIntegerMinMax(&val, 0, 255);
|
UART_GetIntegerMinMax(&val, 0, 255);
|
||||||
printf("Writing value 0x%02x to EEPROM offset 0x%02x\n\r", val, addr);
|
printf("Writing value 0x%02lx to EEPROM offset 0x%02lx\n\r", val, addr);
|
||||||
eeprom_write_byte(0x50, addr, val);
|
eeprom_write_byte(0x50, addr, val);
|
||||||
break;
|
break;
|
||||||
|
#endif /* ALLOW_PEER_ERASE */
|
||||||
case 'r':
|
case 'r':
|
||||||
printf("Please enter EEPROM offset:\n\r");
|
printf("Please enter EEPROM offset:\n\r");
|
||||||
UART_GetIntegerMinMax(&addr, 0, 255);
|
UART_GetIntegerMinMax(&addr, 0, 255);
|
||||||
printf("EEPROM[0x%02x] = 0x%02x\n\r", addr, eeprom_read_byte(0x50, addr));
|
printf("EEPROM[0x%02lx] = 0x%02x\n\r", addr, eeprom_read_byte(0x50, addr));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
printf("Unknown command '%c'\n\r", ch);
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
@@ -140,12 +189,24 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
/* returns '1' in case we should break any endless loop */
|
/* returns '1' in case we should break any endless loop */
|
||||||
void board_exec_dbg_cmd(int ch)
|
void board_exec_dbg_cmd(int ch)
|
||||||
{
|
{
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
|
/* this variable controls if it is allowed to assert/release the ERASE line.
|
||||||
|
this is done to prevent accidental ERASE on noisy serial input since only one character can trigger the ERASE.
|
||||||
|
*/
|
||||||
|
static bool allow_erase = false;
|
||||||
|
#endif
|
||||||
|
|
||||||
switch (ch) {
|
switch (ch) {
|
||||||
case '?':
|
case '?':
|
||||||
printf("\t?\thelp\n\r");
|
printf("\t?\thelp\n\r");
|
||||||
printf("\tR\treset SAM3\n\r");
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
printf("\tl\tswitch off LED 1\n\r");
|
||||||
|
printf("\tL\tswitch off LED 1\n\r");
|
||||||
|
printf("\tg\tswitch off LED 2\n\r");
|
||||||
|
printf("\tG\tswitch off LED 2\n\r");
|
||||||
if (qmod_sam3_is_12()) {
|
if (qmod_sam3_is_12()) {
|
||||||
printf("\tE\tprogram EEPROM\n\r");
|
printf("\tE\tprogram EEPROM\n\r");
|
||||||
|
printf("\te\tErase EEPROM\n\r");
|
||||||
printf("\tO\tEnable PRTPWR_OVERRIDE\n\r");
|
printf("\tO\tEnable PRTPWR_OVERRIDE\n\r");
|
||||||
printf("\to\tDisable PRTPWR_OVERRIDE\n\r");
|
printf("\to\tDisable PRTPWR_OVERRIDE\n\r");
|
||||||
printf("\tH\tRelease HUB RESET (high)\n\r");
|
printf("\tH\tRelease HUB RESET (high)\n\r");
|
||||||
@@ -155,8 +216,11 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
}
|
}
|
||||||
printf("\tX\tRelease peer SAM3 from reset\n\r");
|
printf("\tX\tRelease peer SAM3 from reset\n\r");
|
||||||
printf("\tx\tAssert peer SAM3 reset\n\r");
|
printf("\tx\tAssert peer SAM3 reset\n\r");
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
printf("\tY\tRelease peer SAM3 ERASE signal\n\r");
|
printf("\tY\tRelease peer SAM3 ERASE signal\n\r");
|
||||||
|
printf("\ta\tAllow asserting peer SAM3 ERASE signal\n\r");
|
||||||
printf("\ty\tAssert peer SAM3 ERASE signal\n\r");
|
printf("\ty\tAssert peer SAM3 ERASE signal\n\r");
|
||||||
|
#endif
|
||||||
printf("\tU\tProceed to USB Initialization\n\r");
|
printf("\tU\tProceed to USB Initialization\n\r");
|
||||||
printf("\t1\tGenerate 1ms reset pulse on WWAN1\n\r");
|
printf("\t1\tGenerate 1ms reset pulse on WWAN1\n\r");
|
||||||
printf("\t2\tGenerate 1ms reset pulse on WWAN2\n\r");
|
printf("\t2\tGenerate 1ms reset pulse on WWAN2\n\r");
|
||||||
@@ -166,6 +230,22 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
USBD_Disconnect();
|
USBD_Disconnect();
|
||||||
NVIC_SystemReset();
|
NVIC_SystemReset();
|
||||||
break;
|
break;
|
||||||
|
case 'l':
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_OFF);
|
||||||
|
printf("LED 1 switched off\n\r");
|
||||||
|
break;
|
||||||
|
case 'L':
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
printf("LED 1 switched on\n\r");
|
||||||
|
break;
|
||||||
|
case 'g':
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_OFF);
|
||||||
|
printf("LED 2 switched off\n\r");
|
||||||
|
break;
|
||||||
|
case 'G':
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
printf("LED 2 switched on\n\r");
|
||||||
|
break;
|
||||||
case 'X':
|
case 'X':
|
||||||
printf("Clearing _SIMTRACExx_RST -> SIMTRACExx_RST high (inactive)\n\r");
|
printf("Clearing _SIMTRACExx_RST -> SIMTRACExx_RST high (inactive)\n\r");
|
||||||
PIO_Clear(&pin_peer_rst);
|
PIO_Clear(&pin_peer_rst);
|
||||||
@@ -174,14 +254,24 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
printf("Setting _SIMTRACExx_RST -> SIMTRACExx_RST low (active)\n\r");
|
printf("Setting _SIMTRACExx_RST -> SIMTRACExx_RST low (active)\n\r");
|
||||||
PIO_Set(&pin_peer_rst);
|
PIO_Set(&pin_peer_rst);
|
||||||
break;
|
break;
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
case 'Y':
|
case 'Y':
|
||||||
printf("Clearing SIMTRACExx_ERASE (inactive)\n\r");
|
printf("Clearing SIMTRACExx_ERASE (inactive)\n\r");
|
||||||
PIO_Clear(&pin_peer_erase);
|
PIO_Clear(&pin_peer_erase);
|
||||||
break;
|
break;
|
||||||
case 'y':
|
case 'a':
|
||||||
printf("Seetting SIMTRACExx_ERASE (active)\n\r");
|
printf("Asserting SIMTRACExx_ERASE allowed on next command\n\r");
|
||||||
PIO_Set(&pin_peer_erase);
|
allow_erase = true;
|
||||||
break;
|
break;
|
||||||
|
case 'y':
|
||||||
|
if (allow_erase) {
|
||||||
|
printf("Setting SIMTRACExx_ERASE (active)\n\r");
|
||||||
|
PIO_Set(&pin_peer_erase);
|
||||||
|
} else {
|
||||||
|
printf("Please first allow setting SIMTRACExx_ERASE\n\r");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
case '1':
|
case '1':
|
||||||
printf("Resetting Modem 1 (of this SAM3)\n\r");
|
printf("Resetting Modem 1 (of this SAM3)\n\r");
|
||||||
wwan_perst_do_reset_pulse(0, 300);
|
wwan_perst_do_reset_pulse(0, 300);
|
||||||
@@ -203,6 +293,13 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
board_exec_dbg_cmd_st12only(ch);
|
board_exec_dbg_cmd_st12only(ch);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
|
// set protection back so it can only run for one command
|
||||||
|
if ('a' != ch) {
|
||||||
|
allow_erase = false;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void board_main_top(void)
|
void board_main_top(void)
|
||||||
@@ -249,3 +346,51 @@ void board_main_top(void)
|
|||||||
card_present_init();
|
card_present_init();
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int uart_has_loopback_jumper(void)
|
||||||
|
{
|
||||||
|
unsigned int i;
|
||||||
|
const Pin uart_loopback_pins[] = {
|
||||||
|
{PIO_PA9A_URXD0, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT},
|
||||||
|
{PIO_PA10A_UTXD0, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Configure UART pins as I/O */
|
||||||
|
PIO_Configure(uart_loopback_pins, PIO_LISTSIZE(uart_loopback_pins));
|
||||||
|
|
||||||
|
/* Send pattern over UART TX and check if it is received on RX
|
||||||
|
* If the loop doesn't get interrupted, RxD always follows TxD and thus a
|
||||||
|
* loopback jumper has been placed on RxD/TxD, and we will boot
|
||||||
|
* into DFU unconditionally
|
||||||
|
*/
|
||||||
|
int has_loopback_jumper = 1;
|
||||||
|
for (i = 0; i < 10; i++) {
|
||||||
|
/* Set TxD high; abort if RxD doesn't go high either */
|
||||||
|
PIO_Set(&uart_loopback_pins[1]);
|
||||||
|
if (!PIO_Get(&uart_loopback_pins[0])) {
|
||||||
|
has_loopback_jumper = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* Set TxD low, abort if RxD doesn't go low either */
|
||||||
|
PIO_Clear(&uart_loopback_pins[1]);
|
||||||
|
if (PIO_Get(&uart_loopback_pins[0])) {
|
||||||
|
has_loopback_jumper = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Put pins back to UART mode */
|
||||||
|
const Pin uart_pins[] = {PINS_UART};
|
||||||
|
PIO_Configure(uart_pins, PIO_LISTSIZE(uart_pins));
|
||||||
|
|
||||||
|
return has_loopback_jumper;
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
/* If the loopback jumper is set, we enter DFU mode */
|
||||||
|
if (uart_has_loopback_jumper())
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,3 +1,21 @@
|
|||||||
|
/* card presence utilities
|
||||||
|
*
|
||||||
|
* (C) 2016-2017 by Harald Welte <hwelte@hmw-consulting.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
|
||||||
|
*/
|
||||||
#include <osmocom/core/timer.h>
|
#include <osmocom/core/timer.h>
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
@@ -17,9 +35,9 @@ int is_card_present(int port)
|
|||||||
const Pin *pin;
|
const Pin *pin;
|
||||||
int present;
|
int present;
|
||||||
|
|
||||||
if (port < 1 || port > NUM_CARDPRES)
|
if (port < 0 || port >= NUM_CARDPRES)
|
||||||
return -1;
|
return -1;
|
||||||
pin = &pin_cardpres[port-1];
|
pin = &pin_cardpres[port];
|
||||||
|
|
||||||
/* Card present signals are low-active, as we have a switch
|
/* Card present signals are low-active, as we have a switch
|
||||||
* against GND and an internal-pull-up in the SAM3 */
|
* against GND and an internal-pull-up in the SAM3 */
|
||||||
@@ -32,12 +50,12 @@ static void cardpres_tmr_cb(void *data)
|
|||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
for (i = 1; i <= ARRAY_SIZE(pin_cardpres); i++) {
|
for (i = 0; i < ARRAY_SIZE(pin_cardpres); i++) {
|
||||||
int state = is_card_present(i);
|
int state = is_card_present(i);
|
||||||
if (state != last_state[i-1]) {
|
if (state != last_state[i]) {
|
||||||
TRACE_INFO("Card Detect %d Status %d -> %d\r\n", i, last_state[i], state);
|
TRACE_INFO("%u: Card Detect Status %d -> %d\r\n", i, last_state[i], state);
|
||||||
/* FIXME: report to USB host */
|
/* FIXME: report to USB host */
|
||||||
last_state[i-1] = state;
|
last_state[i] = state;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* I2C EEPROM memory read and write utilities
|
||||||
|
*
|
||||||
|
* 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 "board.h"
|
#include "board.h"
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
@@ -169,6 +185,8 @@ int eeprom_write_byte(uint8_t slave, uint8_t addr, uint8_t byte)
|
|||||||
nack = i2c_write_byte(false, true, byte);
|
nack = i2c_write_byte(false, true, byte);
|
||||||
if (nack)
|
if (nack)
|
||||||
goto out_stop;
|
goto out_stop;
|
||||||
|
/* Wait tWR time to ensure EEPROM is writing correctly (tWR = 5 ms for AT24C02) */
|
||||||
|
mdelay(5);
|
||||||
|
|
||||||
out_stop:
|
out_stop:
|
||||||
i2c_stop_cond();
|
i2c_stop_cond();
|
||||||
|
|||||||
@@ -1,11 +1,24 @@
|
|||||||
/* Code to read/track the status of the WWAN LEDs of attached modems
|
/* Code to read/track the status of the WWAN LEDs of attached modems
|
||||||
*
|
*
|
||||||
* Depending on the board this is running on, it might be possible
|
* 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
|
||||||
|
*/
|
||||||
|
/* Depending on the board this is running on, it might be possible
|
||||||
* for the controller to read the status of the WWAN LED output lines of
|
* for the controller to read the status of the WWAN LED output lines of
|
||||||
* the cellular modem. If the board supports this, it sets the
|
* the cellular modem. If the board supports this, it sets the
|
||||||
* PIN_WWAN1 and/or PIN_WWAN2 defines in its board.h file.
|
* PIN_WWAN1 and/or PIN_WWAN2 defines in its board.h file.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "wwan_led.h"
|
#include "wwan_led.h"
|
||||||
|
|
||||||
@@ -14,9 +27,9 @@ static const Pin pin_wwan1 = PIN_WWAN1;
|
|||||||
|
|
||||||
static void wwan1_irqhandler(const Pin *pPin)
|
static void wwan1_irqhandler(const Pin *pPin)
|
||||||
{
|
{
|
||||||
int active = wwan_led_active(1);
|
int active = wwan_led_active(0);
|
||||||
|
|
||||||
TRACE_INFO("WWAN1 LED %u\r\n", active);
|
TRACE_INFO("0: WWAN LED %u\r\n", active);
|
||||||
|
|
||||||
/* TODO: notify host via USB */
|
/* TODO: notify host via USB */
|
||||||
}
|
}
|
||||||
@@ -27,8 +40,8 @@ static const Pin pin_wwan2 = PIN_WWAN2;
|
|||||||
|
|
||||||
static void wwan2_irqhandler(const Pin *pPin)
|
static void wwan2_irqhandler(const Pin *pPin)
|
||||||
{
|
{
|
||||||
int active = wwan_led_active(2);
|
int active = wwan_led_active(1);
|
||||||
TRACE_INFO("WWAN2 LED %u\r\n", active);
|
TRACE_INFO("1: WWAN LED %u\r\n", active);
|
||||||
|
|
||||||
/* TODO: notify host via USB */
|
/* TODO: notify host via USB */
|
||||||
}
|
}
|
||||||
@@ -42,12 +55,12 @@ int wwan_led_active(int wwan)
|
|||||||
|
|
||||||
switch (wwan) {
|
switch (wwan) {
|
||||||
#ifdef PIN_WWAN1
|
#ifdef PIN_WWAN1
|
||||||
case 1:
|
case 0:
|
||||||
pin = &pin_wwan1;
|
pin = &pin_wwan1;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
#ifdef PIN_WWAN2
|
#ifdef PIN_WWAN2
|
||||||
case 2:
|
case 1:
|
||||||
pin = &pin_wwan2;
|
pin = &pin_wwan2;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,29 +1,45 @@
|
|||||||
/* Code to control the PERST lines of attached modems
|
/* Code to control the PERST lines of attached modems
|
||||||
*
|
*
|
||||||
* Depending on the board this is running on, it might be possible
|
* 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
|
||||||
|
*/
|
||||||
|
/* Depending on the board this is running on, it might be possible
|
||||||
* for the controller to set the status of the PERST input line of
|
* for the controller to set the status of the PERST input line of
|
||||||
* the cellular modem. If the board supports this, it sets the
|
* the cellular modem. If the board supports this, it sets the
|
||||||
* PIN_PERST1 and/or PIN_PERST2 defines in its board.h file.
|
* PIN_PERST1 and/or PIN_PERST2 defines in its board.h file.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "trace.h"
|
#include "trace.h"
|
||||||
#include "wwan_perst.h"
|
#include "wwan_perst.h"
|
||||||
#include "osmocom/core/timer.h"
|
#include <osmocom/core/timer.h>
|
||||||
|
|
||||||
struct wwan_perst {
|
struct wwan_perst {
|
||||||
|
uint8_t idx;
|
||||||
const Pin pin;
|
const Pin pin;
|
||||||
struct osmo_timer_list timer;
|
struct osmo_timer_list timer;
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef PIN_PERST1
|
#ifdef PIN_PERST1
|
||||||
static struct wwan_perst perst1 = {
|
static struct wwan_perst perst1 = {
|
||||||
|
.idx = 0,
|
||||||
.pin = PIN_PERST1,
|
.pin = PIN_PERST1,
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef PIN_PERST2
|
#ifdef PIN_PERST2
|
||||||
static struct wwan_perst perst2 = {
|
static struct wwan_perst perst2 = {
|
||||||
|
.idx = 1,
|
||||||
.pin = PIN_PERST2,
|
.pin = PIN_PERST2,
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
@@ -34,7 +50,7 @@ static void perst_tmr_cb(void *data)
|
|||||||
{
|
{
|
||||||
struct wwan_perst *perst = data;
|
struct wwan_perst *perst = data;
|
||||||
/* release the (low-active) reset */
|
/* release the (low-active) reset */
|
||||||
TRACE_INFO("De-asserting modem reset\r\n");
|
TRACE_INFO("%u: De-asserting modem reset\r\n", perst->idx);
|
||||||
PIO_Clear(&perst->pin);
|
PIO_Clear(&perst->pin);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -106,5 +122,6 @@ int wwan_perst_init(void)
|
|||||||
perst2.timer.data = (void *) &perst2;
|
perst2.timer.data = (void *) &perst2;
|
||||||
num_perst++;
|
num_perst++;
|
||||||
#endif
|
#endif
|
||||||
|
initialized = 1;
|
||||||
return num_perst;
|
return num_perst;
|
||||||
}
|
}
|
||||||
|
|||||||
163
firmware/libboard/sam3p256/include/board.h
Normal file
163
firmware/libboard/sam3p256/include/board.h
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
/* Olimiex SAM3S-P256 board definition
|
||||||
|
*
|
||||||
|
* (C) 2019 by Harald Welte <hwelte@hmw-consulting.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
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
|
/* Name of the board */
|
||||||
|
#define BOARD_NAME "SAM3S-P256"
|
||||||
|
/* Board definition */
|
||||||
|
#define simtrace
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
|
#define BOARD_MAINOSC 12000000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58000000
|
||||||
|
|
||||||
|
/** MCU pin connected to yellow LED2 */
|
||||||
|
#define PIO_LED_RED PIO_PA17
|
||||||
|
/** MCU pin connected to green LED1 */
|
||||||
|
#define PIO_LED_GREEN PIO_PA18
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/** Pin configuration **/
|
||||||
|
/* Button to force bootloader start (shorted to ground when pressed */
|
||||||
|
#define PIN_BOOTLOADER_SW {PIO_PA20, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
#if 0
|
||||||
|
/* Enable powering the card using the second 3.3 V output of the LDO (active high) */
|
||||||
|
#define SIM_PWEN_PIN {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Enable powering the SIM card */
|
||||||
|
#define PWR_PINS SIM_PWEN_PIN
|
||||||
|
/* Card presence pin */
|
||||||
|
#define SW_SIM PIO_PA8
|
||||||
|
/* Pull card presence pin high (shorted to ground in card slot when card is present) */
|
||||||
|
#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_DEGLITCH | PIO_IT_EDGE }
|
||||||
|
|
||||||
|
/** Smart card connection **/
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_SIM_RST {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Card I/O data signal input/output (I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO {PIO_PA6A_TXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Card CLK clock input (CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK {PIO_PA2B_SCK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin to measure card I/O timing (to start measuring the ETU on I/O activity; connected I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO_INPUT {PIO_PA1B_TIOB0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK_INPUT {PIO_PA4B_TCLK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pins used to measure ETU timing (using timer counter) */
|
||||||
|
#define PINS_TC PIN_SIM_IO_INPUT, PIN_SIM_CLK_INPUT
|
||||||
|
|
||||||
|
/** Phone connection **/
|
||||||
|
/* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_VCC {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
/* Phone USIM slot 1 RST pin (active low; RST_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_nRST {PIO_PA24, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_RISE_EDGE | PIO_DEGLITCH }
|
||||||
|
/* Phone I/O data signal input/output (I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO {PIO_PA22A_TXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Phone CLK clock input (CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK {PIO_PA23A_SCK1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used for phone USIM slot 1 communication */
|
||||||
|
#define PINS_USIM1 PIN_PHONE_IO, PIN_PHONE_CLK, PIN_PHONE_CLK_INPUT, PIN_USIM1_VCC, PIN_PHONE_IO_INPUT, PIN_USIM1_nRST
|
||||||
|
/* Phone I/O data signal input/output (unused USART RX input; connected to I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO_INPUT {PIO_PA21A_RXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK_INPUT {PIO_PA29B_TCLK2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Default pin configuration **/
|
||||||
|
/* Disconnect VPP, CLK, and RST lines between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_SC_SW_DEFAULT {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect I/O line between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_IO_SW_DEFAULT {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_DEFAULT PIN_SC_SW_DEFAULT, PIN_IO_SW_DEFAULT
|
||||||
|
|
||||||
|
/** Sniffer configuration **/
|
||||||
|
/* Connect VPP, CLK, and RST lines between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_SC_SW_SNIFF {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect I/O line between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_IO_SW_SNIFF {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_SNIFF PIN_SC_SW_SNIFF, PIN_IO_SW_SNIFF
|
||||||
|
/* Card RST reset signal input (use as input since the phone will drive it) */
|
||||||
|
#define PIN_SIM_RST_SNIFF {PIO_PA7, PIOA, ID_PIOA, PIO_INPUT, PIO_DEGLITCH | PIO_IT_EDGE}
|
||||||
|
/* Pins used to sniff phone-card communication */
|
||||||
|
#define PINS_SIM_SNIFF PIN_SIM_IO, PIN_SIM_CLK, PIN_SIM_RST_SNIFF
|
||||||
|
/* Disable power converter 4.5-6V to 3.3V (active high) */
|
||||||
|
#define PIN_SIM_PWEN_SNIFF {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Enable power switch to forward VCC_PHONE to VCC_SIM (active high) */
|
||||||
|
#define PIN_VCC_FWD_SNIFF {VCC_FWD, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Use phone VCC to power card */
|
||||||
|
#define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF
|
||||||
|
|
||||||
|
/** CCID configuration */
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_ISO7816_RSTMC {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* ISO7816-communication related pins */
|
||||||
|
#define PINS_ISO7816 PIN_SIM_IO, PIN_SIM_CLK, PIN_ISO7816_RSTMC // SIM_PWEN_PIN, PIN_SIM_IO2, PIN_SIM_CLK2
|
||||||
|
|
||||||
|
/** External SPI flash interface **/
|
||||||
|
/* SPI MISO pin definition */
|
||||||
|
#define PIN_SPI_MISO {PIO_PA12A_MISO, PIOA, PIOA, PIO_PERIPH_A, PIO_PULLUP}
|
||||||
|
/* SPI MOSI pin definition */
|
||||||
|
#define PIN_SPI_MOSI {PIO_PA13A_MOSI, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI SCK pin definition */
|
||||||
|
#define PIN_SPI_SCK {PIO_PA14A_SPCK, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI pins definition. Contains MISO, MOSI & SCK */
|
||||||
|
#define PINS_SPI PIN_SPI_MISO, PIN_SPI_MOSI, PIN_SPI_SCK
|
||||||
|
/* SPI chip select 0 pin definition */
|
||||||
|
#define PIN_SPI_NPCS0 {PIO_PA11A_NPCS0, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI flash write protect pin (active low, pulled low) */
|
||||||
|
#define PIN_SPI_WP {PA15, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/** Pin configuration to control USB pull-up on D+
|
||||||
|
* @details the USB pull-up on D+ is enable by default on the board but can be disabled by setting PA16 high
|
||||||
|
*/
|
||||||
|
#define PIN_USB_PULLUP {PIO_PA16, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** USB definitions */
|
||||||
|
/* OpenMoko SIMtrace 2 USB vendor ID */
|
||||||
|
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO
|
||||||
|
/* OpenMoko SIMtrace 2 USB product ID (main application/runtime mode) */
|
||||||
|
#define BOARD_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2
|
||||||
|
/* OpenMoko SIMtrace 2 DFU USB product ID (DFU bootloader/DFU mode) */
|
||||||
|
#define BOARD_DFU_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2_DFU
|
||||||
|
/* USB release number (bcdDevice, shown as 0.00) */
|
||||||
|
#define BOARD_USB_RELEASE 0x000
|
||||||
|
/* Indicate SIMtrace is bus power in USB attributes */
|
||||||
|
#define BOARD_USB_BMATTRIBUTES USBConfigurationDescriptor_BUSPOWERED_NORWAKEUP
|
||||||
|
|
||||||
|
/** Supported modes */
|
||||||
|
/* SIMtrace board supports sniffer mode */
|
||||||
|
#define HAVE_SNIFFER
|
||||||
|
/* SIMtrace board supports CCID mode */
|
||||||
|
//#define HAVE_CCID
|
||||||
|
/* SIMtrace board supports card emulation mode */
|
||||||
|
//#define HAVE_CARDEM
|
||||||
|
/* SIMtrace board supports man-in-the-middle mode */
|
||||||
|
//#define HAVE_MITM
|
||||||
68
firmware/libboard/sam3p256/source/board_sam3p256.c
Normal file
68
firmware/libboard/sam3p256/source/board_sam3p256.c
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/* Olimex SAM3S-P256 specific application code
|
||||||
|
*
|
||||||
|
* (C) 2017,2019 by Harald Welte <laforge@gnumonks.org>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
|
#include "board.h"
|
||||||
|
#include "simtrace.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "sim_switch.h"
|
||||||
|
#include <osmocom/core/timer.h>
|
||||||
|
#include "usb_buf.h"
|
||||||
|
|
||||||
|
void board_exec_dbg_cmd(int ch)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case '?':
|
||||||
|
printf("\t?\thelp\n\r");
|
||||||
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
printf("Asking NVIC to reset us\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void board_main_top(void)
|
||||||
|
{
|
||||||
|
#ifndef APPLICATION_dfu
|
||||||
|
usb_buf_init();
|
||||||
|
|
||||||
|
/* Initialize checking for card insert/remove events */
|
||||||
|
//card_present_init();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
const Pin bl_sw_pin = PIN_BOOTLOADER_SW;
|
||||||
|
|
||||||
|
PIO_Configure(&bl_sw_pin, 1);
|
||||||
|
|
||||||
|
/* Enter DFU bootloader in case the respective button is pressed */
|
||||||
|
if (PIO_Get(&bl_sw_pin) == 0) {
|
||||||
|
/* do not print to early since the console is not initialized yet */
|
||||||
|
//printf("BOOTLOADER switch pressed -> Force DFU\n\r");
|
||||||
|
return 1;
|
||||||
|
} else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -1,87 +1,162 @@
|
|||||||
|
/* SIMtrace with SAM3S board definition
|
||||||
|
*
|
||||||
|
* (C) 2016-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
#include "board_common.h"
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
/** Name of the board */
|
/* Name of the board */
|
||||||
#define BOARD_NAME "SAM3S-SIMTRACE"
|
#define BOARD_NAME "SAM3S-SIMTRACE"
|
||||||
/** Board definition */
|
/* Board definition */
|
||||||
#define simtrace
|
#define simtrace
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
#define BOARD_MAINOSC 18432000
|
#define BOARD_MAINOSC 18432000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58982400 // 18.432 * 16 / 5
|
||||||
|
|
||||||
/** Phone (SIM card emulator)/CCID Reader/MITM configuration **/
|
/** MCU pin connected to red LED */
|
||||||
/* Normally the communication lines between phone and SIM card are disconnected */
|
#define PIO_LED_RED PIO_PA17
|
||||||
// Disconnect SIM card I/O, VPP line from the phone lines
|
/** MCU pin connected to green LED */
|
||||||
// FIXME: Per default pins are input, therefore high-impedance, therefore they don not activate the bus switch, right?
|
#define PIO_LED_GREEN PIO_PA18
|
||||||
#define PIN_SC_SW_DEFAULT {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
/** red LED pin definition */
|
||||||
// Disconnect SIM card RST, CLK line from the phone lines
|
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
#define PIN_IO_SW_DEFAULT {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
/** green LED pin definition */
|
||||||
#define PINS_BUS_DEFAULT PIN_SC_SW_DEFAULT, PIN_IO_SW_DEFAULT
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/** Pin configuration **/
|
||||||
|
/* Button to force bootloader start (shorted to ground when pressed */
|
||||||
|
#define PIN_BOOTLOADER_SW {PIO_PA31, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP}
|
||||||
|
/* Enable powering the card using the second 3.3 V output of the LDO (active high) */
|
||||||
|
#define SIM_PWEN_PIN {PIO_PA5, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Enable powering the SIM card */
|
||||||
|
#define PWR_PINS SIM_PWEN_PIN
|
||||||
|
/* Card presence pin */
|
||||||
|
#define SW_SIM PIO_PA8
|
||||||
|
/* Pull card presence pin high (shorted to ground in card slot when card is present) */
|
||||||
|
#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_DEGLITCH | PIO_IT_EDGE }
|
||||||
|
|
||||||
|
/** Smart card connection **/
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_SIM_RST {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Card I/O data signal input/output (I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO {PIO_PA6A_TXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Card CLK clock input (CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK {PIO_PA2B_SCK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin to measure card I/O timing (to start measuring the ETU on I/O activity; connected I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO_INPUT {PIO_PA1B_TIOB0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK_INPUT {PIO_PA4B_TCLK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pins used to measure ETU timing (using timer counter) */
|
||||||
|
#define PINS_TC PIN_SIM_IO_INPUT, PIN_SIM_CLK_INPUT
|
||||||
|
|
||||||
|
/** Phone connection **/
|
||||||
|
/* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_VCC {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
/* Phone USIM slot 1 RST pin (active low; RST_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_nRST {PIO_PA24, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_RISE_EDGE | PIO_DEGLITCH }
|
||||||
|
/* Phone I/O data signal input/output (I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO {PIO_PA22A_TXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Phone CLK clock input (CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK {PIO_PA23A_SCK1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used for phone USIM slot 1 communication */
|
||||||
|
#define PINS_USIM1 PIN_PHONE_IO, PIN_PHONE_CLK, PIN_PHONE_CLK_INPUT, PIN_USIM1_VCC, PIN_PHONE_IO_INPUT, PIN_USIM1_nRST
|
||||||
|
/* Phone I/O data signal input/output (unused USART RX input; connected to I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO_INPUT {PIO_PA21A_RXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK_INPUT {PIO_PA29B_TCLK2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Default pin configuration **/
|
||||||
|
/* Disconnect VPP, CLK, and RST lines between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_SC_SW_DEFAULT {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect I/O line between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_IO_SW_DEFAULT {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_DEFAULT PIN_SC_SW_DEFAULT, PIN_IO_SW_DEFAULT
|
||||||
|
|
||||||
/** Sniffer configuration **/
|
/** Sniffer configuration **/
|
||||||
// Connect VPP, CLK and RST lines from smartcard to the phone
|
/* Connect VPP, CLK, and RST lines between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
#define PIN_SC_SW_SNIFF {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
#define PIN_SC_SW_SNIFF {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect I/O line between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
#define PIN_IO_SW_SNIFF {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
#define PIN_IO_SW_SNIFF {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
#define PINS_BUS_SNIFF PIN_SC_SW_SNIFF, PIN_IO_SW_SNIFF
|
#define PINS_BUS_SNIFF PIN_SC_SW_SNIFF, PIN_IO_SW_SNIFF
|
||||||
|
/* Card RST reset signal input (use as input since the phone will drive it) */
|
||||||
|
#define PIN_SIM_RST_SNIFF {PIO_PA7, PIOA, ID_PIOA, PIO_INPUT, PIO_DEGLITCH | PIO_IT_EDGE}
|
||||||
|
/* Pins used to sniff phone-card communication */
|
||||||
|
#define PINS_SIM_SNIFF PIN_SIM_IO, PIN_SIM_CLK, PIN_SIM_RST_SNIFF
|
||||||
|
/* Disable power converter 4.5-6V to 3.3V (active high) */
|
||||||
|
#define PIN_SIM_PWEN_SNIFF {PIO_PA5, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Enable power switch to forward VCC_PHONE to VCC_SIM (active high) */
|
||||||
|
#define PIN_VCC_FWD_SNIFF {PIO_PA26, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Use phone VCC to power card */
|
||||||
|
#define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF
|
||||||
|
|
||||||
#define PINS_SIM_SNIFF_SIM PIN_PHONE_IO, PIN_PHONE_CLK
|
/** CCID configuration */
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_ISO7816_RSTMC {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* ISO7816-communication related pins */
|
||||||
|
#define PINS_ISO7816 PIN_SIM_IO, PIN_SIM_CLK, PIN_ISO7816_RSTMC // SIM_PWEN_PIN, PIN_SIM_IO2, PIN_SIM_CLK2
|
||||||
|
|
||||||
#define SIM_PWEN_PIN {PIO_PA5, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
/** External SPI flash interface **/
|
||||||
|
/* SPI MISO pin definition */
|
||||||
|
#define PIN_SPI_MISO {PIO_PA12A_MISO, PIOA, PIOA, PIO_PERIPH_A, PIO_PULLUP}
|
||||||
|
/* SPI MOSI pin definition */
|
||||||
|
#define PIN_SPI_MOSI {PIO_PA13A_MOSI, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI SCK pin definition */
|
||||||
|
#define PIN_SPI_SCK {PIO_PA14A_SPCK, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI pins definition. Contains MISO, MOSI & SCK */
|
||||||
|
#define PINS_SPI PIN_SPI_MISO, PIN_SPI_MOSI, PIN_SPI_SCK
|
||||||
|
/* SPI chip select 0 pin definition */
|
||||||
|
#define PIN_SPI_NPCS0 {PIO_PA11A_NPCS0, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI flash write protect pin (active low, pulled low) */
|
||||||
|
#define PIN_SPI_WP {PA15, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
#define PWR_PINS \
|
/** Pin configuration to control USB pull-up on D+
|
||||||
/* Enable power converter 4.5-6V to 3.3V; low: off */ \
|
* @details the USB pull-up on D+ is enable by default on the board but can be disabled by setting PA16 high
|
||||||
{SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}, \
|
*/
|
||||||
/* Enable second power converter: VCC_PHONE to VCC_SIM; high: on */ \
|
#define PIN_USB_PULLUP {PIO_PA16, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
{VCC_FWD, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
|
||||||
|
|
||||||
#define SW_SIM PIO_PA8
|
|
||||||
#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_DEGLITCH | PIO_IT_EDGE }
|
|
||||||
//#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOB, ID_PIOB, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_IT_EDGE}
|
|
||||||
|
|
||||||
/// PIN used for resetting the smartcard
|
|
||||||
// FIXME: Card is resetted with pin set to 0 --> PIO_OUTPUT_1 as default is right?
|
|
||||||
#define PIN_ISO7816_RSTMC {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
|
||||||
|
|
||||||
/// Pins used for connect the smartcard
|
|
||||||
#define PIN_SIM_IO_INPUT {PIO_PA1, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
|
||||||
#define PIN_SIM_IO {PIO_PA6, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
|
||||||
#define PIN_SIM_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
|
||||||
#define PIN_SIM_CLK_INPUT {PIO_PA4, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
|
||||||
//#define PINS_ISO7816 PIN_USART1_TXD, PIN_USART1_SCK, PIN_ISO7816_RSTMC
|
|
||||||
#define PINS_ISO7816 PIN_SIM_IO, PIN_SIM_CLK, PIN_ISO7816_RSTMC // SIM_PWEN_PIN, PIN_SIM_IO2, PIN_SIM_CLK2
|
|
||||||
|
|
||||||
#define PINS_TC PIN_SIM_IO_INPUT, PIN_SIM_CLK_INPUT
|
|
||||||
|
|
||||||
#define VCC_PHONE {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
|
||||||
#define PIN_ISO7816_RST_PHONE {PIO_PA24, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_RISE_EDGE | PIO_DEGLITCH }
|
|
||||||
#define PIN_PHONE_IO_INPUT {PIO_PA21, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
|
||||||
#define PIN_PHONE_IO {PIO_PA22, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
|
||||||
#define PIN_PHONE_CLK {PIO_PA23A_SCK1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT} // External Clock Input on PA28
|
|
||||||
//#define PIN_PHONE_CLK {PIO_PA23A_SCK1, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT} // External Clock Input on PA28
|
|
||||||
#define PIN_PHONE_CLK_INPUT {PIO_PA29, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
|
||||||
#define PINS_ISO7816_PHONE PIN_PHONE_IO, PIN_PHONE_CLK, PIN_PHONE_CLK_INPUT, VCC_PHONE, PIN_PHONE_IO_INPUT, PIN_ISO7816_RST_PHONE
|
|
||||||
//, VCC_PHONE
|
|
||||||
|
|
||||||
|
|
||||||
//** SPI interface **/
|
|
||||||
/// SPI MISO pin definition (PA12).
|
|
||||||
#define PIN_SPI_MISO {1 << 12, PIOA, PIOA, PIO_PERIPH_A, PIO_PULLUP}
|
|
||||||
/// SPI MOSI pin definition (PA13).
|
|
||||||
#define PIN_SPI_MOSI {1 << 13, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
|
||||||
/// SPI SPCK pin definition (PA14).
|
|
||||||
#define PIN_SPI_SPCK {1 << 14, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
|
||||||
/// SPI pins definition. Contains MISO, MOSI & SPCK (PA12, PA13 & PA14).
|
|
||||||
#define PINS_SPI PIN_SPI_MISO, PIN_SPI_MOSI, PIN_SPI_SPCK
|
|
||||||
/// SPI chip select 0 pin definition (PA11).
|
|
||||||
#define PIN_SPI_NPCS0 {1 << 11, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
|
||||||
|
|
||||||
|
/** USB definitions */
|
||||||
|
/* OpenMoko SIMtrace 2 USB vendor ID */
|
||||||
|
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO
|
||||||
|
/* OpenMoko SIMtrace 2 USB product ID (main application/runtime mode) */
|
||||||
|
#define BOARD_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2
|
||||||
|
/* OpenMoko SIMtrace 2 DFU USB product ID (DFU bootloader/DFU mode) */
|
||||||
|
#define BOARD_DFU_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2_DFU
|
||||||
|
/* USB release number (bcdDevice, shown as 0.00) */
|
||||||
|
#define BOARD_USB_RELEASE 0x000
|
||||||
|
/* Indicate SIMtrace is bus power in USB attributes */
|
||||||
#define BOARD_USB_BMATTRIBUTES USBConfigurationDescriptor_BUSPOWERED_NORWAKEUP
|
#define BOARD_USB_BMATTRIBUTES USBConfigurationDescriptor_BUSPOWERED_NORWAKEUP
|
||||||
|
|
||||||
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO
|
/** Supported modes */
|
||||||
#define BOARD_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2
|
/* SIMtrace board supports sniffer mode */
|
||||||
#define BOARD_DFU_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2_DFU
|
|
||||||
#define BOARD_USB_RELEASE 0x000
|
|
||||||
|
|
||||||
#define HAVE_SNIFFER
|
#define HAVE_SNIFFER
|
||||||
#define HAVE_CCID
|
/* SIMtrace board supports CCID mode */
|
||||||
#define HAVE_CARDEM
|
//#define HAVE_CCID
|
||||||
#define HAVE_MITM
|
/* SIMtrace board supports card emulation mode */
|
||||||
|
//#define HAVE_CARDEM
|
||||||
|
/* SIMtrace board supports man-in-the-middle mode */
|
||||||
|
//#define HAVE_MITM
|
||||||
|
|||||||
68
firmware/libboard/simtrace/source/board_simtrace.c
Normal file
68
firmware/libboard/simtrace/source/board_simtrace.c
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/* SIMtrace with SAM3S specific application code
|
||||||
|
*
|
||||||
|
* (C) 2017 by Harald Welte <laforge@gnumonks.org>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
|
#include "board.h"
|
||||||
|
#include "simtrace.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "sim_switch.h"
|
||||||
|
#include <osmocom/core/timer.h>
|
||||||
|
#include "usb_buf.h"
|
||||||
|
|
||||||
|
void board_exec_dbg_cmd(int ch)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case '?':
|
||||||
|
printf("\t?\thelp\n\r");
|
||||||
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
printf("Asking NVIC to reset us\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void board_main_top(void)
|
||||||
|
{
|
||||||
|
#ifndef APPLICATION_dfu
|
||||||
|
usb_buf_init();
|
||||||
|
|
||||||
|
/* Initialize checking for card insert/remove events */
|
||||||
|
//card_present_init();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
const Pin bl_sw_pin = PIN_BOOTLOADER_SW;
|
||||||
|
|
||||||
|
PIO_Configure(&bl_sw_pin, 1);
|
||||||
|
|
||||||
|
/* Enter DFU bootloader in case the respective button is pressed */
|
||||||
|
if (PIO_Get(&bl_sw_pin) == 0) {
|
||||||
|
/* do not print to early since the console is not initialized yet */
|
||||||
|
//printf("BOOTLOADER switch pressed -> Force DFU\n\r");
|
||||||
|
return 1;
|
||||||
|
} else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -65,41 +65,41 @@
|
|||||||
// Definitions
|
// Definitions
|
||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
#if defined(NOASSERT)
|
#if defined(NOASSERT)
|
||||||
#define ASSERT(...)
|
#define ASSERT(...)
|
||||||
#define SANITY_CHECK(...)
|
#define SANITY_CHECK(...)
|
||||||
#else
|
#else
|
||||||
|
|
||||||
#if (TRACE_LEVEL == 0)
|
#if (TRACE_LEVEL == 0)
|
||||||
/// Checks that the given condition is true,
|
/// Checks that the given condition is true,
|
||||||
/// otherwise stops the program execution.
|
/// otherwise stops the program execution.
|
||||||
/// \param condition Condition to verify.
|
/// \param condition Condition to verify.
|
||||||
#define ASSERT(condition) { \
|
#define ASSERT(condition) { \
|
||||||
if (!(condition)) { \
|
if (!(condition)) { \
|
||||||
while (1); \
|
while (1); \
|
||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Performs the same duty as the ASSERT() macro
|
/// Performs the same duty as the ASSERT() macro
|
||||||
/// \param condition Condition to verify.
|
/// \param condition Condition to verify.
|
||||||
#define SANITY_CHECK(condition) ASSERT(condition, ...)
|
#define SANITY_CHECK(condition) ASSERT(condition, ...)
|
||||||
|
|
||||||
#else
|
#else
|
||||||
/// Checks that the given condition is true, otherwise displays an error
|
/// Checks that the given condition is true, otherwise displays an error
|
||||||
/// message and stops the program execution.
|
/// message and stops the program execution.
|
||||||
/// \param condition Condition to verify.
|
/// \param condition Condition to verify.
|
||||||
#define ASSERT(condition) { \
|
#define ASSERT(condition) { \
|
||||||
if (!(condition)) { \
|
if (!(condition)) { \
|
||||||
printf("-F- ASSERT: %s %s:%d\n\r", #condition, __BASE_FILE__, __LINE__); \
|
printf("-F- ASSERT: %s %s:%d\n\r", #condition, __BASE_FILE__, __LINE__); \
|
||||||
while (1); \
|
while (1); \
|
||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
#define SANITY_ERROR "Sanity check failed at %s:%d\n\r"
|
#define SANITY_ERROR "Sanity check failed at %s:%d\n\r"
|
||||||
|
|
||||||
/// Performs the same duty as the ASSERT() macro, except a default error
|
/// Performs the same duty as the ASSERT() macro, except a default error
|
||||||
/// message is output if the condition is false.
|
/// message is output if the condition is false.
|
||||||
/// \param condition Condition to verify.
|
/// \param condition Condition to verify.
|
||||||
#define SANITY_CHECK(condition) ASSERT(condition, SANITY_ERROR, __FILE__, __LINE__)
|
#define SANITY_CHECK(condition) ASSERT(condition, SANITY_ERROR, __FILE__, __LINE__)
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,22 @@
|
|||||||
|
/* ISO7816-3 state machine for the card side
|
||||||
|
*
|
||||||
|
* (C) 2010-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
@@ -10,8 +29,18 @@ enum card_io {
|
|||||||
CARD_IO_CLK,
|
CARD_IO_CLK,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan,
|
/** initialise card slot
|
||||||
uint8_t in_ep, uint8_t irq_ep);
|
* @param[in] slot_num slot number (arbitrary number)
|
||||||
|
* @param[in] tc_chan timer counter channel (to measure the ETU)
|
||||||
|
* @param[in] uart_chan UART peripheral channel
|
||||||
|
* @param[in] in_ep USB IN end point number
|
||||||
|
* @param[in] irq_ep USB INTerrupt end point number
|
||||||
|
* @param[in] vcc_active initial VCC signal state (true = on)
|
||||||
|
* @param[in] in_reset initial RST signal state (true = reset asserted)
|
||||||
|
* @param[in] clocked initial CLK signat state (true = active)
|
||||||
|
* @return main card handle reference
|
||||||
|
*/
|
||||||
|
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan, uint8_t in_ep, uint8_t irq_ep, bool vcc_active, bool in_reset, bool clocked);
|
||||||
|
|
||||||
/* process a single byte received from the reader */
|
/* process a single byte received from the reader */
|
||||||
void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte);
|
void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte);
|
||||||
@@ -36,3 +65,4 @@ int card_emu_uart_update_fidi(uint8_t uart_chan, unsigned int fidi);
|
|||||||
int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte);
|
int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte);
|
||||||
void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx);
|
void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx);
|
||||||
void card_emu_uart_wait_tx_idle(uint8_t uart_chan);
|
void card_emu_uart_wait_tx_idle(uint8_t uart_chan);
|
||||||
|
void card_emu_uart_interrupt(uint8_t uart_chan);
|
||||||
|
|||||||
@@ -145,129 +145,129 @@ typedef struct
|
|||||||
/// 6.1.11.2 PIN Verification Data Structure
|
/// 6.1.11.2 PIN Verification Data Structure
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
/// Number of seconds.
|
/// Number of seconds.
|
||||||
unsigned char bTimerOut;
|
unsigned char bTimerOut;
|
||||||
/// Several parameters for the PIN format options
|
/// Several parameters for the PIN format options
|
||||||
unsigned char bmFormatString;
|
unsigned char bmFormatString;
|
||||||
/// Define the length of the PIN to present in the APDU command
|
/// Define the length of the PIN to present in the APDU command
|
||||||
unsigned char bmPINBlockString;
|
unsigned char bmPINBlockString;
|
||||||
/// Allows the length PIN insertion in the APDU command
|
/// Allows the length PIN insertion in the APDU command
|
||||||
unsigned char bmPINLengthFormat;
|
unsigned char bmPINLengthFormat;
|
||||||
/// Minimum PIN size in digit and Maximum PIN size in digit
|
/// Minimum PIN size in digit and Maximum PIN size in digit
|
||||||
unsigned char wPINMaxExtraDigit;
|
unsigned char wPINMaxExtraDigit;
|
||||||
/// The value is a bit wise OR operation.
|
/// The value is a bit wise OR operation.
|
||||||
unsigned char bEntryValidationCondition;
|
unsigned char bEntryValidationCondition;
|
||||||
/// Number of messages to display for the PIN modify command
|
/// Number of messages to display for the PIN modify command
|
||||||
unsigned char bNumberMessage;
|
unsigned char bNumberMessage;
|
||||||
/// Language used to display the messages.
|
/// Language used to display the messages.
|
||||||
unsigned char wLangId;
|
unsigned char wLangId;
|
||||||
/// Message index in the Reader message table
|
/// Message index in the Reader message table
|
||||||
unsigned char bMsgIndex;
|
unsigned char bMsgIndex;
|
||||||
/// T=1 I-block prologue field to use
|
/// T=1 I-block prologue field to use
|
||||||
unsigned char bTeoPrologue[3];
|
unsigned char bTeoPrologue[3];
|
||||||
/// APDU to send to the ICC
|
/// APDU to send to the ICC
|
||||||
unsigned char abPINApdu[255];
|
unsigned char abPINApdu[255];
|
||||||
}__attribute__ ((packed)) S_ccid_PIN_Verification;
|
}__attribute__ ((packed)) S_ccid_PIN_Verification;
|
||||||
|
|
||||||
|
|
||||||
/// 6.1.11.7 PIN Modification Data Structure
|
/// 6.1.11.7 PIN Modification Data Structure
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
/// Number of seconds. If 00h then CCID default value is used.
|
/// Number of seconds. If 00h then CCID default value is used.
|
||||||
unsigned char bTimeOut;
|
unsigned char bTimeOut;
|
||||||
/// Several parameters for the PIN format options (defined in § 6.1.11.4)
|
/// Several parameters for the PIN format options (defined in § 6.1.11.4)
|
||||||
unsigned char bmFormatString4;
|
unsigned char bmFormatString4;
|
||||||
/// Define the length of the PIN to present in the APDU command
|
/// Define the length of the PIN to present in the APDU command
|
||||||
unsigned char bmPINBlockString;
|
unsigned char bmPINBlockString;
|
||||||
/// Allows the length PIN insertion in the APDU command (defined in § 6.1.11.6)
|
/// Allows the length PIN insertion in the APDU command (defined in § 6.1.11.6)
|
||||||
unsigned char bmPinLengthFormat;
|
unsigned char bmPinLengthFormat;
|
||||||
/// Insertion position offset in byte for the current PIN
|
/// Insertion position offset in byte for the current PIN
|
||||||
unsigned char bInsertionOffsetOld;
|
unsigned char bInsertionOffsetOld;
|
||||||
/// Insertion position offset in byte for the new PIN
|
/// Insertion position offset in byte for the new PIN
|
||||||
unsigned char bInsertionOffsetNew;
|
unsigned char bInsertionOffsetNew;
|
||||||
/// XXYYh
|
/// XXYYh
|
||||||
/// XX: Minimum PIN size in digit
|
/// XX: Minimum PIN size in digit
|
||||||
/// YY: Maximum PIN size in digit
|
/// YY: Maximum PIN size in digit
|
||||||
unsigned char wPINMaxExtraDigit;
|
unsigned char wPINMaxExtraDigit;
|
||||||
/// 00h,01h,02h,03h
|
/// 00h,01h,02h,03h
|
||||||
/// Indicates if a confirmation is requested before acceptance of a new PIN (meaning that the user has to enter this new PIN twice before it is accepted)
|
/// Indicates if a confirmation is requested before acceptance of a new PIN (meaning that the user has to enter this new PIN twice before it is accepted)
|
||||||
/// Indicates if the current PIN must be entered and set in the same APDU field of not.
|
/// Indicates if the current PIN must be entered and set in the same APDU field of not.
|
||||||
unsigned char bConfirmPIN;
|
unsigned char bConfirmPIN;
|
||||||
/// The value is a bit wise OR operation.
|
/// The value is a bit wise OR operation.
|
||||||
/// 01h Max size reached
|
/// 01h Max size reached
|
||||||
/// 02h Validation key pressed
|
/// 02h Validation key pressed
|
||||||
/// 04h Timeout occurred
|
/// 04h Timeout occurred
|
||||||
unsigned char bEntryValidationCondition;
|
unsigned char bEntryValidationCondition;
|
||||||
/// 00h,01h,02h,03h,or FFh
|
/// 00h,01h,02h,03h,or FFh
|
||||||
/// Number of messages to display for the PIN modify command.
|
/// Number of messages to display for the PIN modify command.
|
||||||
unsigned char bNumberMessage;
|
unsigned char bNumberMessage;
|
||||||
/// Language used to display the messages. The 16 bit
|
/// Language used to display the messages. The 16 bit
|
||||||
unsigned char wLangId;
|
unsigned char wLangId;
|
||||||
/// Message index in the Reader message table (should be 00h or 01h).
|
/// Message index in the Reader message table (should be 00h or 01h).
|
||||||
unsigned char bMsgIndex1;
|
unsigned char bMsgIndex1;
|
||||||
/// Message index in the Reader message table (should be 01h or 02h).
|
/// Message index in the Reader message table (should be 01h or 02h).
|
||||||
unsigned char bMsgIndex2;
|
unsigned char bMsgIndex2;
|
||||||
/// Message index in the Reader message table (should be 02h).
|
/// Message index in the Reader message table (should be 02h).
|
||||||
unsigned char bMsgIndex3;
|
unsigned char bMsgIndex3;
|
||||||
/// T=1 I-block prologue field to use. Significant only if protocol in use is T=1.
|
/// T=1 I-block prologue field to use. Significant only if protocol in use is T=1.
|
||||||
unsigned char bTeoPrologue[3];
|
unsigned char bTeoPrologue[3];
|
||||||
/// Byte array APDU to send to the ICC
|
/// Byte array APDU to send to the ICC
|
||||||
unsigned char abPINApdu[255];
|
unsigned char abPINApdu[255];
|
||||||
}__attribute__ ((packed)) S_ccid_PIN_Modification;
|
}__attribute__ ((packed)) S_ccid_PIN_Modification;
|
||||||
|
|
||||||
/// Protocol Data Structure for Protocol T=0 (bProtocolNum=0, dwLength=00000005h)
|
/// Protocol Data Structure for Protocol T=0 (bProtocolNum=0, dwLength=00000005h)
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
/// B7-4 – FI – Index into the table 7 in ISO/IEC 7816-3:1997 selecting a
|
/// B7-4 – FI – Index into the table 7 in ISO/IEC 7816-3:1997 selecting a
|
||||||
/// clock rate conversion factor
|
/// clock rate conversion factor
|
||||||
/// B3-0 – DI - Index into the table 8 in ISO/IEC 7816-3:1997 selecting a
|
/// B3-0 – DI - Index into the table 8 in ISO/IEC 7816-3:1997 selecting a
|
||||||
/// baud rate conversion factor
|
/// baud rate conversion factor
|
||||||
unsigned char bmFindexDindex;
|
unsigned char bmFindexDindex;
|
||||||
/// For T=0 ,B0 – 0b, B7-2 – 000000b
|
/// For T=0 ,B0 – 0b, B7-2 – 000000b
|
||||||
/// B1 – Convention used (b1=0 for direct, b1=1 for inverse)
|
/// B1 – Convention used (b1=0 for direct, b1=1 for inverse)
|
||||||
unsigned char bmTCCKST0; // 0 to 2
|
unsigned char bmTCCKST0; // 0 to 2
|
||||||
/// Extra Guardtime between two characters. Add 0 to 254 etu to the normal
|
/// Extra Guardtime between two characters. Add 0 to 254 etu to the normal
|
||||||
/// guardtime of 12etu. FFh is the same as 00h.
|
/// guardtime of 12etu. FFh is the same as 00h.
|
||||||
unsigned char bGuardTimeT0; // 0 to FF
|
unsigned char bGuardTimeT0; // 0 to FF
|
||||||
/// WI for T=0 used to define WWT
|
/// WI for T=0 used to define WWT
|
||||||
unsigned char bWaitingIntegerT0; // 0 to FF
|
unsigned char bWaitingIntegerT0; // 0 to FF
|
||||||
/// ICC Clock Stop Support
|
/// ICC Clock Stop Support
|
||||||
/// 00 = Stopping the Clock is not allowed
|
/// 00 = Stopping the Clock is not allowed
|
||||||
/// 01 = Stop with Clock signal Low
|
/// 01 = Stop with Clock signal Low
|
||||||
/// 02 = Stop with Clock signal High
|
/// 02 = Stop with Clock signal High
|
||||||
/// 03 = Stop with Clock either High or Low
|
/// 03 = Stop with Clock either High or Low
|
||||||
unsigned char bClockStop; // 0 to 3
|
unsigned char bClockStop; // 0 to 3
|
||||||
} __attribute__ ((packed)) S_ccid_protocol_t0;
|
} __attribute__ ((packed)) S_ccid_protocol_t0;
|
||||||
|
|
||||||
|
|
||||||
/// Protocol Data Structure for Protocol T=1 (bProtocolNum=1, dwLength=00000007h)
|
/// Protocol Data Structure for Protocol T=1 (bProtocolNum=1, dwLength=00000007h)
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
/// B7-4 – FI – Index into the table 7 in ISO/IEC 7816-3:1997 selecting a
|
/// B7-4 – FI – Index into the table 7 in ISO/IEC 7816-3:1997 selecting a
|
||||||
/// clock rate conversion factor
|
/// clock rate conversion factor
|
||||||
/// B3-0 – DI - Index into the table 8 in ISO/IEC 7816-3:1997 selecting a
|
/// B3-0 – DI - Index into the table 8 in ISO/IEC 7816-3:1997 selecting a
|
||||||
/// baud rate conversion factor
|
/// baud rate conversion factor
|
||||||
unsigned char bmFindexDindex;
|
unsigned char bmFindexDindex;
|
||||||
/// For T=1, B7-2 – 000100b
|
/// For T=1, B7-2 – 000100b
|
||||||
/// B0 – Checksum type (b0=0 for LRC, b0=1 for CRC
|
/// B0 – Checksum type (b0=0 for LRC, b0=1 for CRC
|
||||||
/// B1 – Convention used (b1=0 for direct, b1=1 for inverse)
|
/// B1 – Convention used (b1=0 for direct, b1=1 for inverse)
|
||||||
unsigned char bmTCCKST1; // 10h, 11h, 12h, 13h
|
unsigned char bmTCCKST1; // 10h, 11h, 12h, 13h
|
||||||
/// Extra Guardtime (0 to 254 etu between two characters).
|
/// Extra Guardtime (0 to 254 etu between two characters).
|
||||||
/// If value is FFh, then guardtime is reduced by 1.
|
/// If value is FFh, then guardtime is reduced by 1.
|
||||||
unsigned char bGuardTimeT1; // 0 to FF
|
unsigned char bGuardTimeT1; // 0 to FF
|
||||||
/// B7-4 = BWI
|
/// B7-4 = BWI
|
||||||
/// B3-0 = CWI
|
/// B3-0 = CWI
|
||||||
unsigned char bmWaitingIntegersT1; // 0 to 9
|
unsigned char bmWaitingIntegersT1; // 0 to 9
|
||||||
/// ICC Clock Stop Support
|
/// ICC Clock Stop Support
|
||||||
/// 00 = Stopping the Clock is not allowed
|
/// 00 = Stopping the Clock is not allowed
|
||||||
/// 01 = Stop with Clock signal Low
|
/// 01 = Stop with Clock signal Low
|
||||||
/// 02 = Stop with Clock signal High
|
/// 02 = Stop with Clock signal High
|
||||||
/// 03 = Stop with Clock either High or Low
|
/// 03 = Stop with Clock either High or Low
|
||||||
unsigned char bClockStop; // 0 to 3
|
unsigned char bClockStop; // 0 to 3
|
||||||
/// Size of negotiated IFSC
|
/// Size of negotiated IFSC
|
||||||
unsigned char bIFSC; // 0 to FE
|
unsigned char bIFSC; // 0 to FE
|
||||||
/// Nad value used by CCID
|
/// Nad value used by CCID
|
||||||
unsigned char bNadValue; // 0 to FF
|
unsigned char bNadValue; // 0 to FF
|
||||||
} __attribute__ ((packed)) S_ccid_protocol_t1;
|
} __attribute__ ((packed)) S_ccid_protocol_t1;
|
||||||
|
|
||||||
|
|
||||||
@@ -357,8 +357,8 @@ typedef struct
|
|||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
extern unsigned char RDRtoPCHardwareError( unsigned char bSlot,
|
extern unsigned char RDRtoPCHardwareError( unsigned char bSlot,
|
||||||
unsigned char bSeq,
|
unsigned char bSeq,
|
||||||
unsigned char bHardwareErrorCode );
|
unsigned char bHardwareErrorCode );
|
||||||
|
|
||||||
/*
|
/*
|
||||||
#if !defined(NOAUTOCALLBACK)
|
#if !defined(NOAUTOCALLBACK)
|
||||||
@@ -368,13 +368,13 @@ extern void USBDCallbacks_RequestReceived(const USBGenericRequest *request);
|
|||||||
extern void CCID_SmartCardRequest( void );
|
extern void CCID_SmartCardRequest( void );
|
||||||
extern void CCIDDriver_Initialize( void );
|
extern void CCIDDriver_Initialize( void );
|
||||||
extern unsigned char CCID_Read(void *pBuffer,
|
extern unsigned char CCID_Read(void *pBuffer,
|
||||||
unsigned int dLength,
|
unsigned int dLength,
|
||||||
TransferCallback fCallback,
|
TransferCallback fCallback,
|
||||||
void *pArgument);
|
void *pArgument);
|
||||||
extern unsigned char CCID_Write(void *pBuffer,
|
extern unsigned char CCID_Write(void *pBuffer,
|
||||||
unsigned int dLength,
|
unsigned int dLength,
|
||||||
TransferCallback fCallback,
|
TransferCallback fCallback,
|
||||||
void *pArgument);
|
void *pArgument);
|
||||||
extern unsigned char CCID_Insertion( void );
|
extern unsigned char CCID_Insertion( void );
|
||||||
extern unsigned char CCID_Removal( void );
|
extern unsigned char CCID_Removal( void );
|
||||||
|
|
||||||
|
|||||||
@@ -76,9 +76,9 @@ extern uint32_t ISO7816_GetChar( uint8_t *pCharToReceive, Usart_info *usart);
|
|||||||
|
|
||||||
extern void ISO7816_IccPowerOff(void);
|
extern void ISO7816_IccPowerOff(void);
|
||||||
extern uint32_t ISO7816_XfrBlockTPDU_T0(const uint8_t *pAPDU,
|
extern uint32_t ISO7816_XfrBlockTPDU_T0(const uint8_t *pAPDU,
|
||||||
uint8_t *pMessage,
|
uint8_t *pMessage,
|
||||||
uint16_t wLength,
|
uint16_t wLength,
|
||||||
uint16_t *retlen);
|
uint16_t *retlen);
|
||||||
extern void ISO7816_Escape( void );
|
extern void ISO7816_Escape( void );
|
||||||
extern void ISO7816_RestartClock(void);
|
extern void ISO7816_RestartClock(void);
|
||||||
extern void ISO7816_StopClock( void );
|
extern void ISO7816_StopClock( void );
|
||||||
|
|||||||
@@ -1,6 +1,30 @@
|
|||||||
|
/* ISO7816-3 Fi/Di tables + computation
|
||||||
|
*
|
||||||
|
* (C) 2010-2015 by Harald Welte <laforge@gnumonks.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
|
/* Table 7 of ISO 7816-3:2006 */
|
||||||
|
extern const uint16_t fi_table[];
|
||||||
|
|
||||||
|
/* Table 8 from ISO 7816-3:2006 */
|
||||||
|
extern const uint8_t di_table[];
|
||||||
|
|
||||||
/* compute the F/D ratio based on Fi and Di values */
|
/* compute the F/D ratio based on Fi and Di values */
|
||||||
int compute_fidi_ratio(uint8_t fi, uint8_t di);
|
int compute_fidi_ratio(uint8_t fi, uint8_t di);
|
||||||
|
|||||||
@@ -1,19 +1,42 @@
|
|||||||
|
/* IRQ-safe linked lists
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <laforge@gnumonks.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "osmocom/core/linuxlist.h"
|
#include <osmocom/core/linuxlist.h>
|
||||||
|
#include "utils.h"
|
||||||
|
|
||||||
static inline void llist_add_irqsafe(struct llist_head *_new,
|
static inline void llist_add_irqsafe(struct llist_head *_new,
|
||||||
struct llist_head *head)
|
struct llist_head *head)
|
||||||
{
|
{
|
||||||
__disable_irq();
|
unsigned long x;
|
||||||
|
|
||||||
|
local_irq_save(x);
|
||||||
llist_add(_new, head);
|
llist_add(_new, head);
|
||||||
__enable_irq();
|
local_irq_restore(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void llist_add_tail_irqsafe(struct llist_head *_new,
|
static inline void llist_add_tail_irqsafe(struct llist_head *_new,
|
||||||
struct llist_head *head)
|
struct llist_head *head)
|
||||||
{
|
{
|
||||||
__disable_irq();
|
unsigned long x;
|
||||||
|
|
||||||
|
local_irq_save(x);
|
||||||
llist_add_tail(_new, head);
|
llist_add_tail(_new, head);
|
||||||
__enable_irq();
|
__enable_irq();
|
||||||
}
|
}
|
||||||
@@ -21,15 +44,16 @@ static inline void llist_add_tail_irqsafe(struct llist_head *_new,
|
|||||||
static inline struct llist_head *llist_head_dequeue_irqsafe(struct llist_head *head)
|
static inline struct llist_head *llist_head_dequeue_irqsafe(struct llist_head *head)
|
||||||
{
|
{
|
||||||
struct llist_head *lh;
|
struct llist_head *lh;
|
||||||
|
unsigned long x;
|
||||||
|
|
||||||
__disable_irq();
|
local_irq_save(x);
|
||||||
if (llist_empty(head)) {
|
if (llist_empty(head)) {
|
||||||
lh = NULL;
|
lh = NULL;
|
||||||
} else {
|
} else {
|
||||||
lh = head->next;
|
lh = head->next;
|
||||||
llist_del(lh);
|
llist_del(lh);
|
||||||
}
|
}
|
||||||
__enable_irq();
|
local_irq_restore(x);
|
||||||
|
|
||||||
return lh;
|
return lh;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* Ring buffer
|
||||||
|
*
|
||||||
|
* 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 SIMTRACE_RINGBUF_H
|
#ifndef SIMTRACE_RINGBUF_H
|
||||||
#define SIMTRACE_RINGBUF_H
|
#define SIMTRACE_RINGBUF_H
|
||||||
|
|
||||||
@@ -5,7 +21,7 @@
|
|||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
|
|
||||||
#define RING_BUFLEN 128
|
#define RING_BUFLEN 512
|
||||||
|
|
||||||
typedef struct ringbuf {
|
typedef struct ringbuf {
|
||||||
uint8_t buf[RING_BUFLEN];
|
uint8_t buf[RING_BUFLEN];
|
||||||
@@ -16,7 +32,7 @@ typedef struct ringbuf {
|
|||||||
void rbuf_reset(volatile ringbuf * rb);
|
void rbuf_reset(volatile ringbuf * rb);
|
||||||
uint8_t rbuf_read(volatile ringbuf * rb);
|
uint8_t rbuf_read(volatile ringbuf * rb);
|
||||||
uint8_t rbuf_peek(volatile ringbuf * rb);
|
uint8_t rbuf_peek(volatile ringbuf * rb);
|
||||||
void rbuf_write(volatile ringbuf * rb, uint8_t item);
|
int rbuf_write(volatile ringbuf * rb, uint8_t item);
|
||||||
bool rbuf_is_empty(volatile ringbuf * rb);
|
bool rbuf_is_empty(volatile ringbuf * rb);
|
||||||
bool rbuf_is_full(volatile ringbuf * rb);
|
bool rbuf_is_full(volatile ringbuf * rb);
|
||||||
|
|
||||||
|
|||||||
@@ -1,24 +1,31 @@
|
|||||||
|
/* SIMtrace 2 mode definitions
|
||||||
|
*
|
||||||
|
* Copyright (c) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* Copyright (c) 2018-2019, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#ifndef SIMTRACE_H
|
#ifndef SIMTRACE_H
|
||||||
#define SIMTRACE_H
|
#define SIMTRACE_H
|
||||||
|
|
||||||
#include "ringbuffer.h"
|
#include "ringbuffer.h"
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
|
#include <usb/device/dfu/dfu.h>
|
||||||
/* Endpoint numbers */
|
|
||||||
#define DATAOUT 1
|
|
||||||
#define DATAIN 2
|
|
||||||
#define INT 3
|
|
||||||
|
|
||||||
#define BUFLEN 512
|
#define BUFLEN 512
|
||||||
|
|
||||||
#define PHONE_DATAOUT 4
|
|
||||||
#define PHONE_DATAIN 5
|
|
||||||
#define PHONE_INT 6
|
|
||||||
|
|
||||||
#define CARDEM_USIM2_DATAOUT DATAOUT
|
|
||||||
#define CARDEM_USIM2_DATAIN DATAIN
|
|
||||||
#define CARDEM_USIM2_INT INT
|
|
||||||
|
|
||||||
#define CLK_MASTER true
|
#define CLK_MASTER true
|
||||||
#define CLK_SLAVE false
|
#define CLK_SLAVE false
|
||||||
|
|
||||||
@@ -51,6 +58,7 @@ enum confNum {
|
|||||||
#ifdef HAVE_MITM
|
#ifdef HAVE_MITM
|
||||||
CFG_NUM_MITM,
|
CFG_NUM_MITM,
|
||||||
#endif
|
#endif
|
||||||
|
CFG_NUM_VERSION,
|
||||||
NUM_CONF
|
NUM_CONF
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -59,24 +67,28 @@ enum confNum {
|
|||||||
/// device using the CCID driver.
|
/// device using the CCID driver.
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
|
||||||
/// Configuration descriptor
|
/// Configuration descriptor
|
||||||
USBConfigurationDescriptor configuration;
|
USBConfigurationDescriptor configuration;
|
||||||
/// Interface descriptor
|
/// Interface descriptor
|
||||||
USBInterfaceDescriptor interface;
|
USBInterfaceDescriptor interface;
|
||||||
/// CCID descriptor
|
/// CCID descriptor
|
||||||
CCIDDescriptor ccid;
|
CCIDDescriptor ccid;
|
||||||
/// Bulk OUT endpoint descriptor
|
/// Bulk OUT endpoint descriptor
|
||||||
USBEndpointDescriptor bulkOut;
|
USBEndpointDescriptor bulkOut;
|
||||||
/// Bulk IN endpoint descriptor
|
/// Bulk IN endpoint descriptor
|
||||||
USBEndpointDescriptor bulkIn;
|
USBEndpointDescriptor bulkIn;
|
||||||
/// Interrupt OUT endpoint descriptor
|
/// Interrupt OUT endpoint descriptor
|
||||||
USBEndpointDescriptor interruptIn;
|
USBEndpointDescriptor interruptIn;
|
||||||
|
DFURT_IF_DESCRIPTOR_STRUCT
|
||||||
} __attribute__ ((packed)) CCIDDriverConfigurationDescriptors;
|
} __attribute__ ((packed)) CCIDDriverConfigurationDescriptors;
|
||||||
|
|
||||||
extern const USBConfigurationDescriptor *configurationDescriptorsArr[];
|
extern const USBConfigurationDescriptor *configurationDescriptorsArr[];
|
||||||
|
|
||||||
int check_data_from_phone();
|
/*! Update USART baud rate to Fi/Di ratio
|
||||||
void update_fidi(uint8_t fidi);
|
* @param[io] usart USART peripheral base address
|
||||||
|
* @param[in] fidi FiDi value as provided in TA interface byte
|
||||||
|
*/
|
||||||
|
void update_fidi(Usart_info *usart, uint8_t fidi);
|
||||||
|
|
||||||
void ISR_PhoneRST( const Pin *pPin);
|
void ISR_PhoneRST( const Pin *pPin);
|
||||||
|
|
||||||
@@ -106,6 +118,9 @@ extern void CCID_run( void );
|
|||||||
extern void mode_cardemu_run(void);
|
extern void mode_cardemu_run(void);
|
||||||
extern void MITM_run( void );
|
extern void MITM_run( void );
|
||||||
|
|
||||||
|
/* IRQ functions */
|
||||||
|
extern void Sniffer_usart0_irq(void);
|
||||||
|
extern void Sniffer_usart1_irq(void);
|
||||||
extern void mode_cardemu_usart0_irq(void);
|
extern void mode_cardemu_usart0_irq(void);
|
||||||
extern void mode_cardemu_usart1_irq(void);
|
extern void mode_cardemu_usart1_irq(void);
|
||||||
|
|
||||||
|
|||||||
@@ -1,26 +1,26 @@
|
|||||||
|
/* SIMtrace2 USB protocol
|
||||||
|
*
|
||||||
|
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <stdbool.h>
|
||||||
/* SIMtrace2 USB protocol */
|
|
||||||
|
|
||||||
/* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.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
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
/***********************************************************************
|
/***********************************************************************
|
||||||
* COMMON HEADER
|
* COMMON HEADER
|
||||||
@@ -30,10 +30,10 @@ enum simtrace_msg_class {
|
|||||||
SIMTRACE_MSGC_GENERIC = 0,
|
SIMTRACE_MSGC_GENERIC = 0,
|
||||||
/* Card Emulation / Forwarding */
|
/* Card Emulation / Forwarding */
|
||||||
SIMTRACE_MSGC_CARDEM,
|
SIMTRACE_MSGC_CARDEM,
|
||||||
/* Modem Control (if modem is attached next to device */
|
/* Modem Control (if modem is attached next to device) */
|
||||||
SIMTRACE_MSGC_MODEM,
|
SIMTRACE_MSGC_MODEM,
|
||||||
/* SIM protocol tracing */
|
/* Reader/phone-car/SIM communication sniff */
|
||||||
SIMTRACE_MSGC_TRACE,
|
SIMTRACE_MSGC_SNIFF,
|
||||||
|
|
||||||
/* first vendor-specific request */
|
/* first vendor-specific request */
|
||||||
_SIMTRACE_MGSC_VENDOR_FIRST = 127,
|
_SIMTRACE_MGSC_VENDOR_FIRST = 127,
|
||||||
@@ -74,10 +74,18 @@ enum simtrace_msg_type_modem {
|
|||||||
SIMTRACE_MSGT_BD_MODEM_STATUS,
|
SIMTRACE_MSGT_BD_MODEM_STATUS,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* SIMTRACE_MSGC_TRACE */
|
/* SIMTRACE_MSGC_SNIFF */
|
||||||
enum simtrace_msg_type_trace {
|
enum simtrace_msg_type_sniff {
|
||||||
/* FIXME */
|
/* Status change (card inserted, reset, ...) */
|
||||||
_dummy,
|
SIMTRACE_MSGT_SNIFF_CHANGE = 0,
|
||||||
|
/* Fi/Di baudrate change */
|
||||||
|
SIMTRACE_MSGT_SNIFF_FIDI,
|
||||||
|
/* ATR data */
|
||||||
|
SIMTRACE_MSGT_SNIFF_ATR,
|
||||||
|
/* PPS (request or response) data */
|
||||||
|
SIMTRACE_MSGT_SNIFF_PPS,
|
||||||
|
/* TPDU data */
|
||||||
|
SIMTRACE_MSGT_SNIFF_TPDU,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* common message header */
|
/* common message header */
|
||||||
@@ -92,7 +100,7 @@ struct simtrace_msg_hdr {
|
|||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
/***********************************************************************
|
/***********************************************************************
|
||||||
* CARD EMULATOR / FORWARDER
|
* Capabilities
|
||||||
***********************************************************************/
|
***********************************************************************/
|
||||||
|
|
||||||
/* generic capabilities */
|
/* generic capabilities */
|
||||||
@@ -107,7 +115,7 @@ enum simtrace_capability_generic {
|
|||||||
SIMTRACE_CAP_LED_1,
|
SIMTRACE_CAP_LED_1,
|
||||||
/* Has LED2 */
|
/* Has LED2 */
|
||||||
SIMTRACE_CAP_LED_2,
|
SIMTRACE_CAP_LED_2,
|
||||||
/* Has Single-Pole Dual-Throw (local/remote SIM */
|
/* Has Single-Pole Dual-Throw (local/remote SIM) */
|
||||||
SIMTRACE_CAP_SPDT,
|
SIMTRACE_CAP_SPDT,
|
||||||
/* Has Bus-Switch (trace / MITM) */
|
/* Has Bus-Switch (trace / MITM) */
|
||||||
SIMTRACE_CAP_BUS_SWITCH,
|
SIMTRACE_CAP_BUS_SWITCH,
|
||||||
@@ -127,7 +135,7 @@ enum simtrace_capability_generic {
|
|||||||
SIMTRACE_CAP_ASSERT_MODEM_RST,
|
SIMTRACE_CAP_ASSERT_MODEM_RST,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* vendor-specific capabilities of sysmoocm devices */
|
/* vendor-specific capabilities of sysmocom devices */
|
||||||
enum simtrace_capability_vendor {
|
enum simtrace_capability_vendor {
|
||||||
/* Can erase a peer SAM3 controller */
|
/* Can erase a peer SAM3 controller */
|
||||||
SIMTRACE_CAP_SYSMO_QMOD_ERASE_PEER,
|
SIMTRACE_CAP_SYSMO_QMOD_ERASE_PEER,
|
||||||
@@ -137,7 +145,6 @@ enum simtrace_capability_vendor {
|
|||||||
SIMTRACE_CAP_SYSMO_QMOD_RESET_HUB,
|
SIMTRACE_CAP_SYSMO_QMOD_RESET_HUB,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
/* SIMTRACE_CMD_BD_BOARD_INFO */
|
/* SIMTRACE_CMD_BD_BOARD_INFO */
|
||||||
struct simtrace_board_info {
|
struct simtrace_board_info {
|
||||||
struct {
|
struct {
|
||||||
@@ -253,7 +260,7 @@ struct cardemu_usb_msg_error {
|
|||||||
|
|
||||||
/* SIMTRACE_MSGT_DT_MODEM_RESET */
|
/* SIMTRACE_MSGT_DT_MODEM_RESET */
|
||||||
struct st_modem_reset {
|
struct st_modem_reset {
|
||||||
/* 0: de-assert reset, 1: assert reset, 2: poulse reset */
|
/* 0: de-assert reset, 1: assert reset, 2: pulse reset */
|
||||||
uint8_t asserted;
|
uint8_t asserted;
|
||||||
/* if above is '2', duration of pulse in ms */
|
/* if above is '2', duration of pulse in ms */
|
||||||
uint16_t pulse_duration_msec;
|
uint16_t pulse_duration_msec;
|
||||||
@@ -276,3 +283,40 @@ struct st_modem_status {
|
|||||||
/* bit-field of changed status bits */
|
/* bit-field of changed status bits */
|
||||||
uint8_t changed_mask;
|
uint8_t changed_mask;
|
||||||
} __attribute__((packed));
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
* SNIFF
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
|
/* SIMTRACE_MSGT_SNIFF_CHANGE flags */
|
||||||
|
#define SNIFF_CHANGE_FLAG_CARD_INSERT (1<<0)
|
||||||
|
#define SNIFF_CHANGE_FLAG_CARD_EJECT (1<<1)
|
||||||
|
#define SNIFF_CHANGE_FLAG_RESET_ASSERT (1<<2)
|
||||||
|
#define SNIFF_CHANGE_FLAG_RESET_DEASSERT (1<<3)
|
||||||
|
#define SNIFF_CHANGE_FLAG_TIMEOUT_WT (1<<4)
|
||||||
|
/* SIMTRACE_MSGT_SNIFF_ATR, SIMTRACE_MSGT_SNIFF_PPS, SIMTRACE_MSGT_SNIFF_TPDU flags */
|
||||||
|
#define SNIFF_DATA_FLAG_ERROR_INCOMPLETE (1<<5)
|
||||||
|
#define SNIFF_DATA_FLAG_ERROR_MALFORMED (1<<6)
|
||||||
|
#define SNIFF_DATA_FLAG_ERROR_CHECKSUM (1<<7)
|
||||||
|
|
||||||
|
/* SIMTRACE_MSGT_SNIFF_CHANGE */
|
||||||
|
struct sniff_change {
|
||||||
|
/* SIMTRACE_MSGT_SNIFF_CHANGE flags */
|
||||||
|
uint32_t flags;
|
||||||
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
|
/* SIMTRACE_MSGT_SNIFF_FIDI */
|
||||||
|
struct sniff_fidi {
|
||||||
|
/* Fi/Di values as encoded in TA1 */
|
||||||
|
uint8_t fidi;
|
||||||
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
|
/* SIMTRACE_MSGT_SNIFF_ATR, SIMTRACE_MSGT_SNIFF_PPS, SIMTRACE_MSGT_SNIFF_TPDU */
|
||||||
|
struct sniff_data {
|
||||||
|
/* data flags */
|
||||||
|
uint32_t flags;
|
||||||
|
/* data length */
|
||||||
|
uint16_t length;
|
||||||
|
/* data */
|
||||||
|
uint8_t data[0];
|
||||||
|
} __attribute__ ((packed));
|
||||||
|
|||||||
67
firmware/libcommon/include/simtrace_usb.h
Normal file
67
firmware/libcommon/include/simtrace_usb.h
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
/* SIMtrace 2 USB definitions
|
||||||
|
*
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.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
|
||||||
|
*/
|
||||||
|
/* SIMtrace USB IDs */
|
||||||
|
#define USB_VENDOR_OPENMOKO 0x1d50
|
||||||
|
#define USB_PRODUCT_OWHW_SAM3_DFU 0x4001 /* was 0x4000 */
|
||||||
|
#define USB_PRODUCT_OWHW_SAM3 0x4001
|
||||||
|
#define USB_PRODUCT_QMOD_HUB 0x4002
|
||||||
|
#define USB_PRODUCT_QMOD_SAM3_DFU 0x4004 /* was 0x4003 */
|
||||||
|
#define USB_PRODUCT_QMOD_SAM3 0x4004
|
||||||
|
#define USB_PRODUCT_SIMTRACE2_DFU 0x60e3 /* was 0x60e2 */
|
||||||
|
#define USB_PRODUCT_SIMTRACE2 0x60e3
|
||||||
|
|
||||||
|
/* USB proprietary class */
|
||||||
|
#define USB_CLASS_PROPRIETARY 0xff
|
||||||
|
|
||||||
|
/* SIMtrace USB sub-classes */
|
||||||
|
/*! Sniffer USB sub-class */
|
||||||
|
#define SIMTRACE_SNIFFER_USB_SUBCLASS 1
|
||||||
|
/*! Card-emulation USB sub-class */
|
||||||
|
#define SIMTRACE_CARDEM_USB_SUBCLASS 2
|
||||||
|
|
||||||
|
/* Generic USB endpoint numbers */
|
||||||
|
/*! Card-side USB data out (host to device) endpoint number */
|
||||||
|
#define SIMTRACE_USB_EP_CARD_DATAOUT 1
|
||||||
|
/*! Card-side USB data in (device to host) endpoint number */
|
||||||
|
#define SIMTRACE_USB_EP_CARD_DATAIN 2
|
||||||
|
/*! Card-side USB interrupt endpoint number */
|
||||||
|
#define SIMTRACE_USB_EP_CARD_INT 3
|
||||||
|
/*! Phone-side USB data out (host to device) endpoint number */
|
||||||
|
#define SIMTRACE_USB_EP_PHONE_DATAOUT 4
|
||||||
|
/*! Phone-side USB data in (device to host) endpoint number */
|
||||||
|
#define SIMTRACE_USB_EP_PHONE_DATAIN 5
|
||||||
|
/*! Phone-side USB interrupt endpoint number */
|
||||||
|
#define SIMTRACE_USB_EP_PHONE_INT 6
|
||||||
|
|
||||||
|
/* Card-emulation USB endpoint numbers */
|
||||||
|
/*! USIM1 USB data out (host to device) endpoint number */
|
||||||
|
#define SIMTRACE_CARDEM_USB_EP_USIM1_DATAOUT 4
|
||||||
|
/*! USIM1 USB data in (device to host) endpoint number */
|
||||||
|
#define SIMTRACE_CARDEM_USB_EP_USIM1_DATAIN 5
|
||||||
|
/*! USIM1 USB interrupt endpoint number */
|
||||||
|
#define SIMTRACE_CARDEM_USB_EP_USIM1_INT 6
|
||||||
|
/*! USIM2 USB data out (host to device) endpoint number */
|
||||||
|
#define SIMTRACE_CARDEM_USB_EP_USIM2_DATAOUT 1
|
||||||
|
/*! USIM2 USB data in (device to host) endpoint number */
|
||||||
|
#define SIMTRACE_CARDEM_USB_EP_USIM2_DATAIN 2
|
||||||
|
/*! USIM2 USB interrupt endpoint number */
|
||||||
|
#define SIMTRACE_CARDEM_USB_EP_USIM2_INT 3
|
||||||
|
|
||||||
|
/*! Maximum number of endpoints */
|
||||||
|
#define BOARD_USB_NUMENDPOINTS 6
|
||||||
@@ -1,3 +1,31 @@
|
|||||||
|
/* ----------------------------------------------------------------------------
|
||||||
|
* ATMEL Microcontroller Software Support
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
|
* Copyright (c) 2008, Atmel Corporation
|
||||||
|
*
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the disclaimer below.
|
||||||
|
*
|
||||||
|
* Atmel's name may not be used to endorse or promote products derived from
|
||||||
|
* this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR
|
||||||
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
|
||||||
|
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
||||||
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
@@ -24,9 +52,14 @@ signed int printf(const char *pFormat, ...);
|
|||||||
signed int sprintf(char *pStr, const char *pFormat, ...);
|
signed int sprintf(char *pStr, const char *pFormat, ...);
|
||||||
signed int puts(const char *pStr);
|
signed int puts(const char *pStr);
|
||||||
|
|
||||||
|
|
||||||
int fputc(int c, FILE *stream);
|
int fputc(int c, FILE *stream);
|
||||||
int fputs(const char *s, FILE *stream);
|
int fputs(const char *s, FILE *stream);
|
||||||
|
|
||||||
#define putc(c, stream) fputc(c, stream)
|
#define putc(c, stream) fputc(c, stream)
|
||||||
#define putchar(c) fputc(c, stdout)
|
#define putchar(c) fputc(c, stdout)
|
||||||
|
|
||||||
|
signed int vfprintf_sync(FILE *pStream, const char *pFormat, va_list ap);
|
||||||
|
signed int vprintf_sync(const char *pFormat, va_list ap);
|
||||||
|
signed int printf_sync(const char *pFormat, ...);
|
||||||
|
int fputc_sync(int c, FILE *stream);
|
||||||
|
int fputs_sync(const char *s, FILE *stream);
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* Memory allocation library
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|||||||
@@ -1,7 +1,23 @@
|
|||||||
|
/* USB buffer library
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "osmocom/core/linuxlist.h"
|
#include <osmocom/core/linuxlist.h>
|
||||||
#include "osmocom/core/msgb.h"
|
#include <osmocom/core/msgb.h>
|
||||||
|
|
||||||
/* buffered USB endpoint (with queue of msgb) */
|
/* buffered USB endpoint (with queue of msgb) */
|
||||||
struct usb_buffered_ep {
|
struct usb_buffered_ep {
|
||||||
|
|||||||
@@ -1,3 +1,19 @@
|
|||||||
|
/* General utilities
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
|
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user