mirror of
https://gitea.osmocom.org/sim-card/simtrace2.git
synced 2026-03-18 22:38:32 +03:00
Compare commits
216 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
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 |
27
Makefile
Normal file
27
Makefile
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
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:
|
||||||
|
make -C host
|
||||||
|
|
||||||
|
clean: fw-clean
|
||||||
|
make -C host clean
|
||||||
|
|
||||||
|
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
|
||||||
11
debian/changelog
vendored
Normal file
11
debian/changelog
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
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
|
||||||
31
debian/control
vendored
Normal file
31
debian/control
vendored
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
Source: simtrace2
|
||||||
|
Maintainer: Harald Welte <laforge@gnumonks.org>
|
||||||
|
Section: devel
|
||||||
|
Priority: optional
|
||||||
|
Build-Depends: debhelper (>= 9),
|
||||||
|
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}
|
||||||
|
Recommends: simtrace2-firmware
|
||||||
|
Description: Host utilities to communicate with SIMtrace2 USB Devices.
|
||||||
4
debian/rules
vendored
Executable file
4
debian/rules
vendored
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/usr/bin/make -f
|
||||||
|
|
||||||
|
%:
|
||||||
|
dh $@
|
||||||
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]))
|
||||||
|
|||||||
@@ -1,24 +1,22 @@
|
|||||||
/* ISO7816-3 state machine for the card side */
|
/* ISO7816-3 state machine for the card side
|
||||||
/* (C) 2010-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* (C) 2010-2017 by Harald Welte <laforge@gnumonks.org>
|
||||||
* it under the terms of the GNU General Public License as published by
|
* (C) 2018 by 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
|
||||||
*/
|
*/
|
||||||
|
|
||||||
//#define TRACE_LEVEL 6
|
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
@@ -33,8 +31,8 @@
|
|||||||
#include "card_emu.h"
|
#include "card_emu.h"
|
||||||
#include "simtrace_prot.h"
|
#include "simtrace_prot.h"
|
||||||
#include "usb_buf.h"
|
#include "usb_buf.h"
|
||||||
#include "osmocom/core/linuxlist.h"
|
#include <osmocom/core/linuxlist.h>
|
||||||
#include "osmocom/core/msgb.h"
|
#include <osmocom/core/msgb.h>
|
||||||
|
|
||||||
|
|
||||||
#define NUM_SLOTS 2
|
#define NUM_SLOTS 2
|
||||||
@@ -56,6 +54,46 @@ enum iso7816_3_card_state {
|
|||||||
ISO_S_IN_TPDU, /* inside a TPDU */
|
ISO_S_IN_TPDU, /* inside a TPDU */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const struct value_string iso7816_3_card_state_names[] = {
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_POWER,
|
||||||
|
.str = "WAIT_POWER",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_CLK,
|
||||||
|
.str = "WAIT_CLK",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_RST,
|
||||||
|
.str = "WAIT_RST",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_ATR,
|
||||||
|
.str = "WAIT_ATR",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_IN_ATR,
|
||||||
|
.str = "IN_ATR",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_IN_PTS,
|
||||||
|
.str = "IN_PTS",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_TPDU,
|
||||||
|
.str = "WAIT_TPDU",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_IN_TPDU,
|
||||||
|
.str = "IN_TPDU",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = 0,
|
||||||
|
.str = NULL,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/* detailed sub-states of ISO_S_IN_PTS */
|
/* detailed sub-states of ISO_S_IN_PTS */
|
||||||
enum pts_state {
|
enum pts_state {
|
||||||
PTS_S_WAIT_REQ_PTSS,
|
PTS_S_WAIT_REQ_PTSS,
|
||||||
@@ -72,6 +110,7 @@ enum pts_state {
|
|||||||
PTS_S_WAIT_RESP_PCK = PTS_S_WAIT_REQ_PCK | 0x10,
|
PTS_S_WAIT_RESP_PCK = PTS_S_WAIT_REQ_PCK | 0x10,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* PTS field byte index */
|
||||||
#define _PTSS 0
|
#define _PTSS 0
|
||||||
#define _PTS0 1
|
#define _PTS0 1
|
||||||
#define _PTS1 2
|
#define _PTS1 2
|
||||||
@@ -87,10 +126,50 @@ enum tpdu_state {
|
|||||||
TPDU_S_WAIT_P2, /* waiting for P2 byte from reader */
|
TPDU_S_WAIT_P2, /* waiting for P2 byte from reader */
|
||||||
TPDU_S_WAIT_P3, /* waiting for P3 byte from reader */
|
TPDU_S_WAIT_P3, /* waiting for P3 byte from reader */
|
||||||
TPDU_S_WAIT_PB, /* waiting for Tx of procedure byte */
|
TPDU_S_WAIT_PB, /* waiting for Tx of procedure byte */
|
||||||
TPDU_S_WAIT_RX, /* waiitng for more data from reader */
|
TPDU_S_WAIT_RX, /* waiting for more data from reader */
|
||||||
TPDU_S_WAIT_TX, /* waiting for more data to reader */
|
TPDU_S_WAIT_TX, /* waiting for more data to reader */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const struct value_string tpdu_state_names[] = {
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_CLA,
|
||||||
|
.str = "WAIT_CLA",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_INS,
|
||||||
|
.str = "WAIT_INS",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_P1,
|
||||||
|
.str = "WAIT_P1",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_P2,
|
||||||
|
.str = "WAIT_P2",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_P3,
|
||||||
|
.str = "WAIT_P3",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_PB,
|
||||||
|
.str = "WAIT_PB",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_RX,
|
||||||
|
.str = "WAIT_RX",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_TX,
|
||||||
|
.str = "WAIT_TX",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = 0,
|
||||||
|
.str = NULL,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/* TPDU field byte index */
|
||||||
#define _CLA 0
|
#define _CLA 0
|
||||||
#define _INS 1
|
#define _INS 1
|
||||||
#define _P1 2
|
#define _P1 2
|
||||||
@@ -98,14 +177,14 @@ enum tpdu_state {
|
|||||||
#define _P3 4
|
#define _P3 4
|
||||||
|
|
||||||
struct card_handle {
|
struct card_handle {
|
||||||
uint32_t num;
|
unsigned int num;
|
||||||
|
|
||||||
enum iso7816_3_card_state state;
|
enum iso7816_3_card_state state;
|
||||||
|
|
||||||
/* signal levels */
|
/* signal levels */
|
||||||
uint8_t vcc_active; /* 1 = on, 0 = off */
|
bool vcc_active; /*< if VCC is active (true = active/ON) */
|
||||||
uint8_t in_reset; /* 1 = RST low, 0 = RST high */
|
bool in_reset; /*< if card is in reset (true = RST low/asserted, false = RST high/ released) */
|
||||||
uint8_t clocked; /* 1 = active, 0 = inactive */
|
bool clocked; /*< if clock is active ( true = active, false = inactive) */
|
||||||
|
|
||||||
/* timing parameters, from PTS */
|
/* timing parameters, from PTS */
|
||||||
uint8_t fi;
|
uint8_t fi;
|
||||||
@@ -175,12 +254,35 @@ void usb_buf_upd_len_and_submit(struct msgb *msg)
|
|||||||
/* Allocate USB buffer and push + initialize simtrace_msg_hdr */
|
/* Allocate USB buffer and push + initialize simtrace_msg_hdr */
|
||||||
struct msgb *usb_buf_alloc_st(uint8_t ep, uint8_t msg_class, uint8_t msg_type)
|
struct msgb *usb_buf_alloc_st(uint8_t ep, uint8_t msg_class, uint8_t msg_type)
|
||||||
{
|
{
|
||||||
struct msgb *msg;
|
struct msgb *msg = NULL;
|
||||||
struct simtrace_msg_hdr *sh;
|
struct simtrace_msg_hdr *sh;
|
||||||
|
|
||||||
msg = usb_buf_alloc(ep);
|
while (!msg) {
|
||||||
if (!msg)
|
msg = usb_buf_alloc(ep); // try to allocate some memory
|
||||||
return NULL;
|
if (!msg) { // allocation failed, we might be out of memory
|
||||||
|
struct llist_head *queue = usb_get_queue(ep);
|
||||||
|
if (!queue) {
|
||||||
|
TRACE_ERROR("ep %u: %s queue does not exist\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
if (llist_empty(queue)) {
|
||||||
|
TRACE_ERROR("ep %u: %s EOMEM (queue already empty)\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
msg = msgb_dequeue(queue);
|
||||||
|
if (!msg) {
|
||||||
|
TRACE_ERROR("ep %u: %s no msg in non-empty queue\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
usb_buf_free(msg);
|
||||||
|
msg = NULL;
|
||||||
|
TRACE_DEBUG("ep %u: %s queue msg dropped\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
msg->l1h = msgb_put(msg, sizeof(*sh));
|
msg->l1h = msgb_put(msg, sizeof(*sh));
|
||||||
sh = (struct simtrace_msg_hdr *) msg->l1h;
|
sh = (struct simtrace_msg_hdr *) msg->l1h;
|
||||||
@@ -192,7 +294,7 @@ struct msgb *usb_buf_alloc_st(uint8_t ep, uint8_t msg_class, uint8_t msg_type)
|
|||||||
return msg;
|
return msg;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Update cardemu_usb_msg_rx_data length + submit bufffer */
|
/* Update cardemu_usb_msg_rx_data length + submit buffer */
|
||||||
static void flush_rx_buffer(struct card_handle *ch)
|
static void flush_rx_buffer(struct card_handle *ch)
|
||||||
{
|
{
|
||||||
struct msgb *msg;
|
struct msgb *msg;
|
||||||
@@ -205,14 +307,17 @@ static void flush_rx_buffer(struct card_handle *ch)
|
|||||||
|
|
||||||
ch->uart_rx_msg = NULL;
|
ch->uart_rx_msg = NULL;
|
||||||
|
|
||||||
/* store length of data payload fild in header */
|
/* store length of data payload field in header */
|
||||||
rd = (struct cardemu_usb_msg_rx_data *) msg->l2h;
|
rd = (struct cardemu_usb_msg_rx_data *) msg->l2h;
|
||||||
rd->data_len = msgb_l2len(msg) - sizeof(*rd);
|
rd->data_len = msgb_l2len(msg) - sizeof(*rd);
|
||||||
|
|
||||||
|
TRACE_INFO("%u: %s (%u)\n\r",
|
||||||
|
ch->num, __func__, rd->data_len);
|
||||||
|
|
||||||
usb_buf_upd_len_and_submit(msg);
|
usb_buf_upd_len_and_submit(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* convert a non-contiguous PTS request/responsei into a contiguous
|
/* convert a non-contiguous PTS request/response into a contiguous
|
||||||
* buffer, returning the number of bytes used in the buffer */
|
* buffer, returning the number of bytes used in the buffer */
|
||||||
static int serialize_pts(uint8_t *out, const uint8_t *in)
|
static int serialize_pts(uint8_t *out, const uint8_t *in)
|
||||||
{
|
{
|
||||||
@@ -287,8 +392,9 @@ static void card_set_state(struct card_handle *ch,
|
|||||||
if (ch->state == new_state)
|
if (ch->state == new_state)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
TRACE_DEBUG("%u: 7816 card state %u -> %u\r\n", ch->num,
|
TRACE_DEBUG("%u: 7816 card state %s -> %s\r\n", ch->num,
|
||||||
ch->state, new_state);
|
get_value_string(iso7816_3_card_state_names, ch->state),
|
||||||
|
get_value_string(iso7816_3_card_state_names, new_state));
|
||||||
ch->state = new_state;
|
ch->state = new_state;
|
||||||
|
|
||||||
switch (new_state) {
|
switch (new_state) {
|
||||||
@@ -297,34 +403,48 @@ static void card_set_state(struct card_handle *ch,
|
|||||||
case ISO_S_WAIT_RST:
|
case ISO_S_WAIT_RST:
|
||||||
/* disable Rx and Tx of UART */
|
/* disable Rx and Tx of UART */
|
||||||
card_emu_uart_enable(ch->uart_chan, 0);
|
card_emu_uart_enable(ch->uart_chan, 0);
|
||||||
|
/* check end activation state (only necessary if the reader to not respect the activation sequence) */
|
||||||
|
if (ch->vcc_active && ch->clocked && !ch->in_reset) {
|
||||||
|
/* enable the TC/ETU counter once reset has been released */
|
||||||
|
tc_etu_enable(ch->tc_chan);
|
||||||
|
/* prepare to send the ATR */
|
||||||
|
card_set_state(ch, ISO_S_WAIT_ATR);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case ISO_S_WAIT_ATR:
|
case ISO_S_WAIT_ATR:
|
||||||
set_pts_state(ch, PTS_S_WAIT_REQ_PTSS);
|
|
||||||
/* Reset to initial Fi / Di ratio */
|
/* Reset to initial Fi / Di ratio */
|
||||||
ch->fi = 1;
|
ch->fi = 1;
|
||||||
ch->di = 1;
|
ch->di = 1;
|
||||||
emu_update_fidi(ch);
|
emu_update_fidi(ch);
|
||||||
/* initialize todefault WI, this will be overwritten if we
|
/* the ATR should only be sent 400 to 40k clock cycles after the RESET.
|
||||||
* receive TC2, and it will be programmed into hardware after
|
* we use the tc_etu mechanism to wait this time.
|
||||||
|
* since the initial ETU is Fd=372/Dd=1 clock cycles long, we have to wait 2-107 ETU.
|
||||||
|
*/
|
||||||
|
tc_etu_set_wtime(ch->tc_chan, 2);
|
||||||
|
/* ensure the TC_ETU timer is enabled */
|
||||||
|
tc_etu_enable(ch->tc_chan);
|
||||||
|
break;
|
||||||
|
case ISO_S_IN_ATR:
|
||||||
|
/* initialize to default WI, this will be overwritten if we
|
||||||
|
* send TC2, and it will be programmed into hardware after
|
||||||
* ATR is finished */
|
* ATR is finished */
|
||||||
ch->wi = ISO7816_3_DEFAULT_WI;
|
ch->wi = ISO7816_3_DEFAULT_WI;
|
||||||
/* update waiting time to initial waiting time */
|
/* update waiting time to initial waiting time */
|
||||||
ch->waiting_time = ISO7816_3_INIT_WTIME;
|
ch->waiting_time = ISO7816_3_INIT_WTIME;
|
||||||
|
/* set initial waiting time */
|
||||||
tc_etu_set_wtime(ch->tc_chan, ch->waiting_time);
|
tc_etu_set_wtime(ch->tc_chan, ch->waiting_time);
|
||||||
/* Set ATR sub-state to initial state */
|
/* Set ATR sub-state to initial state */
|
||||||
ch->atr.idx = 0;
|
ch->atr.idx = 0;
|
||||||
//set_atr_state(ch, ATR_S_WAIT_TS);
|
/* enable USART transmission to reader */
|
||||||
/* Notice that we are just coming out of reset */
|
|
||||||
//ch->sh.flags |= SIMTRACE_FLAG_ATR;
|
|
||||||
card_emu_uart_enable(ch->uart_chan, ENABLE_TX);
|
card_emu_uart_enable(ch->uart_chan, ENABLE_TX);
|
||||||
break;
|
/* trigger USART TX IRQ to sent first ATR byte TS */
|
||||||
|
card_emu_uart_interrupt(ch->uart_chan);
|
||||||
break;
|
break;
|
||||||
case ISO_S_WAIT_TPDU:
|
case ISO_S_WAIT_TPDU:
|
||||||
/* enable the receiver, disable transmitter */
|
/* enable the receiver, disable transmitter */
|
||||||
set_tpdu_state(ch, TPDU_S_WAIT_CLA);
|
set_tpdu_state(ch, TPDU_S_WAIT_CLA);
|
||||||
card_emu_uart_enable(ch->uart_chan, ENABLE_RX);
|
card_emu_uart_enable(ch->uart_chan, ENABLE_RX);
|
||||||
break;
|
break;
|
||||||
case ISO_S_IN_ATR:
|
|
||||||
case ISO_S_IN_PTS:
|
case ISO_S_IN_PTS:
|
||||||
case ISO_S_IN_TPDU:
|
case ISO_S_IN_TPDU:
|
||||||
/* do nothing */
|
/* do nothing */
|
||||||
@@ -332,12 +452,80 @@ static void card_set_state(struct card_handle *ch,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**********************************************************************
|
||||||
|
* ATR handling
|
||||||
|
**********************************************************************/
|
||||||
|
|
||||||
|
/*! Transmit ATR data to reader
|
||||||
|
* @param[in] ch card interface connected to reader
|
||||||
|
* @return numbers of bytes transmitted
|
||||||
|
*/
|
||||||
|
static int tx_byte_atr(struct card_handle *ch)
|
||||||
|
{
|
||||||
|
if (NULL == ch) {
|
||||||
|
TRACE_ERROR("ATR TX: no card handle provided\n\r");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (ISO_S_IN_ATR != ch->state) {
|
||||||
|
TRACE_ERROR("%u: ATR TX: no in ATR state\n\r", ch->num);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Transmit ATR */
|
||||||
|
if (ch->atr.idx < ch->atr.len) {
|
||||||
|
uint8_t byte = ch->atr.atr[ch->atr.idx++];
|
||||||
|
card_emu_uart_tx(ch->uart_chan, byte);
|
||||||
|
return 1;
|
||||||
|
} else { /* The ATR has been completely transmitted */
|
||||||
|
/* search for TC2 to updated WI */
|
||||||
|
ch->wi = ISO7816_3_DEFAULT_WI;
|
||||||
|
if (ch->atr.len >= 2 && ch->atr.atr[1] & 0xf0) { /* Y1 has some data */
|
||||||
|
uint8_t atr_td1 = 2;
|
||||||
|
if (ch->atr.atr[1] & 0x10) { /* TA1 is present */
|
||||||
|
atr_td1++;
|
||||||
|
}
|
||||||
|
if (ch->atr.atr[1] & 0x20) { /* TB1 is present */
|
||||||
|
atr_td1++;
|
||||||
|
}
|
||||||
|
if (ch->atr.atr[1] & 0x40) { /* TC1 is present */
|
||||||
|
atr_td1++;
|
||||||
|
}
|
||||||
|
if (ch->atr.atr[1] & 0x80) { /* TD1 is present */
|
||||||
|
if (ch->atr.len > atr_td1 && ch->atr.atr[atr_td1] & 0xf0) { /* Y2 has some data */
|
||||||
|
uint8_t atr_tc2 = atr_td1+1;
|
||||||
|
if (ch->atr.atr[atr_td1] & 0x10) { /* TA2 is present */
|
||||||
|
atr_tc2++;
|
||||||
|
}
|
||||||
|
if (ch->atr.atr[atr_td1] & 0x20) { /* TB2 is present */
|
||||||
|
atr_tc2++;
|
||||||
|
}
|
||||||
|
if (ch->atr.atr[atr_td1] & 0x40) { /* TC2 is present */
|
||||||
|
if (ch->atr.len > atr_tc2 && ch->atr.atr[atr_tc2]) { /* TC2 encodes WI */
|
||||||
|
ch->wi = ch->atr.atr[atr_tc2]; /* set WI */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* update waiting time (see ISO 7816-3 10.2) */
|
||||||
|
ch->waiting_time = ch->wi * 960 * ch->fi;
|
||||||
|
tc_etu_set_wtime(ch->tc_chan, ch->waiting_time);
|
||||||
|
/* reset PTS to initial state */
|
||||||
|
set_pts_state(ch, PTS_S_WAIT_REQ_PTSS);
|
||||||
|
/* go to next state */
|
||||||
|
card_set_state(ch, ISO_S_WAIT_TPDU);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* return number of bytes transmitted */
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
/**********************************************************************
|
/**********************************************************************
|
||||||
* PTS / PPS handling
|
* PTS / PPS handling
|
||||||
**********************************************************************/
|
**********************************************************************/
|
||||||
|
|
||||||
/* Update the ATR sub-state */
|
/* Update the PTS sub-state */
|
||||||
static void set_pts_state(struct card_handle *ch, enum pts_state new_ptss)
|
static void set_pts_state(struct card_handle *ch, enum pts_state new_ptss)
|
||||||
{
|
{
|
||||||
TRACE_DEBUG("%u: 7816 PTS state %u -> %u\r\n",
|
TRACE_DEBUG("%u: 7816 PTS state %u -> %u\r\n",
|
||||||
@@ -558,9 +746,9 @@ static void set_tpdu_state(struct card_handle *ch, enum tpdu_state new_ts)
|
|||||||
if (ch->tpdu.state == new_ts)
|
if (ch->tpdu.state == new_ts)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
TRACE_DEBUG("%u: 7816 TPDU state %u -> %u\r\n", ch->num,
|
TRACE_DEBUG("%u: 7816 TPDU state %s -> %s\r\n", ch->num,
|
||||||
ch->tpdu.state, new_ts);
|
get_value_string(tpdu_state_names, ch->tpdu.state),
|
||||||
|
get_value_string(tpdu_state_names, new_ts));
|
||||||
ch->tpdu.state = new_ts;
|
ch->tpdu.state = new_ts;
|
||||||
|
|
||||||
switch (new_ts) {
|
switch (new_ts) {
|
||||||
@@ -718,7 +906,7 @@ static int tx_byte_tpdu(struct card_handle *ch)
|
|||||||
|
|
||||||
card_emu_uart_tx(ch->uart_chan, byte);
|
card_emu_uart_tx(ch->uart_chan, byte);
|
||||||
|
|
||||||
/* this must happen _after_ the byte has been transmittd */
|
/* this must happen _after_ the byte has been transmitted */
|
||||||
switch (ch->tpdu.state) {
|
switch (ch->tpdu.state) {
|
||||||
case TPDU_S_WAIT_PB:
|
case TPDU_S_WAIT_PB:
|
||||||
/* if we just transmitted the procedure byte, we need to decide
|
/* if we just transmitted the procedure byte, we need to decide
|
||||||
@@ -765,14 +953,6 @@ void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte)
|
|||||||
ch->stats.rx_bytes++;
|
ch->stats.rx_bytes++;
|
||||||
|
|
||||||
switch (ch->state) {
|
switch (ch->state) {
|
||||||
case ISO_S_WAIT_POWER:
|
|
||||||
case ISO_S_WAIT_CLK:
|
|
||||||
case ISO_S_WAIT_RST:
|
|
||||||
case ISO_S_WAIT_ATR:
|
|
||||||
TRACE_ERROR("%u: Received UART char in invalid 7816 state "
|
|
||||||
"%u\r\n", ch->num, ch->state);
|
|
||||||
/* we shouldn't receive any data from the reader yet! */
|
|
||||||
break;
|
|
||||||
case ISO_S_WAIT_TPDU:
|
case ISO_S_WAIT_TPDU:
|
||||||
if (byte == 0xff) {
|
if (byte == 0xff) {
|
||||||
new_state = process_byte_pts(ch, byte);
|
new_state = process_byte_pts(ch, byte);
|
||||||
@@ -786,6 +966,10 @@ void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte)
|
|||||||
case ISO_S_IN_PTS:
|
case ISO_S_IN_PTS:
|
||||||
new_state = process_byte_pts(ch, byte);
|
new_state = process_byte_pts(ch, byte);
|
||||||
goto out_silent;
|
goto out_silent;
|
||||||
|
default:
|
||||||
|
TRACE_ERROR("%u: Received UART char in invalid 7816 state "
|
||||||
|
"%u\r\n", ch->num, ch->state);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
out_silent:
|
out_silent:
|
||||||
@@ -800,17 +984,7 @@ int card_emu_tx_byte(struct card_handle *ch)
|
|||||||
|
|
||||||
switch (ch->state) {
|
switch (ch->state) {
|
||||||
case ISO_S_IN_ATR:
|
case ISO_S_IN_ATR:
|
||||||
if (ch->atr.idx < ch->atr.len) {
|
rc = tx_byte_atr(ch);
|
||||||
uint8_t byte;
|
|
||||||
byte = ch->atr.atr[ch->atr.idx++];
|
|
||||||
rc = 1;
|
|
||||||
|
|
||||||
card_emu_uart_tx(ch->uart_chan, byte);
|
|
||||||
|
|
||||||
/* detect end of ATR */
|
|
||||||
if (ch->atr.idx >= ch->atr.len)
|
|
||||||
card_set_state(ch, ISO_S_WAIT_TPDU);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case ISO_S_IN_PTS:
|
case ISO_S_IN_PTS:
|
||||||
rc = tx_byte_pts(ch);
|
rc = tx_byte_pts(ch);
|
||||||
@@ -905,9 +1079,8 @@ void card_emu_io_statechg(struct card_handle *ch, enum card_io io, int active)
|
|||||||
if (ch->vcc_active && ch->clocked) {
|
if (ch->vcc_active && ch->clocked) {
|
||||||
/* enable the TC/ETU counter once reset has been released */
|
/* enable the TC/ETU counter once reset has been released */
|
||||||
tc_etu_enable(ch->tc_chan);
|
tc_etu_enable(ch->tc_chan);
|
||||||
|
/* prepare to send the ATR */
|
||||||
card_set_state(ch, ISO_S_WAIT_ATR);
|
card_set_state(ch, ISO_S_WAIT_ATR);
|
||||||
/* FIXME: wait 400 to 40k clock cycles before sending ATR */
|
|
||||||
card_set_state(ch, ISO_S_IN_ATR);
|
|
||||||
}
|
}
|
||||||
} else if (active && !ch->in_reset) {
|
} else if (active && !ch->in_reset) {
|
||||||
TRACE_INFO("%u: RST asserted\r\n", ch->num);
|
TRACE_INFO("%u: RST asserted\r\n", ch->num);
|
||||||
@@ -928,7 +1101,15 @@ int card_emu_set_atr(struct card_handle *ch, const uint8_t *atr, uint8_t len)
|
|||||||
ch->atr.len = len;
|
ch->atr.len = len;
|
||||||
ch->atr.idx = 0;
|
ch->atr.idx = 0;
|
||||||
|
|
||||||
/* FIXME: race condition with trasmitting ATR to reader? */
|
#if TRACE_LEVEL >= TRACE_LEVEL_INFO
|
||||||
|
uint8_t i;
|
||||||
|
TRACE_INFO("%u: ATR set: ", ch->num);
|
||||||
|
for (i = 0; i < ch->atr.len; i++) {
|
||||||
|
TRACE_INFO_WP("%02x ", atr[i]);
|
||||||
|
}
|
||||||
|
TRACE_INFO_WP("\n\r");
|
||||||
|
#endif
|
||||||
|
/* FIXME: race condition with transmitting ATR to reader? */
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -959,7 +1140,15 @@ void tc_etu_wtime_half_expired(void *handle)
|
|||||||
void tc_etu_wtime_expired(void *handle)
|
void tc_etu_wtime_expired(void *handle)
|
||||||
{
|
{
|
||||||
struct card_handle *ch = handle;
|
struct card_handle *ch = handle;
|
||||||
TRACE_ERROR("%u: wtime_exp\r\n", ch->num);
|
switch (ch->state) {
|
||||||
|
case ISO_S_WAIT_ATR:
|
||||||
|
/* ISO 7816-3 6.2.1 time tc has passed, we can now send the ATR */
|
||||||
|
card_set_state(ch, ISO_S_IN_ATR);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
TRACE_ERROR("%u: wtime_exp\r\n", ch->num);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* shortest ATR found in smartcard_list.txt */
|
/* shortest ATR found in smartcard_list.txt */
|
||||||
@@ -967,8 +1156,7 @@ static const uint8_t default_atr[] = { 0x3B, 0x02, 0x14, 0x50 };
|
|||||||
|
|
||||||
static struct card_handle card_handles[NUM_SLOTS];
|
static struct card_handle card_handles[NUM_SLOTS];
|
||||||
|
|
||||||
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan,
|
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)
|
||||||
uint8_t in_ep, uint8_t irq_ep)
|
|
||||||
{
|
{
|
||||||
struct card_handle *ch;
|
struct card_handle *ch;
|
||||||
|
|
||||||
@@ -981,14 +1169,13 @@ struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uar
|
|||||||
|
|
||||||
INIT_LLIST_HEAD(&ch->uart_tx_queue);
|
INIT_LLIST_HEAD(&ch->uart_tx_queue);
|
||||||
|
|
||||||
/* initialize the card_handle with reasonabe defaults */
|
|
||||||
ch->num = slot_num;
|
ch->num = slot_num;
|
||||||
ch->irq_ep = irq_ep;
|
ch->irq_ep = irq_ep;
|
||||||
ch->in_ep = in_ep;
|
ch->in_ep = in_ep;
|
||||||
ch->state = ISO_S_WAIT_POWER;
|
ch->state = ISO_S_WAIT_POWER;
|
||||||
ch->vcc_active = 0;
|
ch->vcc_active = vcc_active;
|
||||||
ch->in_reset = 1;
|
ch->in_reset = in_reset;
|
||||||
ch->clocked = 0;
|
ch->clocked = clocked;
|
||||||
|
|
||||||
ch->fi = 0;
|
ch->fi = 0;
|
||||||
ch->di = 1;
|
ch->di = 1;
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user