71 Commits

Author SHA1 Message Date
Kévin Redon
f48aac560c sniffing: decrease USB IRQ prioprity to prevent USART overrun
Change-Id: I870a0aa8e251bbb53249c54bfcaa45de5b5a9486

Handling the USB message queue is done in an ISR and take quite some time.
This can cause a USART/SIM sniffing buffer overrun, resulting in data loss.
By setting the USB IRQ lower than the USART IRQ, the USB ISR can be
interrupted (for short) and no data gets lost.
2018-07-03 16:18:00 +02:00
Kévin Redon
d28d3fc4c0 sniffing: fix procedure byte handling and make TPDU parsing for strict
Change-Id: If991152f11c4b864ab1386f21dc13c335e6b281f
2018-07-03 16:17:00 +02:00
Kévin Redon
50d071fb0e sniffer: also send incomplete (e.g. timeout) data (PPS/ATR/TPDU)
Change-Id: Ib070aca181042b477f1ffec48d63dc56c1e4609a
2018-07-03 16:15:58 +02:00
Kévin Redon
36be182ea2 sniffing: move conversion convertion and flag processing from ISR to main loop to keep ISR fast and focus on data capture
Change-Id: Ieefa8a5f81dbcc12c1ad3059660dbffa0c1a4961
2018-07-03 16:10:04 +02:00
Kévin Redon
bdb0795e61 buffer: increase buffer size to 512 to cope with fast and long TPDUs
Change-Id: I194c90cf09306a982d80c5bf1222397af6e658a9
2018-07-03 16:03:17 +02:00
Kévin Redon
b65e4b6823 console: drop data to be send when buffer is already full
Change-Id: Ia625b09eb30bb7b43edd3989f697d8ef33200f28

don't wait for space to be available in the buffer since since would
prevent from processing non-console (e.g. debug) more important data
2018-07-03 15:59:57 +02:00
Kévin Redon
05cc8bd36a sniffer: display F and D values frim PPS
Change-Id: I3641dcb6c24695a6d3dd3a1ee4333f56a07c99f0
2018-07-03 15:57:03 +02:00
Kévin Redon
de6e3488a0 Merge branch 'kredon/simtrace' of ssh://gerrit.osmocom.org:29418/simtrace2 into kredon/simtrace
Change-Id: Id868520d6603f2bcb79a0aaaf6413dc83938524b
2018-07-01 19:02:40 +02:00
Kévin Redon
f6c2f4def3 host USB: add host application to receive and display USB sniffing messages sent by firmware
Change-Id: Idefbf21e0bbd2a1e3647fe9aebaf88d1b62dae2d
2018-07-01 18:33:28 +02:00
Kévin Redon
f9f1261c18 sniffer USB: implement USB communication and send parsed messages
Change-Id: Ice7817480705f2124b08c1ff9a8826558b6d8b2b
2018-07-01 18:33:28 +02:00
Kévin Redon
310ad81bad USB device: add USB message structure for sniffer communication
Change-Id: Id2c6f32ade2fec9b9ef91bd8c5e1fd195f2d7351
2018-07-01 18:33:28 +02:00
Kévin Redon
1c84e5e312 host USB: use central SIMtrace USB ID definition header
Change-Id: Id18e64fba0a2c308a8aef7d3865200bf0237cae9
2018-07-01 18:33:28 +02:00
Kévin Redon
c06b1d6f7c host usb_util fix: used provided class, sub-class, and interfave when finding matching interfaces
Change-Id: Ibc06e751e6ca0f9e9a40d82c4eeddfb975240f91
2018-07-01 18:33:28 +02:00
Kévin Redon
0c9079f1ae USB: add central file to define USB IDs, classes, and endpoints
Change-Id: Iba81f32a92c68a973e8e7adbc4c2a1064ba5290f
2018-07-01 18:33:28 +02:00
Kévin Redon
cb700ef087 reintroduce adc2uv used in boardver_adc.c and mode_cardemu.c
Change-Id: I52e3919adfd1d888eb130b5ec9298315c4e507c9
2018-07-01 18:33:28 +02:00
Kévin Redon
4635c71c06 minor: fix typos in comment
Change-Id: I01b49e047a586dff449d4e134751108e391a8822
2018-07-01 18:33:28 +02:00
Kévin Redon
6123460055 sniff: add WT timeout detection using USART timeout (TC is not required)
Change-Id: I4ec6e812e7e1eb91005027d2e864fc315550d79c
2018-07-01 18:33:28 +02:00
Kévin Redon
15a3384e9a sniff: add TPDU parsing (TPDUs become APDUs on the upper layer)
Change-Id: I09d050d95bd2ab140fe6b4926a37278eb08cc347
2018-07-01 18:33:28 +02:00
Kévin Redon
f40c10d2f2 sniff: print parsed ATR and PPS; use red LED to show main application is running; use green LED to indicate activity (message parsed)
Change-Id: I8e906bdbf2c91e608757ae442dfb241f981b8f1e
2018-07-01 18:33:28 +02:00
Kévin Redon
8fd88b8a59 LED: add short LED pulse blinking pattern
Change-Id: I0fdc2f902a3b92da6aa9b9c8500abae8a2f79447
2018-07-01 18:33:28 +02:00
Kévin Redon
10553e7f0c update_fidi: remove debug output since this function is called in time critical ISR
Change-Id: I08f407d407a18dae3f360ddc64769ddfaeb5b559
2018-07-01 18:33:28 +02:00
Kévin Redon
833a540efa DFU: switch green LED on to indicate DFU mode, and red LED to indicate flashing activity
Change-Id: I8e34fd869ed94ad122d6a17f5a432f5a09b820bb
2018-07-01 18:33:28 +02:00
Kévin Redon
23d7306ef0 board: fix LED pin definition
Change-Id: Ia6c80c0268dec708845e1dad281caaa42027f9db
2018-07-01 18:33:28 +02:00
Kévin Redon
4646e04d1f DFU: remove force bootloader button debug message since the console is output message is not initialized yet
Change-Id: Ibea0105929a8dc38b43dacd9d1e576d7b51d0c6a
2018-07-01 18:33:28 +02:00
Kévin Redon
216a2149e1 sniffer: use ISR to store sniffed data in buffer, add ATR and PPS parsing, and PPS related FiDi update 2018-07-01 18:33:28 +02:00
Kévin Redon
13f720b650 trace: increase watchdog for 500 to 2000 ms to provide more time handling buffered data 2018-07-01 18:33:28 +02:00
Kévin Redon
ae1ce57d4c ISO7816: change update_fidi to use provided USART, and disable write protection for USART register if required 2018-07-01 18:33:28 +02:00
Kévin Redon
5dce338c6b SIMtrace: enable interrupt on edge dection for SIM_RST pin to reset the sniffer ISO state 2018-07-01 18:33:28 +02:00
Kévin Redon
ad6e7eff00 SIMtrace: fix default SIM_RST pin state to allow phone controlled reset 2018-07-01 18:33:28 +02:00
Kévin Redon
5b63e437bc SIMtrace: only enable main sniffing mode on SIMtrace board 2018-07-01 18:33:28 +02:00
Kévin Redon
915f1636b0 sniffer: add state definitions, improve IRQ handling, update pins configuration 2018-07-01 18:33:28 +02:00
Kévin Redon
04a63b386e simtrace: add support for sniffing on both USART 2018-07-01 18:33:28 +02:00
Kévin Redon
b88c73bcfd board: comment USART definitions and add corresponding IRQ numbers 2018-07-01 18:33:28 +02:00
Kévin Redon
1225e41aad simtrace: add dedicated power pins configuration for sniffing 2018-07-01 18:33:28 +02:00
Kévin Redon
7ddf46ce1c sniff: use USART 0 instead of USART 1
Use USART 0 connected to the SIM card side to sniff the communication.
The card side can also measure ETU times.
Do proper pin initialization.
This code can already capture the ATR communication between phone and card.
2018-07-01 18:33:28 +02:00
Kévin Redon
5a7c848b43 SIMtrace board: comment and fix pin definition 2018-07-01 18:33:28 +02:00
Kévin Redon
aa70887ce0 sniff mode: handle USART 1 RX interrupt to show sniffer data 2018-07-01 18:33:28 +02:00
Kévin Redon
baff8d2c22 add more USB configuration checks and error messages 2018-07-01 18:33:28 +02:00
Kévin Redon
7bcbae3ad9 enable (empty) sniffer support for SIMtrace board 2018-07-01 18:33:28 +02:00
Kévin Redon
33d62aa17a copy working cardem app to trace
because the applications share the board capabilities defined in
libboard/*/include/board.h and USB configurations are enabled according
to the previously defined capabilities in libcommon/source.usb.c, all
applications actually offer the same functions.
thus creating the trace application is only mainly a cosmetic change, as the
sniffer function will also be present and enabled in the cardem application.
2018-07-01 18:33:28 +02:00
Kévin Redon
e4cd52c2e3 fix: remove unused code
adc2uv is not used in boardver_adc.c.
a FIXME comment says it should be shared with mode_cardemu.c.
the exact same code is already available in mode_cardemu.c
2018-07-01 18:33:28 +02:00
Kévin Redon
a484b02271 sniffer: use ISR to store sniffed data in buffer, add ATR and PPS parsing, and PPS related FiDi update 2018-06-25 16:00:33 +02:00
Kévin Redon
8643420daa trace: increase watchdog for 500 to 2000 ms to provide more time handling buffered data 2018-06-25 15:57:44 +02:00
Kévin Redon
0aafbac7eb ring buffer: increase buffer size from 128 to 256 to cope with large debug output 2018-06-25 15:56:33 +02:00
Kévin Redon
67e181fb15 console: use buffer and interrupts instead of busy loops for UART debug output 2018-06-25 15:55:33 +02:00
Kévin Redon
2bac56494f ISO7816: change update_fidi to use provided USART, and disable write protection for USART register if required 2018-06-25 15:53:19 +02:00
Kévin Redon
f908f659fc SIMtrace: enable interrupt on edge dection for SIM_RST pin to reset the sniffer ISO state 2018-06-25 15:49:28 +02:00
Kévin Redon
42af4949a7 SIMtrace: fix default SIM_RST pin state to allow phone controlled reset 2018-06-24 11:31:36 +02:00
Kévin Redon
4814b15bbf SIMtrace: only enable main sniffing mode on SIMtrace board 2018-06-24 11:30:31 +02:00
Kévin Redon
78a8ab71e1 DFU: fix typo in USB strings 2018-06-24 11:27:09 +02:00
Kévin Redon
6b38297e20 DFU: incread watchdog timeout and restart watchdog before writing in flash to prevent the watchdog to trigger while flashing 2018-06-17 22:36:44 +02:00
Kévin Redon
3f8a4c28e8 DFU: only boot the application if it has a valid start 2018-06-17 22:35:17 +02:00
Kévin Redon
41d01b9d54 DFU: uncomment print message when DFU is forced using the button 2018-06-17 22:34:47 +02:00
Kévin Redon
88b2b077ef DFU: unlock the flash before writing, verify written data, and relock it 2018-06-17 22:33:29 +02:00
Kévin Redon
9a921ac630 USB: implement USB reset by setting the on-board pull-up on D+ low 2018-06-17 22:31:21 +02:00
Kévin Redon
2ab8f8e3fd sniffer: add state definitions, improve IRQ handling, update pins configuration 2018-06-11 13:46:35 +02:00
Kévin Redon
a34471d923 simtrace: add support for sniffing on both USART 2018-06-11 13:45:16 +02:00
Kévin Redon
762276e271 board: comment USART definitions and add corresponding IRQ numbers 2018-06-11 13:43:27 +02:00
Kévin Redon
2d972f5910 simtrace: add dedicated power pins configuration for sniffing 2018-06-11 13:42:23 +02:00
Kévin Redon
d0b4a9da81 sniff: use USART 0 instead of USART 1
Use USART 0 connected to the SIM card side to sniff the communication.
The card side can also measure ETU times.
Do proper pin initialization.
This code can already capture the ATR communication between phone and card.
2018-06-07 18:56:47 +02:00
Kévin Redon
7f4f8983dd USBD: send empty packet when non-existing descriptor string is requested
Sometimes descriptor string 0xee is requested.
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.
2018-06-06 17:03:24 +02:00
Kévin Redon
86ea4faef6 SIMtrace board: comment and fix pin definition 2018-06-06 17:02:33 +02:00
Kévin Redon
a1f198276d sniff mode: handle USART 1 RX interrupt to show sniffer data 2018-06-06 16:13:48 +02:00
Kévin Redon
b53ab5b4ef add more USB configuration checks and error messages 2018-06-04 16:30:48 +02:00
Kévin Redon
38a467e630 enable (empty) sniffer support for SIMtrace board 2018-06-04 16:30:01 +02:00
Kévin Redon
7b51f72e83 copy working cardem app to trace
because the applications share the board capabilities defined in
libboard/*/include/board.h and USB configurations are enabled according
to the previously defined capabilities in libcommon/source.usb.c, all
applications actually offer the same functions.
thus creating the trace application is only mainly a cosmetic change, as the
sniffer function will also be present and enabled in the cardem application.
2018-06-04 16:21:34 +02:00
Kévin Redon
f79ae1c54a dfu: fix address destination check and add stack overwrite check in USBDFU_handle_dnload
During DFU download the destination start address is checked to not exceed the
RAM or flash end address, but it is also necessary to check if the end of the
data to be downloaded is also within the allowed range.
When downloading to RAM it is also necessary to check if the data to be
downloaded does not overwrite (i.e. corrupt) the stack.
2018-06-01 11:02:56 +02:00
Kévin Redon
de2d03ca22 README: rewrite to better explain environment variables and point to the wiki for flashing 2018-05-21 19:35:56 +02:00
Kévin Redon
fe72bf11fd fix pointer casting warning
fixes following warning:
libboard/common/source/board_cstartup_gnu.c:137:11: warning: assignment to 'void (*)(void)' from 'unsigned int' makes pointer from integer without a cast [-Wint-conversion]
  appReset = pSrc[1];
2018-05-21 19:34:01 +02:00
Kévin Redon
0471d2a390 fix: initialize uninitialized variable in USBDFU_DFU_RequestHandler 2018-05-21 18:52:06 +02:00
Kévin Redon
5be1b612f7 fix: remove unused code
adc2uv is not used in boardver_adc.c.
a FIXME comment says it should be shared with mode_cardemu.c.
the exact same code is already available in mode_cardemu.c
2018-05-21 17:51:34 +02:00
153 changed files with 9212 additions and 13659 deletions

1
.gitignore vendored
View File

@@ -21,4 +21,3 @@ host/simtrace2-list
host/simtrace2-remsim host/simtrace2-remsim
host/simtrace2-remsim-usb2udp host/simtrace2-remsim-usb2udp
usb_strings_generated.h usb_strings_generated.h
firmware/usbstring/usbstring

View File

@@ -1,32 +0,0 @@
all: fw utils
define RULES
fw-$(1)-$(2):
make -C firmware BOARD=$(1) APP=$(2)
fw-$(1)-$(2)-clean:
make -C firmware BOARD=$(1) APP=$(2) clean
endef
$(eval $(call RULES,simtrace,dfu))
$(eval $(call RULES,simtrace,trace))
$(eval $(call RULES,simtrace,cardem))
$(eval $(call RULES,qmod,dfu))
$(eval $(call RULES,qmod,cardem))
fw-clean: fw-simtrace-dfu-clean fw-simtrace-trace-clean fw-simtrace-cardem-clean fw-qmod-dfu-clean fw-qmod-cardem-clean
fw: fw-simtrace-dfu fw-simtrace-trace fw-simtrace-cardem fw-qmod-dfu fw-qmod-cardem
utils:
(cd host && \
autoreconf -fi && \
./configure --prefix=/usr --disable-werror && \
make)
clean: fw-clean
if [ -e host/Makefile ]; then \
make -C host clean; \
fi
install:
make -C firmware install
make -C host install

View File

@@ -9,7 +9,6 @@ fi
set -e set -e
publish="$1"
base="$PWD" base="$PWD"
deps="$base/deps" deps="$base/deps"
inst="$deps/install" inst="$deps/install"
@@ -25,7 +24,7 @@ export PKG_CONFIG_PATH="$inst/lib/pkgconfig:$PKG_CONFIG_PATH"
export LD_LIBRARY_PATH="$inst/lib" export LD_LIBRARY_PATH="$inst/lib"
BUILDS="" BUILDS=""
BUILDS+="simtrace/dfu simtrace/cardem simtrace/trace " # simtrace/triple_play BUILDS+="simtrace/dfu simtrace/cardem " # simtrace/trace simtrace/triple_play
BUILDS+="qmod/dfu qmod/cardem " BUILDS+="qmod/dfu qmod/cardem "
BUILDS+="owhw/dfu owhw/cardem " BUILDS+="owhw/dfu owhw/cardem "
@@ -36,6 +35,7 @@ for build in $BUILDS; do
echo echo
echo "=============== $board / $app START ==============" echo "=============== $board / $app START =============="
make BOARD="$board" APP="$app" make BOARD="$board" APP="$app"
make BOARD="$board" APP="$app" clean
echo "=============== $board / $app RES:$? ==============" echo "=============== $board / $app RES:$? =============="
done done
@@ -50,41 +50,8 @@ make clean
echo echo
echo "=============== HOST START ==============" echo "=============== HOST START =============="
cd $TOPDIR/host cd $TOPDIR/host
autoreconf --install --force make clean
./configure --enable-sanitize --enable-werror make
$MAKE $PARALLEL_MAKE make clean
#$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 osmo-clean-workspace.sh

17
debian/changelog vendored
View File

@@ -1,17 +0,0 @@
simtrace2 (0.5.2) UNRELEASED; urgency=medium
* adapt to host tools in autotools
-- Harald Welte <lafore@gnumonks.org> Thu, 28 Nov 2019 00:44:57 +0100
simtrace2 (0.5.1) unstable; urgency=medium
* Backwards-compatibility with older (released, non-master) libosmocore
-- Harald Welte <lafore@gnumonks.org> Sun, 26 Aug 2018 11:50:36 +0200
simtrace2 (0.5) unstable; urgency=medium
* Initial debian package release.
-- Harald Welte <lafore@gnumonks.org> Sun, 26 Aug 2018 10:37:19 +0200

1
debian/compat vendored
View File

@@ -1 +0,0 @@
9

65
debian/control vendored
View File

@@ -1,65 +0,0 @@
Source: simtrace2
Maintainer: Harald Welte <laforge@gnumonks.org>
Section: devel
Priority: optional
Build-Depends: debhelper (>= 9),
autotools-dev,
autoconf,
automake,
libtool,
pkg-config,
git,
dh-autoreconf,
libosmocore-dev,
libpcsclite-dev,
libnewlib-arm-none-eabi,
libusb-1.0-0-dev,
gcc-arm-none-eabi
Standards-Version: 3.9.8
Vcs-Git: git://git.osmocom.org/simtrace2.git
Vcs-Browser: http://git.osmocom.org/simtrace2/
Homepage: http://osmocom.org/projects/simtrace2/wiki
Package: simtrace2-firmware
Section: devel
Architecture: all
Recommends: dfu-util
Description: Firmware for SAM3 based SIMtrace2 USB Devices.
Open Source firmware for the Cortex-M3 microcontroller in the
"Osmocom SIMtrace2" USB-attached peripheral device. Will only work in
SAM3S-based SIMtrace2, not in its SAM7S-based predecessor SIMtrace!
Package: simtrace2-utils
Section: devel
Architecture: any
Multi-Arch: same
Depends: ${shlibs:Depends}, ${misc:Depends}, libosmo-simtrace2-0
Recommends: simtrace2-firmware
Description: Host utilities to communicate with SIMtrace2 USB Devices.
Package: libosmo-simtrace2-0
Section: libs
Architecture: any
Multi-Arch: same
Depends: ${shlibs:Depends}, ${misc:Depends}
Description: Osmocom SIMtrace2 library
This library contains core "driver" functionality to interface with the
Osmocom SIMtrace2 (and compatible) USB device firmware. It enables
applications to implement SIM card / smart card tracing as well as
SIM / smart card emulation functions.
Package: libosmo-simtrace2-dev
Section: libdevel
Architecture: any
Multi-Arch: same
Depends: libosmo-simtrace2-0, ${misc:Depends}
Description: Development headers for Osmocom SIMtrace2 library
This library contains core "driver" functionality to interface with the
Osmocom SIMtrace2 (and compatible) USB device firmware. It enables
applications to implement SIM card / smart card tracing as well as
SIM / smart card emulation functions.
.
The header files provided by this package may be used to develop
with any of the libosmocore libraries.
.
Also static libraries are installed with this package.

View File

@@ -1 +0,0 @@
usr/lib/libosmo-simtrace2*.so.*

View File

@@ -1,5 +0,0 @@
usr/include/*
usr/lib/lib*.a
usr/lib/lib*.so
usr/lib/lib*.la
usr/lib/pkgconfig/*

19
debian/rules vendored
View File

@@ -1,19 +0,0 @@
#!/usr/bin/make -f
# Uncomment this to turn on verbose mode.
#export DH_VERBOSE=1
DEBIAN := $(shell dpkg-parsechangelog | grep ^Version: | cut -d' ' -f2)
DEBVERS := $(shell echo '$(DEBIAN)' | cut -d- -f1)
VERSION := $(shell echo '$(DEBVERS)' | sed -e 's/[+-].*//' -e 's/~//g')
export DEB_BUILD_MAINT_OPTIONS = hardening=+all
export DEB_LDFLAGS_MAINT_STRIP = -Wl,-Bsymbolic-functions
%:
dh $@
override_dh_autoreconf:
cd host && dh_autoreconf

View File

@@ -1 +0,0 @@
usr/share/simtrace2/*.bin

View File

@@ -1 +0,0 @@
usr/bin/simtrace2-*

View File

@@ -28,7 +28,6 @@
# 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
#------------------------------------------------------------------------------- #-------------------------------------------------------------------------------
@@ -37,16 +36,14 @@ GIT_VERSION=$(shell $(TOP)/git-version-gen $(TOP)/.tarvers)
# (can be overriden by adding CHIP=chip and BOARD=board to the command-line) # (can be overriden by adding CHIP=chip and BOARD=board to the command-line)
CHIP ?= sam3s4 CHIP ?= sam3s4
BOARD ?= qmod BOARD ?= qmod
APP ?= dfu
# Defines which are the available memory targets for the SAM3S-EK board. # Defines which are the available memory targets for the SAM3S-EK board.
ifeq ($(APP), dfu) MEMORIES ?= flash dfu
MEMORIES ?= flash
else
MEMORIES ?= dfu
endif
# Output directories and filename # Output file basename
APP ?= dfu
# Output directories
OUTPUT = $(BOARD)-$(APP) OUTPUT = $(BOARD)-$(APP)
BIN = bin BIN = bin
OBJ = obj/$(BOARD) OBJ = obj/$(BOARD)
@@ -76,6 +73,7 @@ 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
@@ -102,7 +100,7 @@ C_LIBUSB_DFU = dfu.c dfu_desc.c dfu_driver.c
C_LIBCOMMON = string.c stdio.c fputs.c usb_buf.c ringbuffer.c pseudo_talloc.c host_communication.c C_LIBCOMMON = string.c stdio.c fputs.c usb_buf.c ringbuffer.c pseudo_talloc.c host_communication.c
C_BOARD = $(notdir $(wildcard libboard/common/source/*.c)) C_BOARD = $(notdir $(wildcard libboard/common/source/*.c))
C_BOARD += $(notdir $(wildcard libboard/$(BOARD)/source/*.c)) C_BOARD += $(notdir $(wildcard libboard/$(BOARD)/source/*.c))
C_APPLEVEL = $(notdir $(wildcard apps/$(APP)/*.c)) C_APPLEVEL = $(notdir $(wildcard apps/$(APP)/*.c))
@@ -122,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
# allow asserting the peer SAM3S ERASE signal to completely erase the flash DEBUG_PHONE_SNIFF?=0
# 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)
@@ -143,12 +141,13 @@ 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 += -Waggregate-return #-Wsign-compare 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
@@ -165,7 +164,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) -DALLOW_PEER_ERASE=$(ALLOW_PEER_ERASE) CFLAGS += -ffunction-sections -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP) -DTRACE_LEVEL=$(TRACE_LEVEL) -DDEBUG_PHONE_SNIFF=$(DEBUG_PHONE_SNIFF)
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)
@@ -173,7 +172,8 @@ 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)
#------------------------------------------------------------------------------- #-------------------------------------------------------------------------------
@@ -197,11 +197,7 @@ $(BIN) $(OBJ):
usbstring/usbstring: usbstring/usbstring.c usbstring/usbstring: usbstring/usbstring.c
gcc $^ -o $@ gcc $^ -o $@
.PHONY: apps/$(APP)/usb_strings.txt.patched apps/$(APP)/usb_strings_generated.h: apps/$(APP)/usb_strings.txt usbstring/usbstring
apps/$(APP)/usb_strings.txt.patched: apps/$(APP)/usb_strings.txt
sed "s/PRODUCT_STRING/$(shell cat libboard/$(BOARD)/product_string.txt)/" $< > $@
apps/$(APP)/usb_strings_generated.h: apps/$(APP)/usb_strings.txt.patched usbstring/usbstring
cat $< | usbstring/usbstring > $@ cat $< | usbstring/usbstring > $@
define RULES define RULES
@@ -210,12 +206,8 @@ 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)
@@ -237,13 +229,8 @@ program:
SERIAL ?= /dev/ttyUSB0 SERIAL ?= /dev/ttyUSB0
log: log:
stty -F $(SERIAL) 921600 stty -F $(SERIAL) 115200
lsof $(SERIAL) && echo "log is already opened" || ( sed -u "s/\r//" $(SERIAL) | ts ) lsof $(SERIAL) && echo "log is already opened" || ( sed -u "s/\r//" $(SERIAL) | ts )
clean: clean:
-rm -f apps/$(APP)/usb_strings.txt.patched
-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

View File

@@ -24,7 +24,6 @@ Current boards supported are:
* `simtrace`: The good old Osmocom SIMtrace PCB with SAM3 instead of SAM7, open hardware. * `simtrace`: The good old Osmocom SIMtrace PCB with SAM3 instead of SAM7, open hardware.
* `qmod`: A sysmocom-proprietary quad mPCIe carrier board, publicly available * `qmod`: A sysmocom-proprietary quad mPCIe carrier board, publicly available
* `owhw`: An undisclosed sysmocom-internal board, not publicly available * `owhw`: An undisclosed sysmocom-internal board, not publicly available
* `octsimtest`: A sysmocom-proprietary production testing board, not publicly available
= Firmware = Firmware
@@ -52,7 +51,6 @@ Current applications supported are:
* `cardem`: To provide remote SIM operation capabilities. * `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) * `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. * `triple_play`: To support the three previous functionalities, using USB configurations.
* `gpio_test`: internal test code
== Memories == Memories
@@ -78,10 +76,6 @@ $ make TRACE_LEVEL=4
``` ```
Accepted values: 0 (NO_TRACE) to 5 (DEBUG) Accepted values: 0 (NO_TRACE) to 5 (DEBUG)
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.
= Flashing = Flashing
To flash a firmware image follow the instructions provided in the [wiki](https://projects.osmocom.org/projects/simtrace2/wiki/). To flash a firmware image follow the instructions provided in the [wiki](https://projects.osmocom.org/projects/simtrace2/wiki/).

View File

@@ -1,22 +1,4 @@
/* SIMtrace 2 firmware card emulation application // FIXME: Copyright license here
*
* (C) 2015-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
*/
/*------------------------------------------------------------------------------ /*------------------------------------------------------------------------------
* Headers * Headers
*------------------------------------------------------------------------------*/ *------------------------------------------------------------------------------*/
@@ -102,11 +84,7 @@ 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);
if (cfgnum < ARRAY_SIZE(config_func_ptrs)) { simtrace_config = cfgnum;
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)
@@ -130,8 +108,7 @@ 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);
} }
@@ -148,39 +125,24 @@ 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 2000ms, with no window */ /* 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(2000) << 16) | WDT_GetPeriod(2000)); (WDT_GetPeriod(500) << 16) | WDT_GetPeriod(500));
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 "\n\r" "SIMtrace2 firmware " GIT_VERSION " (C) 2010-2016 by Harald Welte\n\r"
"(C) 2010-2017 by Harald Welte, 2018-2019 by Kevin Redon\n\r"
"=============================================================================\n\r"); "=============================================================================\n\r");
#if (TRACE_LEVEL >= TRACE_LEVEL_INFO) TRACE_INFO("Chip ID: 0x%08x (Ext 0x%08x)\n\r", CHIPID->CHIPID_CIDR, CHIPID->CHIPID_EXID);
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]);
uint8_t reset_cause = (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos; TRACE_INFO("Reset Cause: 0x%x\n\r", (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();
@@ -209,9 +171,7 @@ 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);
if (config_func_ptrs[simtrace_config].init) { 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");
@@ -239,17 +199,11 @@ 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);
if (config_func_ptrs[last_simtrace_config].exit) { config_func_ptrs[last_simtrace_config].exit();
config_func_ptrs[last_simtrace_config].exit(); 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;
} else { } else {
if (config_func_ptrs[simtrace_config].run) { config_func_ptrs[simtrace_config].run();
config_func_ptrs[simtrace_config].run();
}
} }
} }
} }

View File

@@ -1,8 +1,8 @@
sysmocom - s.f.m.c. GmbH sysmocom - s.f.m.c. GmbH
PRODUCT_STRING SIMtrace 2 compatible device
SIMtrace Sniffer SIMtrace Sniffer
SIMtrace CCID SIMtrace CCID
SIMtrace Card Emulation SIMtrace Phone
SIMtrace MITM SIMtrace MITM
CardEmulator Modem 1 CardEmulator Modem 1
CardEmulator Modem 2 CardEmulator Modem 2

View File

@@ -1,28 +1,8 @@
/* SIMtrace 2 firmware USB DFU bootloader
*
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
* (C) 2018-2019 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>
@@ -66,11 +46,7 @@ int USBDFU_handle_dnload(uint8_t altif, unsigned int offset,
WDT_Restart(WDT); WDT_Restart(WDT);
} }
#if TRACE_LEVEL >= TRACE_LEVEL_INFO printf("dnload(altif=%u, offset=%u, len=%u)\n\r", altif, offset, len);
TRACE_INFO("dnload(altif=%u, offset=%u, len=%u)\n\r", altif, offset, len);
#else
printf("DL off=%u\n\r", offset);
#endif
#ifdef PINS_LEDS #ifdef PINS_LEDS
PIO_Clear(&pinsLeds[LED_NUM_RED]); PIO_Clear(&pinsLeds[LED_NUM_RED]);
@@ -118,6 +94,7 @@ int USBDFU_handle_dnload(uint8_t altif, unsigned int offset,
rc = DFU_RET_ZLP; rc = DFU_RET_ZLP;
break; 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);
rc = DFU_RET_STALL; rc = DFU_RET_STALL;
break; break;
@@ -183,30 +160,22 @@ WEAK int board_override_enter_dfu(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 (board_override_enter_dfu()) { if (board_override_enter_dfu())
return 2; return 1;
}
/* 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) || ((uint8_t *)app_part[0] > IRAM_END)) { if ((app_part[0] < IRAM_ADDR) ||
return 3; ((uint8_t *)app_part[0] > IRAM_END))
} 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 4; return 1;
}
return 0; return 0;
} }
@@ -246,14 +215,17 @@ extern int main(void)
PIO_Clear(&pinsLeds[LED_NUM_GREEN]); PIO_Clear(&pinsLeds[LED_NUM_GREEN]);
#endif #endif
/* Enable watchdog for 500ms, with no window */
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
(WDT_GetPeriod(500) << 16) | WDT_GetPeriod(500));
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\n\r" "DFU bootloader %s for board %s (C) 2010-2017 by Harald Welte\n\r"
"(C) 2010-2017 by Harald Welte, 2018-2019 by Kevin Redon\n\r"
"=============================================================================\n\r", "=============================================================================\n\r",
manifest_revision, manifest_board); manifest_revision, manifest_board);
@@ -261,36 +233,7 @@ 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%lx\n\r", reset_cause); TRACE_INFO("Reset Cause: 0x%x\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)
@@ -300,17 +243,11 @@ extern int main(void)
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 */ /* Signal USB reset by disabling the pull-up on USB D+ for at least 10 ms */
USBD_Disconnect();
#ifdef PIN_USB_PULLUP
const Pin usb_dp_pullup = PIN_USB_PULLUP; const Pin usb_dp_pullup = PIN_USB_PULLUP;
PIO_Configure(&usb_dp_pullup, 1); PIO_Configure(&usb_dp_pullup, 1);
PIO_Set(&usb_dp_pullup); PIO_Set(&usb_dp_pullup);
#endif mdelay(15);
mdelay(50);
#ifdef PIN_USB_PULLUP
PIO_Clear(&usb_dp_pullup); PIO_Clear(&usb_dp_pullup);
#endif
USBDFU_Initialize(&dfu_descriptors); USBDFU_Initialize(&dfu_descriptors);
while (USBD_GetState() < USBD_STATE_CONFIGURED) { while (USBD_GetState() < USBD_STATE_CONFIGURED) {
@@ -318,8 +255,8 @@ extern int main(void)
check_exec_dbg_cmd(); check_exec_dbg_cmd();
#if 1 #if 1
if (i >= MAX_USB_ITER * 3) { if (i >= MAX_USB_ITER * 3) {
TRACE_ERROR("Resetting board (USB could not be configured)\n\r"); TRACE_ERROR("Resetting board (USB could "
g_dfu->magic = USB_DFU_MAGIC; // start the bootloader after reboot "not be configured)\n\r");
USBD_Disconnect(); USBD_Disconnect();
NVIC_SystemReset(); NVIC_SystemReset();
} }

View File

@@ -1,5 +1,5 @@
sysmocom - s.f.m.c. GmbH sysmocom - s.f.m.c. GmbH
PRODUCT_STRING SIMtrace 2 compatible device
DFU (Device Firmware Upgrade) DFU (Device Firmware Upgrade)
RAM RAM
Flash (Application Partition) Flash (Application Partition)

View File

@@ -1,3 +0,0 @@
C_FILES += $(C_LIBUSB_RT)
C_FILES += freq_ctr.c

View File

@@ -1,55 +0,0 @@
#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;
}

View File

@@ -1,54 +0,0 @@
#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();
}
}

View File

@@ -1,10 +0,0 @@
sysmocom - s.f.m.c. GmbH
PRODUCT_STRING
SIMtrace Sniffer
SIMtrace CCID
SIMtrace Card Emulation
SIMtrace MITM
CardEmulator Modem 1
CardEmulator Modem 2
CardEmulator Modem 3
CardEmulator Modem 4

View File

@@ -1,3 +0,0 @@
C_FILES += $(C_LIBUSB_RT)
C_FILES += gpio_test.c

View File

@@ -1,8 +0,0 @@
#include <stdint.h>
#include "utils.h"
#include "chip.h"
void gpio_test_init(void)
{
printf("FIXME run tests here\n\r");
}

View File

@@ -1,54 +0,0 @@
#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();
}
}

View File

@@ -1,10 +0,0 @@
sysmocom - s.f.m.c. GmbH
PRODUCT_STRING
SIMtrace Sniffer
SIMtrace CCID
SIMtrace Card Emulation
SIMtrace MITM
CardEmulator Modem 1
CardEmulator Modem 2
CardEmulator Modem 3
CardEmulator Modem 4

View File

@@ -1,21 +1,21 @@
/* SIMtrace 2 firmware sniffer application /*
* (C) 2010-2017 by Harald Welte <hwelte@sysmocom.de>
* (C) 2018 by Kevin Redon <kredon@sysmocom.de>
* All Rights Reserved
* *
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de> * This program is free software; you can redistribute it and/or modify
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de> * it under the terms of the GNU Affero General Public License as published by
* * the Free Software Foundation; either version 3 of the License, or
* This program is free software; you can redistribute it and/or * (at your option) any later version.
* 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, * This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of * but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
* *
* 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 * Headers
@@ -104,6 +104,9 @@ 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);
if (cfgnum >= sizeof(config_func_ptrs)/sizeof(config_func_ptrs[0])) {
TRACE_FATAL_WP("no functions defined for configuration %d\n\r", cfgnum);
}
simtrace_config = cfgnum; simtrace_config = cfgnum;
} }
@@ -142,15 +145,10 @@ extern int main(void)
enum confNum last_simtrace_config = simtrace_config; enum confNum last_simtrace_config = simtrace_config;
unsigned int i = 0; unsigned int i = 0;
/* Configure LED output /* Configure LED output (red = on, green = activity */
* red on = power
* red blink = error
* green on = running
* green blink = activity
*/
led_init(); led_init();
led_blink(LED_RED, BLINK_ALWAYS_ON); led_blink(LED_RED, BLINK_ALWAYS_ON);
led_blink(LED_GREEN, BLINK_ALWAYS_ON); led_blink(LED_GREEN, BLINK_ALWAYS_OFF);
/* Enable watchdog for 2000 ms, with no window */ /* Enable watchdog for 2000 ms, 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 |
@@ -160,22 +158,23 @@ extern int main(void)
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%08lx (Ext 0x%08lx)\n\r", CHIPID->CHIPID_CIDR, CHIPID->CHIPID_EXID); TRACE_INFO("Chip ID: 0x%08x (Ext 0x%08x)\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%lx\n\r", (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos); TRACE_INFO("Reset Cause: 0x%x\n\r", (RSTC->RSTC_SR & RSTC_SR_RSTTYP_Msk) >> RSTC_SR_RSTTYP_Pos);
TRACE_INFO("USB configuration used: %d\n\r", simtrace_config); TRACE_INFO("USB configuration used: %d\n\r", simtrace_config);
board_main_top(); board_main_top();
TRACE_INFO("USB init...\n\r"); TRACE_INFO("USB init...\n\r");
SIMtrace_USB_Initialize(); SIMtrace_USB_Initialize();
TRACE_INFO_WP("USBD_Inited\n\r");
while (USBD_GetState() < USBD_STATE_CONFIGURED) { while (USBD_GetState() < USBD_STATE_CONFIGURED) {
WDT_Restart(WDT); WDT_Restart(WDT);
@@ -192,13 +191,26 @@ extern int main(void)
} }
TRACE_INFO("calling configure of all configurations...\n\r"); TRACE_INFO("calling configure of all configurations...\n\r");
for (i = 1; i < ARRAY_SIZE(config_func_ptrs); i++) { for (i = 1; i < sizeof(config_func_ptrs) / sizeof(config_func_ptrs[0]); ++i) {
if (config_func_ptrs[i].configure) if (config_func_ptrs[i].configure) {
config_func_ptrs[i].configure(); config_func_ptrs[i].configure();
} else {
TRACE_WARNING("no configure function defined for configuration %d\n\r", i);
}
} }
TRACE_INFO("cfg %d\n\r", simtrace_config);
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 (simtrace_config >= sizeof(config_func_ptrs)/sizeof(config_func_ptrs[0])) {
TRACE_ERROR("no functions defined for configuration %d\n\r", simtrace_config);
} else {
if (config_func_ptrs[simtrace_config].init) {
config_func_ptrs[simtrace_config].init();
} else {
TRACE_ERROR("no init function defined for configuration %d\n\r", simtrace_config);
}
}
last_simtrace_config = simtrace_config; last_simtrace_config = simtrace_config;
TRACE_INFO("entering main loop...\n\r"); TRACE_INFO("entering main loop...\n\r");
@@ -226,11 +238,31 @@ 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 (last_simtrace_config < sizeof(config_func_ptrs)/sizeof(config_func_ptrs[0])) {
config_func_ptrs[simtrace_config].init(); if (config_func_ptrs[last_simtrace_config].exit) {
config_func_ptrs[last_simtrace_config].exit();
} else {
TRACE_WARNING("exit not defined for configuration %d\n\r", last_simtrace_config);
}
} else {
TRACE_ERROR("no functions defined for configuration %d\n\r", last_simtrace_config);
}
if (simtrace_config < sizeof(config_func_ptrs)/sizeof(config_func_ptrs[0])) {
if (config_func_ptrs[simtrace_config].init) {
config_func_ptrs[simtrace_config].init();
} else {
TRACE_WARNING("init not defined for configuration %d\n\r", simtrace_config);
}
} else {
TRACE_FATAL("no functions defined for configuration %d\n\r", simtrace_config);
}
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();
} else {
TRACE_ERROR("run not defined for configuration %d\n\r", simtrace_config);
}
} }
} }
} }

View File

@@ -1,8 +1,8 @@
sysmocom - s.f.m.c. GmbH sysmocom - s.f.m.c. GmbH
PRODUCT_STRING SIMtrace 2 compatible device
SIMtrace Sniffer SIMtrace Sniffer
SIMtrace CCID SIMtrace CCID
SIMtrace Card Emulation SIMtrace Phone
SIMtrace MITM SIMtrace MITM
CardEmulator Modem 1 CardEmulator Modem 1
CardEmulator Modem 2 CardEmulator Modem 2

View File

@@ -1,21 +1,4 @@
/* SIMtrace 2 firmware card emulation, CCID, and sniffer application // FIXME: Copyright license here
*
* (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
*------------------------------------------------------------------------------*/ *------------------------------------------------------------------------------*/
@@ -151,7 +134,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");

View File

@@ -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_ */

View File

@@ -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_ */

View File

@@ -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

View File

@@ -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 */
} }

View File

@@ -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;
} }

View File

@@ -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)) ) ;
} }
} }

View File

@@ -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 ;
} }

View File

@@ -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 ;
} }

View File

@@ -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;
} }

View File

@@ -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) ;
} }

View File

@@ -251,7 +251,7 @@ static void GetDescriptor(
switch (type) { switch (type) {
case USBGenericDescriptor_DEVICE: case USBGenericDescriptor_DEVICE:
TRACE_DEBUG_WP("Dev "); TRACE_INFO_WP("Dev ");
/* Adjust length and send descriptor */ /* Adjust length and send descriptor */
@@ -263,7 +263,7 @@ static void GetDescriptor(
break; break;
case USBGenericDescriptor_CONFIGURATION: case USBGenericDescriptor_CONFIGURATION:
TRACE_DEBUG_WP("Cfg "); TRACE_INFO_WP("Cfg ");
/* Adjust length and send descriptor */ /* Adjust length and send descriptor */
@@ -280,7 +280,7 @@ static void GetDescriptor(
break; break;
case USBGenericDescriptor_DEVICEQUALIFIER: case USBGenericDescriptor_DEVICEQUALIFIER:
TRACE_DEBUG_WP("Qua "); TRACE_INFO_WP("Qua ");
/* Check if descriptor exists */ /* Check if descriptor exists */
@@ -301,7 +301,7 @@ static void GetDescriptor(
break; break;
case USBGenericDescriptor_OTHERSPEEDCONFIGURATION: case USBGenericDescriptor_OTHERSPEEDCONFIGURATION:
TRACE_DEBUG_WP("OSC "); TRACE_INFO_WP("OSC ");
/* Check if descriptor exists */ /* Check if descriptor exists */
@@ -327,7 +327,7 @@ static void GetDescriptor(
break; break;
case USBGenericDescriptor_STRING: case USBGenericDescriptor_STRING:
TRACE_DEBUG_WP("Str%d ", indexRDesc); TRACE_INFO_WP("Str%d ", indexRDesc);
/* Check if descriptor exists */ /* Check if descriptor exists */
@@ -504,13 +504,13 @@ void USBDDriver_RequestHandler(
uint32_t length; uint32_t length;
uint32_t address; uint32_t address;
TRACE_DEBUG_WP("Std "); TRACE_INFO_WP("Std ");
/* Check request code */ /* Check request code */
switch (USBGenericRequest_GetRequest(pRequest)) { switch (USBGenericRequest_GetRequest(pRequest)) {
case USBGenericRequest_GETDESCRIPTOR: case USBGenericRequest_GETDESCRIPTOR:
TRACE_DEBUG_WP("gDesc "); TRACE_INFO_WP("gDesc ");
/* Send the requested descriptor */ /* Send the requested descriptor */
type = USBGetDescriptorRequest_GetDescriptorType(pRequest); type = USBGetDescriptorRequest_GetDescriptorType(pRequest);
@@ -520,7 +520,7 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_SETADDRESS: case USBGenericRequest_SETADDRESS:
TRACE_DEBUG_WP("sAddr "); TRACE_INFO_WP("sAddr ");
/* Sends a zero-length packet and then set the device address */ /* Sends a zero-length packet and then set the device address */
address = USBSetAddressRequest_GetAddress(pRequest); address = USBSetAddressRequest_GetAddress(pRequest);
@@ -528,7 +528,7 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_SETCONFIGURATION: case USBGenericRequest_SETCONFIGURATION:
TRACE_DEBUG_WP("sCfg "); TRACE_INFO_WP("sCfg ");
/* Set the requested configuration */ /* Set the requested configuration */
cfgnum = USBSetConfigurationRequest_GetConfiguration(pRequest); cfgnum = USBSetConfigurationRequest_GetConfiguration(pRequest);
@@ -536,27 +536,27 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_GETCONFIGURATION: case USBGenericRequest_GETCONFIGURATION:
TRACE_DEBUG_WP("gCfg "); TRACE_INFO_WP("gCfg ");
/* Send the current configuration number */ /* Send the current configuration number */
GetConfiguration(pDriver); GetConfiguration(pDriver);
break; break;
case USBGenericRequest_GETSTATUS: case USBGenericRequest_GETSTATUS:
TRACE_DEBUG_WP("gSta "); TRACE_INFO_WP("gSta ");
/* Check who is the recipient */ /* Check who is the recipient */
switch (USBGenericRequest_GetRecipient(pRequest)) { switch (USBGenericRequest_GetRecipient(pRequest)) {
case USBGenericRequest_DEVICE: case USBGenericRequest_DEVICE:
TRACE_DEBUG_WP("Dev "); TRACE_INFO_WP("Dev ");
/* Send the device status */ /* Send the device status */
GetDeviceStatus(pDriver); GetDeviceStatus(pDriver);
break; break;
case USBGenericRequest_ENDPOINT: case USBGenericRequest_ENDPOINT:
TRACE_DEBUG_WP("Ept "); TRACE_INFO_WP("Ept ");
/* Send the endpoint status */ /* Send the endpoint status */
eptnum = USBGenericRequest_GetEndpointNumber(pRequest); eptnum = USBGenericRequest_GetEndpointNumber(pRequest);
@@ -572,13 +572,13 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_CLEARFEATURE: case USBGenericRequest_CLEARFEATURE:
TRACE_DEBUG_WP("cFeat "); TRACE_INFO_WP("cFeat ");
/* Check which is the requested feature */ /* Check which is the requested feature */
switch (USBFeatureRequest_GetFeatureSelector(pRequest)) { switch (USBFeatureRequest_GetFeatureSelector(pRequest)) {
case USBFeatureRequest_ENDPOINTHALT: case USBFeatureRequest_ENDPOINTHALT:
TRACE_DEBUG_WP("Hlt "); TRACE_INFO_WP("Hlt ");
/* Unhalt endpoint and send a zero-length packet */ /* Unhalt endpoint and send a zero-length packet */
USBD_Unhalt(USBGenericRequest_GetEndpointNumber(pRequest)); USBD_Unhalt(USBGenericRequest_GetEndpointNumber(pRequest));
@@ -586,7 +586,7 @@ void USBDDriver_RequestHandler(
break; break;
case USBFeatureRequest_DEVICEREMOTEWAKEUP: case USBFeatureRequest_DEVICEREMOTEWAKEUP:
TRACE_DEBUG_WP("RmWU "); TRACE_INFO_WP("RmWU ");
/* Disable remote wake-up and send a zero-length packet */ /* Disable remote wake-up and send a zero-length packet */
pDriver->isRemoteWakeUpEnabled = 0; pDriver->isRemoteWakeUpEnabled = 0;
@@ -602,13 +602,13 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_SETFEATURE: case USBGenericRequest_SETFEATURE:
TRACE_DEBUG_WP("sFeat "); TRACE_INFO_WP("sFeat ");
/* Check which is the selected feature */ /* Check which is the selected feature */
switch (USBFeatureRequest_GetFeatureSelector(pRequest)) { switch (USBFeatureRequest_GetFeatureSelector(pRequest)) {
case USBFeatureRequest_DEVICEREMOTEWAKEUP: case USBFeatureRequest_DEVICEREMOTEWAKEUP:
TRACE_DEBUG_WP("RmWU "); TRACE_INFO_WP("RmWU ");
/* Enable remote wake-up and send a ZLP */ /* Enable remote wake-up and send a ZLP */
pDriver->isRemoteWakeUpEnabled = 1; pDriver->isRemoteWakeUpEnabled = 1;
@@ -616,25 +616,25 @@ void USBDDriver_RequestHandler(
break; break;
case USBFeatureRequest_ENDPOINTHALT: case USBFeatureRequest_ENDPOINTHALT:
TRACE_DEBUG_WP("Halt "); TRACE_INFO_WP("Halt ");
/* Halt endpoint */ /* Halt endpoint */
USBD_Halt(USBGenericRequest_GetEndpointNumber(pRequest)); USBD_Halt(USBGenericRequest_GetEndpointNumber(pRequest));
USBD_Write(0, 0, 0, 0, 0); USBD_Write(0, 0, 0, 0, 0);
break; break;
case USBFeatureRequest_OTG_B_HNP_ENABLE: case USBFeatureRequest_OTG_B_HNP_ENABLE:
TRACE_DEBUG_WP("OTG_B_HNP_ENABLE "); TRACE_INFO_WP("OTG_B_HNP_ENABLE ");
pDriver->otg_features_supported |= pDriver->otg_features_supported |=
1<<USBFeatureRequest_OTG_B_HNP_ENABLE; 1<<USBFeatureRequest_OTG_B_HNP_ENABLE;
USBD_Write(0, 0, 0, 0, 0); USBD_Write(0, 0, 0, 0, 0);
break; break;
case USBFeatureRequest_OTG_A_HNP_SUPPORT: case USBFeatureRequest_OTG_A_HNP_SUPPORT:
TRACE_DEBUG_WP("OTG_A_HNP_SUPPORT "); TRACE_INFO_WP("OTG_A_HNP_SUPPORT ");
pDriver->otg_features_supported |= pDriver->otg_features_supported |=
1<<USBFeatureRequest_OTG_A_HNP_SUPPORT; 1<<USBFeatureRequest_OTG_A_HNP_SUPPORT;
USBD_Write(0, 0, 0, 0, 0); USBD_Write(0, 0, 0, 0, 0);
break; break;
case USBFeatureRequest_OTG_A_ALT_HNP_SUPPORT: case USBFeatureRequest_OTG_A_ALT_HNP_SUPPORT:
TRACE_DEBUG_WP("OTG_A_ALT_HNP_SUPPORT "); TRACE_INFO_WP("OTG_A_ALT_HNP_SUPPORT ");
pDriver->otg_features_supported |= pDriver->otg_features_supported |=
1<<USBFeatureRequest_OTG_A_ALT_HNP_SUPPORT; 1<<USBFeatureRequest_OTG_A_ALT_HNP_SUPPORT;
USBD_Write(0, 0, 0, 0, 0); USBD_Write(0, 0, 0, 0, 0);
@@ -649,7 +649,7 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_SETINTERFACE: case USBGenericRequest_SETINTERFACE:
TRACE_DEBUG_WP("sInterface "); TRACE_INFO_WP("sInterface ");
infnum = USBInterfaceRequest_GetInterface(pRequest); infnum = USBInterfaceRequest_GetInterface(pRequest);
setting = USBInterfaceRequest_GetAlternateSetting(pRequest); setting = USBInterfaceRequest_GetAlternateSetting(pRequest);
@@ -657,7 +657,7 @@ void USBDDriver_RequestHandler(
break; break;
case USBGenericRequest_GETINTERFACE: case USBGenericRequest_GETINTERFACE:
TRACE_DEBUG_WP("gInterface "); TRACE_INFO_WP("gInterface ");
infnum = USBInterfaceRequest_GetInterface(pRequest); infnum = USBInterfaceRequest_GetInterface(pRequest);
GetInterface(pDriver, infnum); GetInterface(pDriver, infnum);

View File

@@ -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(void); void set_usb_serial_str(const uint8_t *serial_usbstr);
void DFURT_SwitchToDFU(void); void DFURT_SwitchToDFU(void);

View File

@@ -13,87 +13,14 @@
#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,
@@ -107,8 +34,12 @@ static const USBDeviceDescriptor fsDevice = {
.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,
.bNumConfigurations = 2, // DFU + version configurations #else
.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 */
@@ -154,74 +85,17 @@ const struct dfu_desc dfu_cfg_descriptor = {
.func_dfu = DFU_FUNC_DESC .func_dfu = DFU_FUNC_DESC
}; };
void set_usb_serial_str(void) #include "usb_strings_generated.h"
#if 0
void set_usb_serial_str(const uint8_t *serial_usbstr)
{ {
unsigned int i; usb_strings[STR_SERIAL] = serial_usbstr;
// 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 = {
@@ -234,6 +108,6 @@ const USBDDriverDescriptors dfu_descriptors = {
.pHsConfiguration = NULL, .pHsConfiguration = NULL,
.pHsQualifier = NULL, .pHsQualifier = NULL,
.pHsOtherSpeed = NULL, .pHsOtherSpeed = NULL,
.pStrings = usb_strings_extended, .pStrings = usb_strings,
.numStrings = ARRAY_SIZE(usb_strings_extended), .numStrings = ARRAY_SIZE(usb_strings),
}; };

View File

@@ -447,7 +447,6 @@ 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();

View File

@@ -208,7 +208,7 @@ void DFURT_SwitchToDFU(void)
* activate itself, rather than boot into the application */ * activate itself, rather than boot into the application */
g_dfu->magic = USB_DFU_MAGIC; g_dfu->magic = USB_DFU_MAGIC;
/* Disconnect the USB by removing the pull-up */ /* Disconnect the USB by remoting the pull-up */
USBD_Disconnect(); USBD_Disconnect();
__disable_irq(); __disable_irq();

View File

@@ -1,22 +1,3 @@
/* 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_
@@ -56,12 +37,18 @@
/** Core definition */ /** Core definition */
#define cortexm3 #define cortexm3
/* LEDs are used to indicate the status #define BOARD_MCK 48000000
* the LED definition is board specific
* most boards have two LEDs, one green and one red #define PIO_LED_RED PIO_PA17
* the red LED indicates of the main firmware is ready (on) or if there is an error (blinking) #define PIO_LED_GREEN PIO_PA18
* 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 */
@@ -76,8 +63,8 @@
#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 baud rate in bps */ /** Console baudrate always using 115200. */
#define CONSOLE_BAUDRATE 921600 #define CONSOLE_BAUDRATE 115200
/** UART peripheral used by the console (UART0). */ /** UART peripheral used by the console (UART0). */
#define CONSOLE_UART UART0 #define CONSOLE_UART UART0
/** UART peripheral ID used by the console (UART0). */ /** UART peripheral ID used by the console (UART0). */
@@ -108,8 +95,18 @@
/* Interrupt request ID of USART peripheral connected to the phone */ /* Interrupt request ID of USART peripheral connected to the phone */
#define IRQ_USART_PHONE USART1_IRQn #define IRQ_USART_PHONE USART1_IRQn
#define SIM_PWEN PIO_PA5
#define VCC_FWD PIO_PA26
/** 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}
// 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_DFU #define BOARD_USB_DFU
#define BOARD_DFU_BOOT_SIZE (16 * 1024) #define BOARD_DFU_BOOT_SIZE (16 * 1024)

View File

@@ -26,6 +26,7 @@
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
* ---------------------------------------------------------------------------- * ----------------------------------------------------------------------------
*/ */
/** /**
* \file * \file
* *

View File

@@ -1,18 +1,4 @@
/* 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);
uint32_t adc2uv(uint16_t adc); uint32_t adc2uv(uint16_t adc);
int get_board_version_adc(void);

View File

@@ -1,17 +1,3 @@
/* 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 {
@@ -31,7 +17,6 @@ enum led_pattern {
BLINK_200O_F = 7, BLINK_200O_F = 7,
BLINK_600O_F = 8, BLINK_600O_F = 8,
BLINK_CUSTOM = 9, BLINK_CUSTOM = 9,
BLINK_2F_O,
_NUM_LED_BLINK _NUM_LED_BLINK
}; };

View File

@@ -1,17 +1,4 @@
/* 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

View File

@@ -1,17 +1,3 @@
/* 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);

View File

@@ -2,7 +2,6 @@
* 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.
* *
@@ -27,17 +26,19 @@
* 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 ) ;

View File

@@ -1,208 +1,215 @@
/* ---------------------------------------------------------------------------- /* ----------------------------------------------------------------------------
* 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> * Copyright (C) 2017, Harald Welte <laforge@gnumonks.org>
* 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
* 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 "board.h"
#include "board.h" #include "board_lowlevel.h"
#include "board_lowlevel.h"
/*----------------------------------------------------------------------------
/*---------------------------------------------------------------------------- * Exported variables
* Exported variables *----------------------------------------------------------------------------*/
*----------------------------------------------------------------------------*/
/* Stack Configuration */
/* Stack Configuration */ #define STACK_SIZE 0x900 /** Stack size (in DWords) */
#define STACK_SIZE 0x900 /** Stack size (in DWords) */ __attribute__ ((aligned(8),section(".stack")))
__attribute__ ((aligned(8),section(".stack"))) uint32_t pdwStack[STACK_SIZE] ;
uint32_t pdwStack[STACK_SIZE] ;
/* Initialize segments */
/* Initialize segments */ extern uint32_t _sfixed;
extern uint32_t _sfixed; extern uint32_t _efixed;
extern uint32_t _efixed; extern uint32_t _etext;
extern uint32_t _etext; extern uint32_t _srelocate;
extern uint32_t _srelocate; extern uint32_t _erelocate;
extern uint32_t _erelocate; extern uint32_t _szero;
extern uint32_t _szero; extern uint32_t _ezero;
extern uint32_t _ezero;
/*----------------------------------------------------------------------------
/*---------------------------------------------------------------------------- * ProtoTypes
* ProtoTypes *----------------------------------------------------------------------------*/
*----------------------------------------------------------------------------*/
/** \cond DOXYGEN_SHOULD_SKIP_THIS */
/** \cond DOXYGEN_SHOULD_SKIP_THIS */ extern int main( void ) ;
extern int main( void ) ; /** \endcond */
/** \endcond */ void ResetException( void ) ;
void ResetException( void ) ;
/*------------------------------------------------------------------------------
/*------------------------------------------------------------------------------ * Exception Table
* Exception Table *------------------------------------------------------------------------------*/
*------------------------------------------------------------------------------*/
__attribute__((section(".vectors")))
__attribute__((section(".vectors"))) IntFunc exception_table[] = {
IntFunc exception_table[] = {
/* Configure Initial Stack Pointer, using linker-generated symbols */
/* Configure Initial Stack Pointer, using linker-generated symbols */ (IntFunc)(&pdwStack[STACK_SIZE-1]),
(IntFunc)(&pdwStack[STACK_SIZE-1]), ResetException,
ResetException,
NMI_Handler,
NMI_Handler, HardFault_Handler,
HardFault_Handler, MemManage_Handler,
MemManage_Handler, BusFault_Handler,
BusFault_Handler, UsageFault_Handler,
UsageFault_Handler, 0, 0, 0, 0, /* Reserved */
0, 0, 0, 0, /* Reserved */ SVC_Handler,
SVC_Handler, DebugMon_Handler,
DebugMon_Handler, 0, /* Reserved */
0, /* Reserved */ PendSV_Handler,
PendSV_Handler, SysTick_Handler,
SysTick_Handler,
/* Configurable interrupts */
/* Configurable interrupts */ SUPC_IrqHandler, /* 0 Supply Controller */
SUPC_IrqHandler, /* 0 Supply Controller */ RSTC_IrqHandler, /* 1 Reset Controller */
RSTC_IrqHandler, /* 1 Reset Controller */ RTC_IrqHandler, /* 2 Real Time Clock */
RTC_IrqHandler, /* 2 Real Time Clock */ RTT_IrqHandler, /* 3 Real Time Timer */
RTT_IrqHandler, /* 3 Real Time Timer */ WDT_IrqHandler, /* 4 Watchdog Timer */
WDT_IrqHandler, /* 4 Watchdog Timer */ PMC_IrqHandler, /* 5 PMC */
PMC_IrqHandler, /* 5 PMC */ EEFC_IrqHandler, /* 6 EEFC */
EEFC_IrqHandler, /* 6 EEFC */ IrqHandlerNotUsed, /* 7 Reserved */
IrqHandlerNotUsed, /* 7 Reserved */ UART0_IrqHandler, /* 8 UART0 */
UART0_IrqHandler, /* 8 UART0 */ UART1_IrqHandler, /* 9 UART1 */
UART1_IrqHandler, /* 9 UART1 */ SMC_IrqHandler, /* 10 SMC */
SMC_IrqHandler, /* 10 SMC */ PIOA_IrqHandler, /* 11 Parallel IO Controller A */
PIOA_IrqHandler, /* 11 Parallel IO Controller A */ PIOB_IrqHandler, /* 12 Parallel IO Controller B */
PIOB_IrqHandler, /* 12 Parallel IO Controller B */ PIOC_IrqHandler, /* 13 Parallel IO Controller C */
PIOC_IrqHandler, /* 13 Parallel IO Controller C */ USART0_IrqHandler, /* 14 USART 0 */
USART0_IrqHandler, /* 14 USART 0 */ USART1_IrqHandler, /* 15 USART 1 */
USART1_IrqHandler, /* 15 USART 1 */ IrqHandlerNotUsed, /* 16 Reserved */
IrqHandlerNotUsed, /* 16 Reserved */ IrqHandlerNotUsed, /* 17 Reserved */
IrqHandlerNotUsed, /* 17 Reserved */ MCI_IrqHandler, /* 18 MCI */
MCI_IrqHandler, /* 18 MCI */ TWI0_IrqHandler, /* 19 TWI 0 */
TWI0_IrqHandler, /* 19 TWI 0 */ TWI1_IrqHandler, /* 20 TWI 1 */
TWI1_IrqHandler, /* 20 TWI 1 */ SPI_IrqHandler, /* 21 SPI */
SPI_IrqHandler, /* 21 SPI */ SSC_IrqHandler, /* 22 SSC */
SSC_IrqHandler, /* 22 SSC */ TC0_IrqHandler, /* 23 Timer Counter 0 */
TC0_IrqHandler, /* 23 Timer Counter 0 */ TC1_IrqHandler, /* 24 Timer Counter 1 */
TC1_IrqHandler, /* 24 Timer Counter 1 */ TC2_IrqHandler, /* 25 Timer Counter 2 */
TC2_IrqHandler, /* 25 Timer Counter 2 */ TC3_IrqHandler, /* 26 Timer Counter 3 */
TC3_IrqHandler, /* 26 Timer Counter 3 */ TC4_IrqHandler, /* 27 Timer Counter 4 */
TC4_IrqHandler, /* 27 Timer Counter 4 */ TC5_IrqHandler, /* 28 Timer Counter 5 */
TC5_IrqHandler, /* 28 Timer Counter 5 */ ADC_IrqHandler, /* 29 ADC controller */
ADC_IrqHandler, /* 29 ADC controller */ DAC_IrqHandler, /* 30 DAC controller */
DAC_IrqHandler, /* 30 DAC controller */ PWM_IrqHandler, /* 31 PWM */
PWM_IrqHandler, /* 31 PWM */ CRCCU_IrqHandler, /* 32 CRC Calculation Unit */
CRCCU_IrqHandler, /* 32 CRC Calculation Unit */ ACC_IrqHandler, /* 33 Analog Comparator */
ACC_IrqHandler, /* 33 Analog Comparator */ USBD_IrqHandler, /* 34 USB Device Port */
USBD_IrqHandler, /* 34 USB Device Port */ IrqHandlerNotUsed /* 35 not used */
IrqHandlerNotUsed /* 35 not used */ };
};
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu)
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu) #include "usb/device/dfu/dfu.h"
#include "usb/device/dfu/dfu.h" static void BootIntoApp(void)
static void BootIntoApp(void) {
{ unsigned int *pSrc;
unsigned int *pSrc; void (*appReset)(void);
void (*appReset)(void);
pSrc = (unsigned int *) ((unsigned char *)IFLASH_ADDR + BOARD_DFU_BOOT_SIZE);
pSrc = (unsigned int *) ((unsigned char *)IFLASH_ADDR + BOARD_DFU_BOOT_SIZE); SCB->VTOR = ((unsigned int)(pSrc)) | (0x0 << 7);
/* set vector table to application vector table (store at the beginning of the application) */ appReset = (void(*)(void))pSrc[1];
SCB->VTOR = (unsigned int)(pSrc);
/* set stack pointer to address provided in the beginning of the application (loaded into a register first) */ g_dfu->state = DFU_STATE_appIDLE;
__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) */ appReset();
appReset = (void(*)(void))pSrc[1]; }
#endif
g_dfu->state = DFU_STATE_appIDLE;
/**
appReset(); * \brief This is the code that gets called on processor reset.
} * To initialize the device, and call the main() routine.
#endif */
void ResetException( void )
/** {
* \brief This is the code that gets called on processor reset. uint32_t *pSrc, *pDest ;
* To initialize the device, and call the main() routine.
*/ /* Low level Initialize */
void ResetException( void ) LowLevelInit() ;
{
uint32_t *pSrc, *pDest ;
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu)
/* Low level Initialize */ /* we are before the text segment has been relocated, so g_dfu is
LowLevelInit() ; * not initialized yet */
g_dfu = &_g_dfu;
if ((g_dfu->magic != USB_DFU_MAGIC) && !USBDFU_OverrideEnterDFU()) {
#if defined(BOARD_USB_DFU) && defined(APPLICATION_dfu) /* start application if valid
if (!USBDFU_OverrideEnterDFU()) { * the application starts with the vector table
UART_Exit(); * the first entry in the vector table is the initial stack pointer (SP) address
__disable_irq(); * the stack will be placed in RAM, which begins at 0x2000 0000
BootIntoApp(); * there is up to 48 KB of RAM (0xc000)
/* Infinite loop */ * since the stack grown "downwards" it should start at the end of the RAM: max 0x2000 c000
while ( 1 ) ; * if the SP is not in this range (e.g. flash has been erased) there is no valid application
} * the second entry in the vector table is the reset address, corresponding to the application start
#endif */
if (((*((uint32_t*)(IFLASH_ADDR+BOARD_DFU_BOOT_SIZE)))&0xFFFF0000)==0x20000000) {
/* Initialize the relocate segment */ BootIntoApp();
pSrc = &_etext ; /* Infinite loop */
pDest = &_srelocate ; while ( 1 ) ;
}
if ( pSrc != pDest ) }
{ #endif
for ( ; pDest < &_erelocate ; )
{ /* Initialize the relocate segment */
*pDest++ = *pSrc++ ; pSrc = &_etext ;
} pDest = &_srelocate ;
}
if ( pSrc != pDest )
/* Clear the zero segment */ {
for ( pDest = &_szero ; pDest < &_ezero ; ) for ( ; pDest < &_erelocate ; )
{ {
*pDest++ = 0; *pDest++ = *pSrc++ ;
} }
}
/* Set the vector table base address */
pSrc = (uint32_t *)&_sfixed; /* Clear the zero segment */
SCB->VTOR = ( (uint32_t)pSrc & SCB_VTOR_TBLOFF_Msk ) ; for ( pDest = &_szero ; pDest < &_ezero ; )
{
if ( ((uint32_t)pSrc >= IRAM_ADDR) && ((uint32_t)pSrc < IRAM_ADDR+IRAM_SIZE) ) *pDest++ = 0;
{ }
SCB->VTOR |= 1 << SCB_VTOR_TBLBASE_Pos ;
} /* Set the vector table base address */
pSrc = (uint32_t *)&_sfixed;
/* App should have disabled interrupts during the transition */ SCB->VTOR = ( (uint32_t)pSrc & SCB_VTOR_TBLOFF_Msk ) ;
__enable_irq();
if ( ((uint32_t)pSrc >= IRAM_ADDR) && ((uint32_t)pSrc < IRAM_ADDR+IRAM_SIZE) )
/* Branch to main function */ {
main() ; SCB->VTOR |= 1 << SCB_VTOR_TBLBASE_Pos ;
}
/* Infinite loop */
while ( 1 ) ; /* App should have disabled interrupts during the transition */
} __enable_irq();
/* Branch to main function */
main() ;
/* Infinite loop */
while ( 1 ) ;
}

View File

@@ -1,220 +1,216 @@
/* ---------------------------------------------------------------------------- /* ----------------------------------------------------------------------------
* 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
* 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 *
* * 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_OSCOUNT (CKGR_MOR_MOSCXTST(0x8)) #define BOARD_MCKR (PMC_MCKR_PRES_CLK | PMC_MCKR_CSS_PLLA_CLK)
#define BOARD_MCKR (PMC_MCKR_PRES_CLK | PMC_MCKR_CSS_PLLA_CLK)
#if (BOARD_MCK == 48000000)
/** configure PLL to generate main clock based on main oscillator frequency */ #if (BOARD_MAINOSC == 18432000)
#if (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 48000000) /* Clock settings at 48MHz for 18 MHz crystal */
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \ #define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
| CKGR_PLLAR_MULA(8-1) \ | CKGR_PLLAR_MULA(13-1) \
| CKGR_PLLAR_PLLACOUNT(0x1) \ | CKGR_PLLAR_PLLACOUNT(0x1) \
| CKGR_PLLAR_DIVA(2)) | CKGR_PLLAR_DIVA(5))
#elif (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 58000000) #elif (BOARD_MAINOSC == 12000000)
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \ /* QMod has 12 MHz clock, so multply by 8 (96 MHz) and divide by 2 */
| CKGR_PLLAR_MULA(29-1) \ #define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
| CKGR_PLLAR_PLLACOUNT(0x1) \ | CKGR_PLLAR_MULA(8-1) \
| CKGR_PLLAR_DIVA(6)) | CKGR_PLLAR_PLLACOUNT(0x1) \
#elif (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 60000000) | CKGR_PLLAR_DIVA(2))
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \ #else
| CKGR_PLLAR_MULA(10-1) \ #error "Please define PLLA config for your MAINOSC frequency"
| CKGR_PLLAR_PLLACOUNT(0x1) \ #endif /* MAINOSC */
| CKGR_PLLAR_DIVA(2)) #elif (BOARD_MCK == 64000000)
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 47923200) #if (BOARD_MAINOSC == 18432000)
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \ /* Clock settings at 64MHz for 18 MHz crystal: 64.512 MHz */
| CKGR_PLLAR_MULA(13-1) \ #define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
| CKGR_PLLAR_PLLACOUNT(0x1) \ | CKGR_PLLAR_MULA(7-1) \
| CKGR_PLLAR_DIVA(5)) | CKGR_PLLAR_PLLACOUNT(0x1) \
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 58982400) | CKGR_PLLAR_DIVA(2))
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \ #elif (BOARD_MAINOSC == 12000000)
| CKGR_PLLAR_MULA(16-1) \ /* QMod has 12 MHz clock, so multply by 10 / div by 2: 60 MHz */
| CKGR_PLLAR_PLLACOUNT(0x1) \ #define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
| CKGR_PLLAR_DIVA(5)) | CKGR_PLLAR_MULA(10-1) \
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 64512000) | CKGR_PLLAR_PLLACOUNT(0x1) \
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \ | CKGR_PLLAR_DIVA(2))
| CKGR_PLLAR_MULA(7-1) \ #error "Please define PLLA config for your MAINOSC frequency"
| CKGR_PLLAR_PLLACOUNT(0x1) \ #endif /* MAINOSC */
| CKGR_PLLAR_DIVA(2)) #else
#else #error "No PLL settings for current BOARD_MCK."
#error "Please define PLLA config for your BOARD_MCK/MAINOSC frequency" #endif
#endif
#if (BOARD_MAINOSC == 12000000)
#if (BOARD_MAINOSC == 12000000) #define PLLB_CFG (CKGR_PLLBR_DIVB(2)|CKGR_PLLBR_MULB(8-1)|CKGR_PLLBR_PLLBCOUNT_Msk)
#define PLLB_CFG (CKGR_PLLBR_DIVB(2)|CKGR_PLLBR_MULB(8-1)|CKGR_PLLBR_PLLBCOUNT_Msk) #elif (BOARD_MAINOSC == 18432000)
#elif (BOARD_MAINOSC == 18432000) #define PLLB_CFG (CKGR_PLLBR_DIVB(5)|CKGR_PLLBR_MULB(13-1)|CKGR_PLLBR_PLLBCOUNT_Msk)
#define PLLB_CFG (CKGR_PLLBR_DIVB(5)|CKGR_PLLBR_MULB(13-1)|CKGR_PLLBR_PLLBCOUNT_Msk) #else
#else #error "Please configure PLLB for your MAINOSC freq"
#error "Please configure PLLB for your MAINOSC freq" #endif
#endif
/* Define clock timeout */
/* Define clock timeout */ #define CLOCK_TIMEOUT 0xFFFFFFFF
#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 */
/* Enable PLLB for USB */ PMC->CKGR_PLLBR = PLLB_CFG;
PMC->CKGR_PLLBR = PLLB_CFG; while ((PMC->PMC_SR & PMC_SR_LOCKB) == 0) ;
while ((PMC->PMC_SR & PMC_SR_LOCKB) == 0) ;
/* USB Clock uses PLLB */
/* USB Clock uses PLLB */ PMC->PMC_USB = PMC_USB_USBDIV(0) /* /1 (no divider) */
PMC->PMC_USB = PMC_USB_USBDIV(0) /* /1 (no divider) */ | PMC_USB_USBS; /* PLLB */
| PMC_USB_USBS; /* PLLB */ }
}
/*----------------------------------------------------------------------------
/*---------------------------------------------------------------------------- * Exported functions
* Exported functions *----------------------------------------------------------------------------*/
*----------------------------------------------------------------------------*/
/**
/** * \brief Performs the low-level initialization of the chip.
* \brief Performs the low-level initialization of the chip. * This includes EFC and master clock configuration.
* This includes EFC and master clock configuration. * It also enable a low level on the pin NRST triggers a user reset.
* 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
/* 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
* lower than 3.0V. As we run the board on 3.3V, any lower voltage * might be some kind of leakage that creeps in some way, but is not
* might be some kind of leakage that creeps in some way, but is not * the "official" power supply */
* the "official" power supply */ SUPC->SUPC_SMMR = SUPC_SMMR_SMTH_3_0V | SUPC_SMMR_SMSMPL_CSM |
SUPC->SUPC_SMMR = SUPC_SMMR_SMTH_3_0V | SUPC_SMMR_SMSMPL_CSM | SUPC_SMMR_SMRSTEN_ENABLE;
SUPC_SMMR_SMRSTEN_ENABLE;
/* enable both LED and green LED */
/* disable ERASE pin to prevent accidental flash erase */ PIOA->PIO_PER |= PIO_LED_RED | PIO_LED_GREEN;
MATRIX->CCFG_SYSIO |= CCFG_SYSIO_SYSIO12; PIOA->PIO_OER |= PIO_LED_RED | PIO_LED_GREEN;
PIOA->PIO_CODR |= PIO_LED_RED | PIO_LED_GREEN;
/* enable both LED and green LED */
PIOA->PIO_PER |= PIO_LED_RED | PIO_LED_GREEN; /* Set 3 FWS for Embedded Flash Access */
PIOA->PIO_OER |= PIO_LED_RED | PIO_LED_GREEN; EFC->EEFC_FMR = EEFC_FMR_FWS(3);
PIOA->PIO_CODR |= PIO_LED_RED | PIO_LED_GREEN;
/* Select external slow clock */
/* Set 3 FWS for Embedded Flash Access */ /* if ((SUPC->SUPC_SR & SUPC_SR_OSCSEL) != SUPC_SR_OSCSEL_CRYST)
EFC->EEFC_FMR = EEFC_FMR_FWS(3); {
SUPC->SUPC_CR = (uint32_t)(SUPC_CR_XTALSEL_CRYSTAL_SEL | SUPC_CR_KEY(0xA5));
/* Select external slow clock */ timeout = 0;
/* if ((SUPC->SUPC_SR & SUPC_SR_OSCSEL) != SUPC_SR_OSCSEL_CRYST) while (!(SUPC->SUPC_SR & SUPC_SR_OSCSEL_CRYST) );
{ }
SUPC->SUPC_CR = (uint32_t)(SUPC_CR_XTALSEL_CRYSTAL_SEL | SUPC_CR_KEY(0xA5)); */
timeout = 0;
while (!(SUPC->SUPC_SR & SUPC_SR_OSCSEL_CRYST) ); #ifndef qmod
} /* Initialize main oscillator */
*/ if ( !(PMC->CKGR_MOR & CKGR_MOR_MOSCSEL) )
{
#ifndef qmod PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN;
/* Initialize main oscillator */ timeout = 0;
if ( !(PMC->CKGR_MOR & CKGR_MOR_MOSCSEL) ) while (!(PMC->PMC_SR & PMC_SR_MOSCXTS) && (timeout++ < CLOCK_TIMEOUT));
{ }
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN;
timeout = 0; /* Switch to 3-20MHz Xtal oscillator */
while (!(PMC->PMC_SR & PMC_SR_MOSCXTS) && (timeout++ < CLOCK_TIMEOUT)); PIOB->PIO_PDR = (1 << 8) | (1 << 9);
} PIOB->PIO_PUDR = (1 << 8) | (1 << 9);
PIOB->PIO_PPDDR = (1 << 8) | (1 << 9);
/* Switch to 3-20MHz Xtal oscillator */ PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN | CKGR_MOR_MOSCSEL;
PIOB->PIO_PDR = (1 << 8) | (1 << 9); /* wait for Main XTAL oscillator stabilization */
PIOB->PIO_PUDR = (1 << 8) | (1 << 9); timeout = 0;
PIOB->PIO_PPDDR = (1 << 8) | (1 << 9); while (!(PMC->PMC_SR & PMC_SR_MOSCSELS) && (timeout++ < CLOCK_TIMEOUT));
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | BOARD_OSCOUNT | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTEN | CKGR_MOR_MOSCSEL; #else
/* wait for Main XTAL oscillator stabilization */ /* QMOD has external 12MHz clock source */
timeout = 0; PIOB->PIO_PDR = (1 << 9);
while (!(PMC->PMC_SR & PMC_SR_MOSCSELS) && (timeout++ < CLOCK_TIMEOUT)); PIOB->PIO_PUDR = (1 << 9);
#else PIOB->PIO_PPDDR = (1 << 9);
/* QMOD has external 12MHz clock source */ PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTBY| CKGR_MOR_MOSCSEL;
PIOB->PIO_PDR = (1 << 9); #endif
PIOB->PIO_PUDR = (1 << 9);
PIOB->PIO_PPDDR = (1 << 9); /* disable the red LED after main clock initialization */
PMC->CKGR_MOR = CKGR_MOR_KEY(0x37) | CKGR_MOR_MOSCRCEN | CKGR_MOR_MOSCXTBY| CKGR_MOR_MOSCSEL; PIOA->PIO_SODR = PIO_LED_RED;
#endif
/* "switch" to main clock as master clock source (should already be the case */
/* disable the red LED after main clock initialization */ PMC->PMC_MCKR = (PMC->PMC_MCKR & ~(uint32_t)PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK;
PIOA->PIO_SODR = PIO_LED_RED; /* wait for master clock to be ready */
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 */
PMC->PMC_MCKR = (PMC->PMC_MCKR & ~(uint32_t)PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK; /* Initialize PLLA */
/* wait for master clock to be ready */ PMC->CKGR_PLLAR = BOARD_PLLAR;
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; ); /* Wait for PLLA to lock */
timeout = 0;
/* Initialize PLLA */ while (!(PMC->PMC_SR & PMC_SR_LOCKA) && (timeout++ < CLOCK_TIMEOUT));
PMC->CKGR_PLLAR = BOARD_PLLAR;
/* Wait for PLLA to lock */ /* Switch to main clock (again ?!?) */
timeout = 0; PMC->PMC_MCKR = (BOARD_MCKR & ~PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK;
while (!(PMC->PMC_SR & PMC_SR_LOCKA) && (timeout++ < CLOCK_TIMEOUT)); /* wait for master clock to be ready */
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
/* Switch to main clock (again ?!?) */
PMC->PMC_MCKR = (BOARD_MCKR & ~PMC_MCKR_CSS_Msk) | PMC_MCKR_CSS_MAIN_CLK; /* switch to PLLA as master clock source */
/* wait for master clock to be ready */ PMC->PMC_MCKR = BOARD_MCKR ;
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; ); /* wait for master clock to be ready */
for ( timeout = 0; !(PMC->PMC_SR & PMC_SR_MCKRDY) && (timeout++ < CLOCK_TIMEOUT) ; );
/* switch to PLLA as master clock source */
PMC->PMC_MCKR = BOARD_MCKR ; /* Configure SysTick for 1ms */
/* wait for master clock to be ready */ SysTick_Config(BOARD_MCK/1000);
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;
/* SysTick based delay function */ /* Interrupt handler for SysTick interrupt */
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 {
void mdelay(unsigned int msecs) } while ((jiffies - jiffies_start) < msecs);
{ }
uint32_t jiffies_start = jiffies;
do {
} while ((jiffies - jiffies_start) < msecs);
}

View File

@@ -1,21 +1,7 @@
/* 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"
#define UV_PER_LSB ((3300 * 1000) / 4096) #define UV_PER_LSB ((3300 * 1000) / 4096)
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;
@@ -86,7 +72,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 => %lu uV\r\n", sample, uv); TRACE_INFO("VERSION_DET ADC=%u => %u uV\r\n", sample, uv);
/* FIXME: convert to board version based on thresholds */ /* FIXME: convert to board version based on thresholds */

View File

@@ -1,22 +1,3 @@
/* 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>
@@ -35,9 +16,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_Clear(&pinsLeds[led]); PIO_Clear(&pinsLeds[led]);
else else
PIO_Set(&pinsLeds[led]); PIO_Set(&pinsLeds[led]);
} }
/* LED blinking code */ /* LED blinking code */
@@ -73,23 +54,16 @@ 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[] = { static const struct blink_state bs_2on_off[] = {
{ 200, 1 }, { 0, 0 }, { 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 {
@@ -135,11 +109,6 @@ 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 {

View File

@@ -1,17 +1,4 @@
/* 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;

View File

@@ -1,21 +1,5 @@
/* 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 "led.h"

View File

@@ -1,453 +1,415 @@
/* ---------------------------------------------------------------------------- /* ----------------------------------------------------------------------------
* 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
* 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 *
* * Implements UART console.
* Implements UART console. *
* */
*/
/*----------------------------------------------------------------------------
/*---------------------------------------------------------------------------- * Headers
* Headers *----------------------------------------------------------------------------*/
*----------------------------------------------------------------------------*/
#include "board.h"
#include "board.h"
#include <stdio.h>
#include <stdio.h> #include <stdint.h>
#include <stdint.h>
#include "ringbuffer.h"
#include "ringbuffer.h"
/*----------------------------------------------------------------------------
/*---------------------------------------------------------------------------- * Definitions
* Definitions *----------------------------------------------------------------------------*/
*----------------------------------------------------------------------------*/
/*----------------------------------------------------------------------------
/*---------------------------------------------------------------------------- * Variables
* Variables *----------------------------------------------------------------------------*/
*----------------------------------------------------------------------------*/
/** Is Console Initialized. */
/** Is Console Initialized. */ static uint8_t _ucIsConsoleInitialized=0;
static uint8_t _ucIsConsoleInitialized=0; /** Ring buffer to queue data to be sent */
/** Ring buffer to queue data to be sent */ static ringbuf uart_tx_buffer;
static ringbuf uart_tx_buffer;
/**
/** * \brief Configures an USART peripheral with the specified parameters.
* \brief Configures an USART peripheral with the specified parameters. *
* * \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). */
*/ extern void UART_Configure( uint32_t baudrate, uint32_t masterClock)
extern void UART_Configure( uint32_t baudrate, uint32_t masterClock) {
{ const Pin pPins[] = CONSOLE_PINS;
const Pin pPins[] = CONSOLE_PINS; Uart *pUart = CONSOLE_UART;
Uart *pUart = CONSOLE_UART;
/* Configure PIO */
/* Configure PIO */ PIO_Configure(pPins, PIO_LISTSIZE(pPins));
PIO_Configure(pPins, PIO_LISTSIZE(pPins));
/* Configure PMC */
/* Configure PMC */ PMC->PMC_PCER0 = 1 << CONSOLE_ID;
PMC->PMC_PCER0 = 1 << CONSOLE_ID;
/* Reset and disable receiver & transmitter */
/* Reset and disable receiver & transmitter */ pUart->UART_CR = UART_CR_RSTRX | UART_CR_RSTTX
pUart->UART_CR = UART_CR_RSTRX | UART_CR_RSTTX | UART_CR_RXDIS | UART_CR_TXDIS;
| UART_CR_RXDIS | UART_CR_TXDIS;
/* Configure mode */
/* Configure mode */ pUart->UART_MR = UART_MR_PAR_NO;
pUart->UART_MR = UART_MR_PAR_NO;
/* Configure baudrate */
/* Configure baudrate */ /* Asynchronous, no oversampling */
/* Asynchronous, no oversampling */ pUart->UART_BRGR = (masterClock / baudrate) / 16;
//pUart->UART_BRGR = (masterClock / baudrate) / 16;
if ((masterClock / baudrate) % 16 >= 7) { /* Disable PDC channel */
pUart->UART_BRGR = ( masterClock / baudrate) / 16 + 1; pUart->UART_PTCR = UART_PTCR_RXTDIS | UART_PTCR_TXTDIS;
} else {
pUart->UART_BRGR = ( masterClock / baudrate) / 16 + 0; /* Reset transmit ring buffer */
} rbuf_reset(&uart_tx_buffer);
/* Disable PDC channel */ /* Enable TX interrupts */
pUart->UART_PTCR = UART_PTCR_RXTDIS | UART_PTCR_TXTDIS; pUart->UART_IER = UART_IER_TXRDY;
NVIC_EnableIRQ(CONSOLE_IRQ);
/* Reset transmit ring buffer */
rbuf_reset(&uart_tx_buffer); /* Enable receiver and transmitter */
pUart->UART_CR = UART_CR_RXEN | UART_CR_TXEN;
/* Enable TX interrupts */
pUart->UART_IER = UART_IER_TXRDY; /* Remember the configuration is complete */
NVIC_EnableIRQ(CONSOLE_IRQ); _ucIsConsoleInitialized=1 ;
}
/* Enable receiver and transmitter */
pUart->UART_CR = UART_CR_RXEN | UART_CR_TXEN; /** Interrupt Service routine to transmit queued data */
void CONSOLE_ISR(void)
/* Remember the configuration is complete */ {
_ucIsConsoleInitialized=1 ; Uart *uart = CONSOLE_UART;
} if (uart->UART_SR & UART_SR_TXRDY) {
if (!rbuf_is_empty(&uart_tx_buffer)) {
/** //uart->UART_IER = UART_IER_TXRDY;
* \brief Disables the USART peripheral and related IRQ uart->UART_THR = rbuf_read(&uart_tx_buffer);
*/ } else {
void UART_Exit(void) uart->UART_IDR = UART_IER_TXRDY;
{ }
if (!_ucIsConsoleInitialized) { }
return; }
}
/**
Uart *pUart = CONSOLE_UART; * \brief Outputs a character on the UART line.
pUart->UART_IDR = UART_IDR_TXRDY; *
pUart->UART_CR = UART_CR_RSTRX | UART_CR_RSTTX | UART_CR_RXDIS | UART_CR_TXDIS | UART_CR_RSTSTA; * \note This function is synchronous (i.e. uses polling).
PMC->PMC_PCDR0 = 1 << CONSOLE_ID; * \param c Character to send.
NVIC_DisableIRQ(CONSOLE_IRQ); */
} extern void UART_PutChar( uint8_t c )
{
/** Interrupt Service routine to transmit queued data */ Uart *pUart = CONSOLE_UART ;
void CONSOLE_ISR(void)
{ /* Initialize console is not already done */
Uart *uart = CONSOLE_UART; if ( !_ucIsConsoleInitialized )
if (uart->UART_SR & UART_SR_TXRDY) { {
if (!rbuf_is_empty(&uart_tx_buffer)) { UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
uart->UART_THR = rbuf_read(&uart_tx_buffer); }
} else {
uart->UART_IDR = UART_IER_TXRDY; /* Only store input if buffer is not full, else drop it */
} bool trigger_isr = false;
} if (rbuf_is_empty(&uart_tx_buffer)) {
} trigger_isr = true;
}
/** if (!rbuf_is_full(&uart_tx_buffer)) {
* \brief Outputs a character on the UART line. rbuf_write(&uart_tx_buffer, c);
* }
* \note This function is asynchronous (i.e. uses a buffer and interrupt to complete the transfer). if (trigger_isr) {
* \param c Character to send. pUart->UART_IER = UART_IER_TXRDY;
*/ CONSOLE_ISR();
void UART_PutChar( uint8_t uc ) }
{ }
Uart *pUart = CONSOLE_UART ;
/**
/* Initialize console is not already done */ * \brief Input a character from the UART line.
if ( !_ucIsConsoleInitialized ) *
{ * \note This function is synchronous
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK); * \return character received.
} */
extern uint32_t UART_GetChar( void )
if (!rbuf_is_full(&uart_tx_buffer)) { {
rbuf_write(&uart_tx_buffer, uc); Uart *pUart = CONSOLE_UART ;
if (!(pUart->UART_IMR & UART_IMR_TXRDY)) {
pUart->UART_IER = UART_IER_TXRDY; if ( !_ucIsConsoleInitialized )
CONSOLE_ISR(); {
} UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
} }
}
while ( (pUart->UART_SR & UART_SR_RXRDY) == 0 )
/** WDT_Restart(WDT);
* \brief Outputs a character on the UART line.
* return pUart->UART_RHR ;
* \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 ) * \brief Check if there is Input from UART line.
{ *
Uart *pUart = CONSOLE_UART ; * \return true if there is Input.
*/
/* Initialize console is not already done */ extern uint32_t UART_IsRxReady( void )
if ( !_ucIsConsoleInitialized ) {
{ Uart *pUart = CONSOLE_UART;
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK);
} if ( !_ucIsConsoleInitialized )
{
while (!(pUart->UART_SR & UART_SR_TXRDY)); /* Wait for transfer buffer to be empty */ UART_Configure( CONSOLE_BAUDRATE, BOARD_MCK ) ;
pUart->UART_THR = uc; /* Send data to UART peripheral */ }
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) */ return (pUart->UART_SR & UART_SR_RXRDY) > 0 ;
} }
/** /**
* \brief Input a character from the UART line. * Displays the content of the given frame on the UART0.
* *
* \note This function is synchronous * \param pucFrame Pointer to the frame to dump.
* \return character received. * \param dwSize Buffer size in bytes.
*/ */
extern uint32_t UART_GetChar( void ) extern void UART_DumpFrame( uint8_t* pucFrame, uint32_t dwSize )
{ {
Uart *pUart = CONSOLE_UART ; uint32_t dw ;
if ( !_ucIsConsoleInitialized ) for ( dw=0 ; dw < dwSize ; dw++ )
{ {
UART_Configure(CONSOLE_BAUDRATE, BOARD_MCK); printf( "%02X ", pucFrame[dw] ) ;
} }
while ( (pUart->UART_SR & UART_SR_RXRDY) == 0 ) printf( "\n\r" ) ;
WDT_Restart(WDT); }
return pUart->UART_RHR ; /**
} * Displays the content of the given buffer on the UART0.
*
/** * \param pucBuffer Pointer to the buffer to dump.
* \brief Check if there is Input from UART line. * \param dwSize Buffer size in bytes.
* * \param dwAddress Start address to display
* \return true if there is Input. */
*/ extern void UART_DumpMemory( uint8_t* pucBuffer, uint32_t dwSize, uint32_t dwAddress )
extern uint32_t UART_IsRxReady( void ) {
{ uint32_t i ;
Uart *pUart = CONSOLE_UART; uint32_t j ;
uint32_t dwLastLineStart ;
if ( !_ucIsConsoleInitialized ) uint8_t* pucTmp ;
{
UART_Configure( CONSOLE_BAUDRATE, BOARD_MCK ) ; for ( i=0 ; i < (dwSize / 16) ; i++ )
} {
printf( "0x%08X: ", (unsigned int)(dwAddress + (i*16)) ) ;
return (pUart->UART_SR & UART_SR_RXRDY) > 0 ; pucTmp = (uint8_t*)&pucBuffer[i*16] ;
}
for ( j=0 ; j < 4 ; j++ )
/** {
* Displays the content of the given frame on the UART0. printf( "%02X%02X%02X%02X ", pucTmp[0], pucTmp[1], pucTmp[2], pucTmp[3] ) ;
* pucTmp += 4 ;
* \param pucFrame Pointer to the frame to dump. }
* \param dwSize Buffer size in bytes.
*/ pucTmp=(uint8_t*)&pucBuffer[i*16] ;
extern void UART_DumpFrame( uint8_t* pucFrame, uint32_t dwSize )
{ for ( j=0 ; j < 16 ; j++ )
uint32_t dw ; {
UART_PutChar( *pucTmp++ ) ;
for ( dw=0 ; dw < dwSize ; dw++ ) }
{
printf( "%02X ", pucFrame[dw] ) ; printf( "\n\r" ) ;
} }
printf( "\n\r" ) ; if ( (dwSize%16) != 0 )
} {
dwLastLineStart=dwSize - (dwSize%16) ;
/**
* Displays the content of the given buffer on the UART0. printf( "0x%08X: ", (unsigned int)(dwAddress + dwLastLineStart) ) ;
* for ( j=dwLastLineStart ; j < dwLastLineStart+16 ; j++ )
* \param pucBuffer Pointer to the buffer to dump. {
* \param dwSize Buffer size in bytes. if ( (j!=dwLastLineStart) && (j%4 == 0) )
* \param dwAddress Start address to display {
*/ printf( " " ) ;
extern void UART_DumpMemory( uint8_t* pucBuffer, uint32_t dwSize, uint32_t dwAddress ) }
{
uint32_t i ; if ( j < dwSize )
uint32_t j ; {
uint32_t dwLastLineStart ; printf( "%02X", pucBuffer[j] ) ;
uint8_t* pucTmp ; }
else
for ( i=0 ; i < (dwSize / 16) ; i++ ) {
{ printf(" ") ;
printf( "0x%08X: ", (unsigned int)(dwAddress + (i*16)) ) ; }
pucTmp = (uint8_t*)&pucBuffer[i*16] ; }
for ( j=0 ; j < 4 ; j++ ) printf( " " ) ;
{ for ( j=dwLastLineStart ; j < dwSize ; j++ )
printf( "%02X%02X%02X%02X ", pucTmp[0], pucTmp[1], pucTmp[2], pucTmp[3] ) ; {
pucTmp += 4 ; UART_PutChar( pucBuffer[j] ) ;
} }
pucTmp=(uint8_t*)&pucBuffer[i*16] ; printf( "\n\r" ) ;
}
for ( j=0 ; j < 16 ; j++ ) }
{
UART_PutChar( *pucTmp++ ) ; /**
} * 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 )
if ( (dwSize%16) != 0 ) {
{ uint8_t ucKey ;
dwLastLineStart=dwSize - (dwSize%16) ; uint8_t ucNbNb=0 ;
uint32_t dwValue=0 ;
printf( "0x%08X: ", (unsigned int)(dwAddress + dwLastLineStart) ) ;
for ( j=dwLastLineStart ; j < dwLastLineStart+16 ; j++ ) while ( 1 )
{ {
if ( (j!=dwLastLineStart) && (j%4 == 0) ) ucKey=UART_GetChar() ;
{ UART_PutChar( ucKey ) ;
printf( " " ) ;
} if ( ucKey >= '0' && ucKey <= '9' )
{
if ( j < dwSize ) dwValue = (dwValue * 10) + (ucKey - '0');
{ ucNbNb++ ;
printf( "%02X", pucBuffer[j] ) ; }
} else
else {
{ if ( ucKey == 0x0D || ucKey == ' ' )
printf(" ") ; {
} if ( ucNbNb == 0 )
} {
printf( "\n\rWrite a number and press ENTER or SPACE!\n\r" ) ;
printf( " " ) ; return 0 ;
for ( j=dwLastLineStart ; j < dwSize ; j++ ) }
{ else
UART_PutChar( pucBuffer[j] ) ; {
} printf( "\n\r" ) ;
*pdwValue=dwValue ;
printf( "\n\r" ) ;
} return 1 ;
} }
}
/** else
* Reads an integer {
* printf( "\n\r'%c' not a number!\n\r", ucKey ) ;
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
*/ return 0 ;
extern uint32_t UART_GetInteger( uint32_t* pdwValue ) }
{ }
uint8_t ucKey ; WDT_Restart(WDT);
uint8_t ucNbNb=0 ; }
uint32_t dwValue=0 ; }
while ( 1 ) /**
{ * Reads an integer and check the value
ucKey=UART_GetChar() ; *
UART_PutChar( ucKey ) ; * \param pdwValue Pointer to the uint32_t variable to contain the input value.
* \param dwMin Minimum value
if ( ucKey >= '0' && ucKey <= '9' ) * \param dwMax Maximum value
{ */
dwValue = (dwValue * 10) + (ucKey - '0'); extern uint32_t UART_GetIntegerMinMax( uint32_t* pdwValue, uint32_t dwMin, uint32_t dwMax )
ucNbNb++ ; {
} uint32_t dwValue=0 ;
else
{ if ( UART_GetInteger( &dwValue ) == 0 )
if ( ucKey == 0x0D || ucKey == ' ' ) {
{ return 0 ;
if ( ucNbNb == 0 ) }
{
printf( "\n\rWrite a number and press ENTER or SPACE!\n\r" ) ; if ( dwValue < dwMin || dwValue > dwMax )
return 0 ; {
} printf( "\n\rThe number have to be between %d and %d\n\r", (int)dwMin, (int)dwMax ) ;
else
{ return 0 ;
printf( "\n\r" ) ; }
*pdwValue=dwValue ;
printf( "\n\r" ) ;
return 1 ;
} *pdwValue = dwValue ;
}
else return 1 ;
{ }
printf( "\n\r'%c' not a number!\n\r", ucKey ) ;
/**
return 0 ; * Reads an hexadecimal number
} *
} * \param pdwValue Pointer to the uint32_t variable to contain the input value.
WDT_Restart(WDT); */
} extern uint32_t UART_GetHexa32( uint32_t* pdwValue )
} {
uint8_t ucKey ;
/** uint32_t dw = 0 ;
* Reads an integer and check the value uint32_t dwValue = 0 ;
*
* \param pdwValue Pointer to the uint32_t variable to contain the input value. for ( dw=0 ; dw < 8 ; dw++ )
* \param dwMin Minimum value {
* \param dwMax Maximum value ucKey = UART_GetChar() ;
*/ UART_PutChar( ucKey ) ;
extern uint32_t UART_GetIntegerMinMax( uint32_t* pdwValue, uint32_t dwMin, uint32_t dwMax )
{ if ( ucKey >= '0' && ucKey <= '9' )
uint32_t dwValue=0 ; {
dwValue = (dwValue * 16) + (ucKey - '0') ;
if ( UART_GetInteger( &dwValue ) == 0 ) }
{ else
return 0 ; {
} if ( ucKey >= 'A' && ucKey <= 'F' )
{
if ( dwValue < dwMin || dwValue > dwMax ) dwValue = (dwValue * 16) + (ucKey - 'A' + 10) ;
{ }
printf( "\n\rThe number have to be between %d and %d\n\r", (int)dwMin, (int)dwMax ) ; else
{
return 0 ; if ( ucKey >= 'a' && ucKey <= 'f' )
} {
dwValue = (dwValue * 16) + (ucKey - 'a' + 10) ;
printf( "\n\r" ) ; }
else
*pdwValue = dwValue ; {
printf( "\n\rIt is not a hexa character!\n\r" ) ;
return 1 ;
} return 0 ;
}
/** }
* Reads an hexadecimal number }
* }
* \param pdwValue Pointer to the uint32_t variable to contain the input value.
*/ printf("\n\r" ) ;
extern uint32_t UART_GetHexa32( uint32_t* pdwValue ) *pdwValue = dwValue ;
{
uint8_t ucKey ; return 1 ;
uint32_t dw = 0 ; }
uint32_t dwValue = 0 ;
#if defined __ICCARM__ /* IAR Ewarm 5.41+ */
for ( dw=0 ; dw < 8 ; dw++ ) /**
{ * \brief Outputs a character on the UART.
ucKey = UART_GetChar() ; *
UART_PutChar( ucKey ) ; * \param c Character to output.
*
if ( ucKey >= '0' && ucKey <= '9' ) * \return The character that was output.
{ */
dwValue = (dwValue * 16) + (ucKey - '0') ; extern WEAK signed int putchar( signed int c )
} {
else UART_PutChar( c ) ;
{
if ( ucKey >= 'A' && ucKey <= 'F' ) return c ;
{ }
dwValue = (dwValue * 16) + (ucKey - 'A' + 10) ; #endif // defined __ICCARM__
}
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__

View File

@@ -1,170 +0,0 @@
/* 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

View File

@@ -1,28 +0,0 @@
/* 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);

View File

@@ -1,25 +0,0 @@
/* 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);

View File

@@ -1 +0,0 @@
sysmoOCTSIM-Tester

View File

@@ -1,81 +0,0 @@
/* 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;
}

View File

@@ -1,225 +0,0 @@
/* 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;
}

View File

@@ -1,140 +0,0 @@
#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;
}

View File

@@ -1,22 +1,3 @@
/* 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" #include "simtrace_usb.h"
@@ -26,28 +7,7 @@
/** 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}

View File

@@ -1 +0,0 @@
OWHW

View File

@@ -1,55 +1,28 @@
/* Card simulator specific functions /* Card simulator specific functions */
/* (C) 2015 by Harald Welte <hwelte@hmw-consulting.de>
* *
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de> * This program is free software; you can redistribute it and/or modify
* (C) 2018-2019, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de> * 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 free software; you can redistribute it and/or modify * This program is distributed in the hope that it will be useful,
* it under the terms of the GNU General Public License as published by * but WITHOUT ANY WARRANTY; without even the implied warranty of
* the Free Software Foundation; either version 2 of the License, or * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* (at your option) any later version. * GNU General Public License for more details.
* *
* This program is distributed in the hope that it will be useful, * You should have received a copy of the GNU General Public License
* but WITHOUT ANY WARRANTY; without even the implied warranty of * along with this program; if not, write to the Free Software
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
* 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)
@@ -63,18 +36,5 @@ 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
}
} }

View File

@@ -1,53 +1,16 @@
/* 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" #include "simtrace_usb.h"
#define LED_USIM1 LED_GREEN
#define LED_USIM2 LED_RED
/** 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}

View File

@@ -1,21 +1,3 @@
/* 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);

View File

@@ -1,19 +1,3 @@
/* 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);

View File

@@ -1,19 +1,3 @@
/* 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);

View File

@@ -1,19 +1,3 @@
/* 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);

View File

@@ -1 +0,0 @@
sysmoQMOD (Quad Modem)

View File

@@ -1,26 +1,9 @@
/* sysmocom quad-modem sysmoQMOD application code /* Quad-modem speciic 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"
@@ -28,7 +11,6 @@
#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};
@@ -47,7 +29,6 @@ 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,
@@ -71,8 +52,11 @@ 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 */
@@ -80,25 +64,22 @@ 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 < ARRAY_SIZE(__eeprom_bin); i++) { for (i = 0; i < 256; i++) {
int rc = eeprom_write_byte(0x50, i, __eeprom_bin[i]); int rc = eeprom_write_byte(0x50, i, __eeprom_bin[i]);
if (rc < 0) { /* if the result was negative, repeat that write */
TRACE_ERROR("Writing EEPROM failed at byte %u: 0x%02x\n\r", if (rc < 0)
i, __eeprom_bin[i]); 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 < ARRAY_SIZE(__eeprom_bin); i++) { for (i = 0; i < 256; i++) {
int byte = eeprom_read_byte(0x50, i); int byte = eeprom_read_byte(0x50, i);
TRACE_DEBUG("0x%02x: %02x\n\r", i, byte); TRACE_INFO("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 */
@@ -106,29 +87,6 @@ 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;
@@ -138,14 +96,9 @@ 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);
@@ -154,7 +107,6 @@ 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);
@@ -171,14 +123,13 @@ 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%02lx to EEPROM offset 0x%02lx\n\r", val, addr); printf("Writing value 0x%02x to EEPROM offset 0x%02x\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%02lx] = 0x%02x\n\r", addr, eeprom_read_byte(0x50, addr)); printf("EEPROM[0x%02x] = 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);
@@ -189,24 +140,12 @@ 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");
@@ -216,11 +155,8 @@ 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");
@@ -230,22 +166,6 @@ 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);
@@ -254,24 +174,14 @@ 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 'a':
printf("Asserting SIMTRACExx_ERASE allowed on next command\n\r");
allow_erase = true;
break;
case 'y': case 'y':
if (allow_erase) { printf("Seetting SIMTRACExx_ERASE (active)\n\r");
printf("Setting SIMTRACExx_ERASE (active)\n\r"); PIO_Set(&pin_peer_erase);
PIO_Set(&pin_peer_erase);
} else {
printf("Please first allow setting SIMTRACExx_ERASE\n\r");
}
break; 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);
@@ -293,13 +203,6 @@ 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)
@@ -332,13 +235,11 @@ void board_main_top(void)
TRACE_INFO("Detected Quad-Modem ST12\n\r"); TRACE_INFO("Detected Quad-Modem ST12\n\r");
} else { } else {
TRACE_INFO("Detected Quad-Modem ST34\n\r"); TRACE_INFO("Detected Quad-Modem ST34\n\r");
#ifndef APPLICATION_dfu
/* make sure we use the second set of USB Strings /* make sure we use the second set of USB Strings
* calling the interfaces "Modem 3" and "Modem 4" rather * calling the interfaces "Modem 3" and "Modem 4" rather
* than 1+2 */ * than 1+2 */
usb_strings[7] = usb_strings[9]; usb_strings[7] = usb_strings[9];
usb_strings[8] = usb_strings[10]; usb_strings[8] = usb_strings[10];
#endif
} }
/* Obtain the circuit board version (currently just prints voltage */ /* Obtain the circuit board version (currently just prints voltage */
@@ -360,32 +261,20 @@ static int uart_has_loopback_jumper(void)
/* Configure UART pins as I/O */ /* Configure UART pins as I/O */
PIO_Configure(uart_loopback_pins, PIO_LISTSIZE(uart_loopback_pins)); 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++) { for (i = 0; i < 10; i++) {
/* Set TxD high; abort if RxD doesn't go high either */ /* Set TxD high; abort if RxD doesn't go high either */
PIO_Set(&uart_loopback_pins[1]); PIO_Set(&uart_loopback_pins[1]);
if (!PIO_Get(&uart_loopback_pins[0])) { if (!PIO_Get(&uart_loopback_pins[0]))
has_loopback_jumper = 0; return 0;
break;
}
/* Set TxD low, abort if RxD doesn't go low either */ /* Set TxD low, abort if RxD doesn't go low either */
PIO_Clear(&uart_loopback_pins[1]); PIO_Clear(&uart_loopback_pins[1]);
if (PIO_Get(&uart_loopback_pins[0])) { if (PIO_Get(&uart_loopback_pins[0]))
has_loopback_jumper = 0; return 0;
break;
}
} }
/* if we reached here, RxD always follows TxD and thus a
/* Put pins back to UART mode */ * loopback jumper has been placed on RxD/TxD, and we will boot
const Pin uart_pins[] = {PINS_UART}; * into DFU unconditionally */
PIO_Configure(uart_pins, PIO_LISTSIZE(uart_pins)); return 1;
return has_loopback_jumper;
} }
int board_override_enter_dfu(void) int board_override_enter_dfu(void)

View File

@@ -1,21 +1,3 @@
/* 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"

View File

@@ -1,19 +1,3 @@
/* 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>
@@ -185,8 +169,6 @@ 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();

View File

@@ -1,24 +1,11 @@
/* 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
* *
* This program is free software; you can redistribute it and/or modify * Depending on the board this is running on, it might be possible
* 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"

View File

@@ -1,24 +1,11 @@
/* Code to control the PERST lines of attached modems /* Code to control the PERST lines of attached modems
* *
* This program is free software; you can redistribute it and/or modify * Depending on the board this is running on, it might be possible
* 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"

View File

@@ -1,163 +0,0 @@
/* 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

View File

@@ -1,68 +0,0 @@
/* 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;
}

View File

@@ -1,22 +1,3 @@
/* 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" #include "simtrace_usb.h"
@@ -26,31 +7,14 @@
/* Board definition */ /* Board definition */
#define simtrace #define simtrace
/** oscillator used as main clock source (in Hz) */ /* Board main oscillator frequency (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 configuration **/ /** Pin configuration **/
/* Button to force bootloader start (shorted to ground when pressed */ /* Button to force bootloader start (shorted to ground when pressed */
#define PIN_BOOTLOADER_SW {PIO_PA31, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP} #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) */ /* 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} #define SIM_PWEN_PIN {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
/* Enable powering the SIM card */ /* Enable powering the SIM card */
#define PWR_PINS SIM_PWEN_PIN #define PWR_PINS SIM_PWEN_PIN
/* Card presence pin */ /* Card presence pin */
@@ -74,9 +38,9 @@
/** Phone connection **/ /** Phone connection **/
/* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */ /* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */
#define PIN_USIM1_VCC {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_EDGE | PIO_DEGLITCH } #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) */ /* 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_EDGE | PIO_DEGLITCH } #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) */ /* 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} #define PIN_PHONE_IO {PIO_PA22A_TXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
/* Phone CLK clock input (CLK_PHONE in schematic) */ /* Phone CLK clock input (CLK_PHONE in schematic) */
@@ -108,9 +72,9 @@
/* Pins used to sniff phone-card communication */ /* Pins used to sniff phone-card communication */
#define PINS_SIM_SNIFF PIN_SIM_IO, PIN_SIM_CLK, PIN_SIM_RST_SNIFF #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) */ /* 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} #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) */ /* 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} #define PIN_VCC_FWD_SNIFF {VCC_FWD, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
/* Use phone VCC to power card */ /* Use phone VCC to power card */
#define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF #define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF
@@ -134,11 +98,6 @@
/* SPI flash write protect pin (active low, pulled low) */ /* SPI flash write protect pin (active low, pulled low) */
#define PIN_SPI_WP {PA15, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT} #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 */ /** USB definitions */
/* OpenMoko SIMtrace 2 USB vendor ID */ /* OpenMoko SIMtrace 2 USB vendor ID */
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO #define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO

View File

@@ -1 +0,0 @@
SIMtrace 2

View File

@@ -1,22 +1,6 @@
/* SIMtrace with SAM3S specific application code /* SIMtrace specific application code */
* /* (C) 2017 by Harald Welte <laforge@gnumonks.org> */
* (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 "board.h"
#include "simtrace.h" #include "simtrace.h"
#include "utils.h" #include "utils.h"

View File

@@ -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

View File

@@ -1,22 +1,3 @@
/* 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>
@@ -29,18 +10,8 @@ enum card_io {
CARD_IO_CLK, CARD_IO_CLK,
}; };
/** initialise card slot struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan,
* @param[in] slot_num slot number (arbitrary number) uint8_t in_ep, uint8_t irq_ep);
* @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);
@@ -65,4 +36,3 @@ 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);

View File

@@ -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 );

View File

@@ -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 );

View File

@@ -1,21 +1,3 @@
/* 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>

View File

@@ -1,21 +1,3 @@
/* 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>

View File

@@ -1,19 +1,3 @@
/* 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
@@ -32,7 +16,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);
int rbuf_write(volatile ringbuf * rb, uint8_t item); void 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);

View File

@@ -1,22 +1,3 @@
/* 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
@@ -58,7 +39,6 @@ enum confNum {
#ifdef HAVE_MITM #ifdef HAVE_MITM
CFG_NUM_MITM, CFG_NUM_MITM,
#endif #endif
CFG_NUM_VERSION,
NUM_CONF NUM_CONF
}; };
@@ -67,19 +47,19 @@ 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 DFURT_IF_DESCRIPTOR_STRUCT
} __attribute__ ((packed)) CCIDDriverConfigurationDescriptors; } __attribute__ ((packed)) CCIDDriverConfigurationDescriptors;
extern const USBConfigurationDescriptor *configurationDescriptorsArr[]; extern const USBConfigurationDescriptor *configurationDescriptorsArr[];
@@ -88,7 +68,7 @@ extern const USBConfigurationDescriptor *configurationDescriptorsArr[];
* @param[io] usart USART peripheral base address * @param[io] usart USART peripheral base address
* @param[in] fidi FiDi value as provided in TA interface byte * @param[in] fidi FiDi value as provided in TA interface byte
*/ */
void update_fidi(Usart_info *usart, uint8_t fidi); void update_fidi(Usart *usart, uint8_t fidi);
void ISR_PhoneRST( const Pin *pPin); void ISR_PhoneRST( const Pin *pPin);

View File

@@ -1,21 +1,21 @@
/* SIMtrace2 USB protocol /* SIMtrace2 USB protocol */
/* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
* *
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de> * This program is free software; you can redistribute it and/or modify
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de> * 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 free software; you can redistribute it and/or modify * This program is distributed in the hope that it will be useful,
* it under the terms of the GNU General Public License as published by * but WITHOUT ANY WARRANTY; without even the implied warranty of
* the Free Software Foundation; either version 2 of the License, or * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* (at your option) any later version. * GNU General Public License for more details.
* *
* This program is distributed in the hope that it will be useful, * You should have received a copy of the GNU General Public License
* but WITHOUT ANY WARRANTY; without even the implied warranty of * along with this program; if not, write to the Free Software
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
* 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
@@ -291,13 +291,9 @@ struct st_modem_status {
/* SIMTRACE_MSGT_SNIFF_CHANGE flags */ /* SIMTRACE_MSGT_SNIFF_CHANGE flags */
#define SNIFF_CHANGE_FLAG_CARD_INSERT (1<<0) #define SNIFF_CHANGE_FLAG_CARD_INSERT (1<<0)
#define SNIFF_CHANGE_FLAG_CARD_EJECT (1<<1) #define SNIFF_CHANGE_FLAG_CARD_EJECT (1<<1)
#define SNIFF_CHANGE_FLAG_RESET_ASSERT (1<<2) #define SNIFF_CHANGE_FLAG_RESET_HOLD (1<<2)
#define SNIFF_CHANGE_FLAG_RESET_DEASSERT (1<<3) #define SNIFF_CHANGE_FLAG_RESET_RELEASE (1<<3)
#define SNIFF_CHANGE_FLAG_TIMEOUT_WT (1<<4) #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 */ /* SIMTRACE_MSGT_SNIFF_CHANGE */
struct sniff_change { struct sniff_change {
@@ -313,8 +309,8 @@ struct sniff_fidi {
/* SIMTRACE_MSGT_SNIFF_ATR, SIMTRACE_MSGT_SNIFF_PPS, SIMTRACE_MSGT_SNIFF_TPDU */ /* SIMTRACE_MSGT_SNIFF_ATR, SIMTRACE_MSGT_SNIFF_PPS, SIMTRACE_MSGT_SNIFF_TPDU */
struct sniff_data { struct sniff_data {
/* data flags */ /* if the data is complete (an error might have occurred during transmission */
uint32_t flags; bool complete;
/* data length */ /* data length */
uint16_t length; uint16_t length;
/* data */ /* data */

View File

@@ -1,21 +1,20 @@
/* SIMtrace 2 USB definitions /* This program is free software; you can redistribute it and/or modify
* * it under the terms of the GNU Affero 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 3 of the License, or
*
* 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. * (at your option) any later version.
* *
* This program is distributed in the hope that it will be useful, * This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of * but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details. * GNU Affero General Public License for more details.
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU Affero General Public License
* along with this program; if not, write to the Free Software * along with this program. If not, see <http://www.gnu.org/licenses/>.
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA *
* (C) 2018 by Kevin Redon <kredon@sysmocom.de>
* All Rights Reserved
*/ */
/* SIMtrace USB IDs */ /* SIMtrace USB IDs */
#define USB_VENDOR_OPENMOKO 0x1d50 #define USB_VENDOR_OPENMOKO 0x1d50
#define USB_PRODUCT_OWHW_SAM3_DFU 0x4001 /* was 0x4000 */ #define USB_PRODUCT_OWHW_SAM3_DFU 0x4001 /* was 0x4000 */

View File

@@ -1,31 +1,3 @@
/* ----------------------------------------------------------------------------
* 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>
@@ -52,14 +24,9 @@ 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);

View File

@@ -1,19 +1,3 @@
/* 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>

View File

@@ -1,19 +1,3 @@
/* 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>

View File

@@ -1,19 +1,3 @@
/* General utilities
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
*/
#pragma once #pragma once
#define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0])) #define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))

Some files were not shown because too many files have changed in this diff Show More