mirror of
https://gitea.osmocom.org/sim-card/simtrace2.git
synced 2026-03-18 22:38:32 +03:00
Compare commits
60 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b7f94dcb9f | ||
|
|
c90de6983c | ||
|
|
6f41349db9 | ||
|
|
208890ad6e | ||
|
|
964cda309d | ||
|
|
331fa5a237 | ||
|
|
6fada5604b | ||
|
|
931fe558df | ||
|
|
d401b12136 | ||
|
|
389a4040d5 | ||
|
|
5db9402a5f | ||
|
|
ac7e73a579 | ||
|
|
7233cf803a | ||
|
|
cdcdcc9f6d | ||
|
|
e876bf53e8 | ||
|
|
f7f1ea864d | ||
|
|
3feadfa910 | ||
|
|
9acff5ee5a | ||
|
|
e5efbb156c | ||
|
|
ede87e067d | ||
|
|
acb7bd9fbe | ||
|
|
4b487b836a | ||
|
|
e0265462d8 | ||
|
|
d14970f95b | ||
|
|
b1a81c130e | ||
|
|
298a5ba722 | ||
|
|
c3ef475ea5 | ||
|
|
155f57abcf | ||
|
|
bc62335768 | ||
|
|
63490361d2 | ||
|
|
6228d187da | ||
|
|
a634c0efee | ||
|
|
bb9b0dc8e8 | ||
|
|
b7e326cad3 | ||
|
|
2fdcf3b38d | ||
|
|
7e5cda5732 | ||
|
|
032fc5f844 | ||
|
|
b1f99c909c | ||
|
|
53b4e593aa | ||
|
|
dc85fbc3e1 | ||
|
|
66ffb6d493 | ||
|
|
5b5d24ebf3 | ||
|
|
faf1e88e48 | ||
|
|
64f69fc4ac | ||
|
|
6303c39a00 | ||
|
|
ad0958e9e3 | ||
|
|
4f3a0356a4 | ||
|
|
7d5d011095 | ||
|
|
1dbcf62295 | ||
|
|
e5f891a825 | ||
|
|
a6bd7178b5 | ||
|
|
ba15387b09 | ||
|
|
c171112994 | ||
|
|
29200c6223 | ||
|
|
80d9476602 | ||
|
|
e2b0f971e5 | ||
|
|
e07640c35a | ||
|
|
82b628524f | ||
|
|
cb78b55c26 | ||
|
|
77ff745cca |
@@ -50,23 +50,34 @@ make clean
|
|||||||
echo
|
echo
|
||||||
echo "=============== HOST START =============="
|
echo "=============== HOST START =============="
|
||||||
cd $TOPDIR/host
|
cd $TOPDIR/host
|
||||||
make clean
|
autoreconf --install --force
|
||||||
make
|
./configure --enable-sanitize --enable-werror
|
||||||
make clean
|
$MAKE $PARALLEL_MAKE
|
||||||
|
#$MAKE distcheck || cat-testlogs.sh
|
||||||
|
make dist
|
||||||
|
|
||||||
|
#if [ "$WITH_MANUALS" = "1" ] && [ "$PUBLISH" = "1" ]; then
|
||||||
|
# make -C "$base/doc/manuals" publish
|
||||||
|
#fi
|
||||||
|
|
||||||
if [ "x$publish" = "x--publish" ]; then
|
if [ "x$publish" = "x--publish" ]; then
|
||||||
echo
|
echo
|
||||||
echo "=============== UPLOAD BUILD =============="
|
echo "=============== UPLOAD BUILD =============="
|
||||||
|
|
||||||
cat > "$WORKSPACE/known_hosts" <<EOF
|
cat > "/build/known_hosts" <<EOF
|
||||||
[rita.osmocom.org]:48 ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQDDgQ9HntlpWNmh953a2Gc8NysKE4orOatVT1wQkyzhARnfYUerRuwyNr1GqMyBKdSI9amYVBXJIOUFcpV81niA7zQRUs66bpIMkE9/rHxBd81SkorEPOIS84W4vm3SZtuNqa+fADcqe88Hcb0ZdTzjKILuwi19gzrQyME2knHY71EOETe9Yow5RD2hTIpB5ecNxI0LUKDq+Ii8HfBvndPBIr0BWYDugckQ3Bocf+yn/tn2/GZieFEyFpBGF/MnLbAAfUKIdeyFRX7ufaiWWz5yKAfEhtziqdAGZaXNaLG6gkpy3EixOAy6ZXuTAk3b3Y0FUmDjhOHllbPmTOcKMry9
|
[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 ecdsa-sha2-nistp256 AAAAE2VjZHNhLXNoYTItbmlzdHAyNTYAAAAIbmlzdHAyNTYAAABBBPdWn1kEousXuKsZ+qJEZTt/NSeASxCrUfNDW3LWtH+d8Ust7ZuKp/vuyG+5pe5pwpPOgFu7TjN+0lVjYJVXH54=
|
||||||
[rita.osmocom.org]:48 ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIK8iivY70EiR5NiGChV39gRLjNpC8lvu1ZdHtdMw2zuX
|
[rita.osmocom.org]:48 ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIK8iivY70EiR5NiGChV39gRLjNpC8lvu1ZdHtdMw2zuX
|
||||||
EOF
|
EOF
|
||||||
SSH_COMMAND="ssh -o 'UserKnownHostsFile=$WORKSPACE/known_hosts' -p 48"
|
SSH_COMMAND="ssh -o 'UserKnownHostsFile=/build/known_hosts' -p 48"
|
||||||
rsync -avz --delete -e "$SSH_COMMAND" $TOPDIR/firmware/bin/*.bin binaries@rita.osmocom.org:web-files/simtrace2/firmware/
|
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
|
fi
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo "=============== HOST CLEAN =============="
|
||||||
|
$MAKE maintainer-clean
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "=============== FIRMWARE CLEAN =============="
|
echo "=============== FIRMWARE CLEAN =============="
|
||||||
cd $TOPDIR/firmware/
|
cd $TOPDIR/firmware/
|
||||||
|
|||||||
8
debian/changelog
vendored
8
debian/changelog
vendored
@@ -1,4 +1,10 @@
|
|||||||
simtrace2 (0.5) UNRELEASED; urgency=medium
|
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.
|
* Initial debian package release.
|
||||||
|
|
||||||
|
|||||||
1
debian/control
vendored
1
debian/control
vendored
@@ -4,6 +4,7 @@ Section: devel
|
|||||||
Priority: optional
|
Priority: optional
|
||||||
Build-Depends: debhelper (>= 9),
|
Build-Depends: debhelper (>= 9),
|
||||||
libosmocore-dev,
|
libosmocore-dev,
|
||||||
|
libpcsclite-dev,
|
||||||
libnewlib-arm-none-eabi,
|
libnewlib-arm-none-eabi,
|
||||||
libusb-1.0-0-dev,
|
libusb-1.0-0-dev,
|
||||||
gcc-arm-none-eabi
|
gcc-arm-none-eabi
|
||||||
|
|||||||
@@ -28,6 +28,7 @@
|
|||||||
|
|
||||||
# Makefile for compiling the Getting Started with SAM3S Microcontrollers project
|
# Makefile for compiling the Getting Started with SAM3S Microcontrollers project
|
||||||
|
|
||||||
|
GIT_VERSION=$(shell $(TOP)/git-version-gen $(TOP)/.tarvers)
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
# User-modifiable options
|
# User-modifiable options
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
@@ -43,7 +44,7 @@ MEMORIES ?= flash dfu
|
|||||||
# Output file basename
|
# Output file basename
|
||||||
APP ?= dfu
|
APP ?= dfu
|
||||||
|
|
||||||
# Output directories
|
# Output directories and filename
|
||||||
OUTPUT = $(BOARD)-$(APP)
|
OUTPUT = $(BOARD)-$(APP)
|
||||||
BIN = bin
|
BIN = bin
|
||||||
OBJ = obj/$(BOARD)
|
OBJ = obj/$(BOARD)
|
||||||
@@ -73,7 +74,6 @@ GDB = $(CROSS_COMPILE)gdb
|
|||||||
NM = $(CROSS_COMPILE)nm
|
NM = $(CROSS_COMPILE)nm
|
||||||
|
|
||||||
TOP=..
|
TOP=..
|
||||||
GIT_VERSION=$(shell $(TOP)/git-version-gen $(TOP)/.tarvers)
|
|
||||||
|
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
# Files
|
# Files
|
||||||
@@ -120,15 +120,15 @@ C_OBJECTS = $(C_FILES:%.c=%.o)
|
|||||||
# TRACE_LEVEL_NO_TRACE 0
|
# TRACE_LEVEL_NO_TRACE 0
|
||||||
TRACE_LEVEL ?= 4
|
TRACE_LEVEL ?= 4
|
||||||
|
|
||||||
DEBUG_PHONE_SNIFF?=0
|
# allow asserting the peer SAM3S ERASE signal to completely erase the flash
|
||||||
|
# only applicable for qmod board
|
||||||
|
ALLOW_PEER_ERASE?=0
|
||||||
|
|
||||||
#CFLAGS+=-DUSB_NO_DEBUG=1
|
#CFLAGS+=-DUSB_NO_DEBUG=1
|
||||||
|
|
||||||
# Optimization level, put in comment for debugging
|
# Optimization level, put in comment for debugging
|
||||||
OPTIMIZATION ?= -Os
|
OPTIMIZATION ?= -Os
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# Flags
|
# Flags
|
||||||
INCLUDES_USB = -I$(AT91LIB)/usb/include -I$(AT91LIB)
|
INCLUDES_USB = -I$(AT91LIB)/usb/include -I$(AT91LIB)
|
||||||
|
|
||||||
@@ -163,7 +163,7 @@ CFLAGS += -Wno-suggest-attribute=noreturn
|
|||||||
#CFLAGS += -Wa,-a,-ad
|
#CFLAGS += -Wa,-a,-ad
|
||||||
CFLAGS += -D__ARM
|
CFLAGS += -D__ARM
|
||||||
CFLAGS += --param max-inline-insns-single=500 -mcpu=cortex-m3 -mthumb # -mfix-cortex-m3-ldrd
|
CFLAGS += --param max-inline-insns-single=500 -mcpu=cortex-m3 -mthumb # -mfix-cortex-m3-ldrd
|
||||||
CFLAGS += -ffunction-sections -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP) -DTRACE_LEVEL=$(TRACE_LEVEL) -DDEBUG_PHONE_SNIFF=$(DEBUG_PHONE_SNIFF)
|
CFLAGS += -ffunction-sections -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP) -DTRACE_LEVEL=$(TRACE_LEVEL) -DALLOW_PEER_ERASE=$(ALLOW_PEER_ERASE)
|
||||||
CFLAGS += -DGIT_VERSION=\"$(GIT_VERSION)\"
|
CFLAGS += -DGIT_VERSION=\"$(GIT_VERSION)\"
|
||||||
CFLAGS += -DBOARD=\"$(BOARD)\" -DBOARD_$(BOARD)
|
CFLAGS += -DBOARD=\"$(BOARD)\" -DBOARD_$(BOARD)
|
||||||
CFLAGS += -DAPPLICATION=\"$(APP)\" -DAPPLICATION_$(APP)
|
CFLAGS += -DAPPLICATION=\"$(APP)\" -DAPPLICATION_$(APP)
|
||||||
@@ -171,8 +171,7 @@ ASFLAGS = -mcpu=cortex-m3 -mthumb -Wall -g $(OPTIMIZATION) $(INCLUDES) -D$(CHIP)
|
|||||||
LDFLAGS = -mcpu=cortex-m3 -mthumb -Wl,--cref -Wl,--check-sections -Wl,--gc-sections -Wl,--entry=ResetException -Wl,--unresolved-symbols=report-all -Wl,--warn-common -Wl,--warn-section-align -Wl,--warn-unresolved-symbols $(LIB)
|
LDFLAGS = -mcpu=cortex-m3 -mthumb -Wl,--cref -Wl,--check-sections -Wl,--gc-sections -Wl,--entry=ResetException -Wl,--unresolved-symbols=report-all -Wl,--warn-common -Wl,--warn-section-align -Wl,--warn-unresolved-symbols $(LIB)
|
||||||
#LD_OPTIONAL=-Wl,--print-gc-sections -Wl,--stats
|
#LD_OPTIONAL=-Wl,--print-gc-sections -Wl,--stats
|
||||||
|
|
||||||
|
# Append BIN directories to output filename
|
||||||
# Append OBJ and BIN directories to output filename
|
|
||||||
OUTPUT := $(BIN)/$(OUTPUT)
|
OUTPUT := $(BIN)/$(OUTPUT)
|
||||||
|
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
@@ -205,8 +204,12 @@ ASM_OBJECTS_$(1) = $(addprefix $(OBJ)/$(1)_, $(ASM_OBJECTS))
|
|||||||
|
|
||||||
$(1): $$(ASM_OBJECTS_$(1)) $$(C_OBJECTS_$(1))
|
$(1): $$(ASM_OBJECTS_$(1)) $$(C_OBJECTS_$(1))
|
||||||
@$(CC) $(LDFLAGS) $(LD_OPTIONAL) -T"libboard/common/resources/$(CHIP)/$$@.ld" -Wl,-Map,$(OUTPUT)-$$@.map -o $(OUTPUT)-$$@.elf $$^ $(LIBS)
|
@$(CC) $(LDFLAGS) $(LD_OPTIONAL) -T"libboard/common/resources/$(CHIP)/$$@.ld" -Wl,-Map,$(OUTPUT)-$$@.map -o $(OUTPUT)-$$@.elf $$^ $(LIBS)
|
||||||
|
cp $(OUTPUT)-$$@.elf $(OUTPUT)-$$@-$(GIT_VERSION).elf
|
||||||
|
cp $(OUTPUT)-$$@.elf $(OUTPUT)-$$@-latest.elf
|
||||||
@$(NM) $(OUTPUT)-$$@.elf >$(OUTPUT)-$$@.elf.txt
|
@$(NM) $(OUTPUT)-$$@.elf >$(OUTPUT)-$$@.elf.txt
|
||||||
@$(OBJCOPY) -O binary $(OUTPUT)-$$@.elf $(OUTPUT)-$$@.bin
|
@$(OBJCOPY) -O binary $(OUTPUT)-$$@.elf $(OUTPUT)-$$@.bin
|
||||||
|
cp $(OUTPUT)-$$@.bin $(OUTPUT)-$$@-$(GIT_VERSION).bin
|
||||||
|
cp $(OUTPUT)-$$@.bin $(OUTPUT)-$$@-latest.bin
|
||||||
@$(SIZE) $$^ $(OUTPUT)-$$@.elf
|
@$(SIZE) $$^ $(OUTPUT)-$$@.elf
|
||||||
|
|
||||||
$$(C_OBJECTS_$(1)): $(OBJ)/$(1)_%.o: %.c Makefile $(OBJ) $(BIN)
|
$$(C_OBJECTS_$(1)): $(OBJ)/$(1)_%.o: %.c Makefile $(OBJ) $(BIN)
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ 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
|
||||||
|
|
||||||
@@ -51,6 +52,7 @@ 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
|
||||||
|
|
||||||
@@ -76,6 +78,10 @@ $ 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/).
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
/* SIMtrace 2 firmware card emulation application
|
/* SIMtrace 2 firmware card emulation application
|
||||||
*
|
*
|
||||||
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
* (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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -101,7 +102,11 @@ static volatile enum confNum simtrace_config = CFG_NUM_CCID;
|
|||||||
void USBDDriverCallbacks_ConfigurationChanged(uint8_t cfgnum)
|
void USBDDriverCallbacks_ConfigurationChanged(uint8_t cfgnum)
|
||||||
{
|
{
|
||||||
TRACE_INFO_WP("cfgChanged%d ", cfgnum);
|
TRACE_INFO_WP("cfgChanged%d ", cfgnum);
|
||||||
simtrace_config = cfgnum;
|
if (cfgnum < ARRAY_SIZE(config_func_ptrs)) {
|
||||||
|
simtrace_config = cfgnum;
|
||||||
|
} else {
|
||||||
|
TRACE_ERROR("trying to set out of bounds config %u\r\n", cfgnum);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void USART1_IrqHandler(void)
|
void USART1_IrqHandler(void)
|
||||||
@@ -125,9 +130,8 @@ static void check_exec_dbg_cmd(void)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
ch = UART_GetChar();
|
ch = UART_GetChar();
|
||||||
/* We must echo the character to make python fdexpect happy, whcih we use in factory testing */
|
/* We must echo the character to make python fdexpect happy, which we use in factory testing */
|
||||||
fputc(ch, stdout);
|
fputc(ch, stdout);
|
||||||
|
|
||||||
board_exec_dbg_cmd(ch);
|
board_exec_dbg_cmd(ch);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -204,7 +208,9 @@ extern int main(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
TRACE_INFO("calling init of config %u...\n\r", simtrace_config);
|
TRACE_INFO("calling init of config %u...\n\r", simtrace_config);
|
||||||
config_func_ptrs[simtrace_config].init();
|
if (config_func_ptrs[simtrace_config].init) {
|
||||||
|
config_func_ptrs[simtrace_config].init();
|
||||||
|
}
|
||||||
last_simtrace_config = simtrace_config;
|
last_simtrace_config = simtrace_config;
|
||||||
|
|
||||||
TRACE_INFO("entering main loop...\n\r");
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
@@ -232,11 +238,17 @@ extern int main(void)
|
|||||||
if (last_simtrace_config != simtrace_config) {
|
if (last_simtrace_config != simtrace_config) {
|
||||||
TRACE_INFO("USB config chg %u -> %u\n\r",
|
TRACE_INFO("USB config chg %u -> %u\n\r",
|
||||||
last_simtrace_config, simtrace_config);
|
last_simtrace_config, simtrace_config);
|
||||||
config_func_ptrs[last_simtrace_config].exit();
|
if (config_func_ptrs[last_simtrace_config].exit) {
|
||||||
config_func_ptrs[simtrace_config].init();
|
config_func_ptrs[last_simtrace_config].exit();
|
||||||
|
}
|
||||||
|
if (config_func_ptrs[simtrace_config].init) {
|
||||||
|
config_func_ptrs[simtrace_config].init();
|
||||||
|
}
|
||||||
last_simtrace_config = simtrace_config;
|
last_simtrace_config = simtrace_config;
|
||||||
} else {
|
} else {
|
||||||
config_func_ptrs[simtrace_config].run();
|
if (config_func_ptrs[simtrace_config].run) {
|
||||||
|
config_func_ptrs[simtrace_config].run();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -256,7 +256,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%x\n\r", reset_cause);
|
TRACE_INFO("Reset Cause: 0x%lx\n\r", reset_cause);
|
||||||
|
|
||||||
#if (TRACE_LEVEL >= TRACE_LEVEL_INFO)
|
#if (TRACE_LEVEL >= TRACE_LEVEL_INFO)
|
||||||
/* Find out why we are in the DFU bootloader, and not the main application */
|
/* Find out why we are in the DFU bootloader, and not the main application */
|
||||||
|
|||||||
3
firmware/apps/freq_ctr/Makefile
Normal file
3
firmware/apps/freq_ctr/Makefile
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
|
C_FILES += freq_ctr.c
|
||||||
55
firmware/apps/freq_ctr/freq_ctr.c
Normal file
55
firmware/apps/freq_ctr/freq_ctr.c
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
#include <stdint.h>
|
||||||
|
#include "utils.h"
|
||||||
|
#include "tc_etu.h"
|
||||||
|
#include "chip.h"
|
||||||
|
|
||||||
|
|
||||||
|
/* pins for Channel 0 of TC-block 0 */
|
||||||
|
#define PIN_TIOA0 {PIO_PA0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/* pins for Channel 1 of TC-block 0 */
|
||||||
|
#define PIN_TIOA1 {PIO_PA15, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
#define PIN_TCLK1 {PIO_PA28, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
static const Pin pins_tc[] = { PIN_TIOA0, PIN_TIOA1, PIN_TCLK1 };
|
||||||
|
|
||||||
|
static TcChannel *tc1 = &TC0->TC_CHANNEL[1];
|
||||||
|
|
||||||
|
void TC1_IrqHandler(void)
|
||||||
|
{
|
||||||
|
uint32_t sr = tc1->TC_SR;
|
||||||
|
printf("TC1=%lu; SR=0x%08lx\r\n", tc1->TC_RA, sr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void freq_ctr_init(void)
|
||||||
|
{
|
||||||
|
TcChannel *tc0 = &TC0->TC_CHANNEL[0];
|
||||||
|
|
||||||
|
PIO_Configure(pins_tc, ARRAY_SIZE(pins_tc));
|
||||||
|
|
||||||
|
PMC_EnablePeripheral(ID_TC0);
|
||||||
|
PMC_EnablePeripheral(ID_TC1);
|
||||||
|
|
||||||
|
/* route TCLK1 to XC1 */
|
||||||
|
TC0->TC_BMR &= ~TC_BMR_TC1XC1S_Msk;
|
||||||
|
TC0->TC_BMR |= TC_BMR_TC1XC1S_TCLK1;
|
||||||
|
|
||||||
|
/* TC0 in wveform mode: Run from SCLK. Raise TIOA on RA; lower TIOA on RC + trigger */
|
||||||
|
tc0->TC_CMR = TC_CMR_TCCLKS_TIMER_CLOCK5 | TC_CMR_BURST_NONE |
|
||||||
|
TC_CMR_EEVTEDG_NONE | TC_CMR_WAVSEL_UP_RC | TC_CMR_WAVE |
|
||||||
|
TC_CMR_ACPA_SET | TC_CMR_ACPC_CLEAR;
|
||||||
|
tc0->TC_RA = 16384; /* set high at 16384 */
|
||||||
|
tc0->TC_RC = 32786; /* set low at 32786 */
|
||||||
|
|
||||||
|
/* TC1 in capture mode: Run from XC1. Trigger on TIOA rising. Load RA on rising */
|
||||||
|
tc1->TC_CMR = TC_CMR_TCCLKS_XC1 | TC_CMR_BURST_NONE |
|
||||||
|
TC_CMR_ETRGEDG_RISING | TC_CMR_ABETRG | TC_CMR_LDRA_RISING;
|
||||||
|
/* Interrupt us if the external trigger happens */
|
||||||
|
tc1->TC_IER = TC_IER_ETRGS;
|
||||||
|
NVIC_EnableIRQ(TC1_IRQn);
|
||||||
|
|
||||||
|
TC0->TC_BCR = TC_BCR_SYNC;
|
||||||
|
|
||||||
|
tc0->TC_CCR = TC_CCR_CLKEN|TC_CCR_SWTRG;
|
||||||
|
tc1->TC_CCR = TC_CCR_CLKEN|TC_CCR_SWTRG;
|
||||||
|
}
|
||||||
54
firmware/apps/freq_ctr/main.c
Normal file
54
firmware/apps/freq_ctr/main.c
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "osmocom/core/timer.h"
|
||||||
|
|
||||||
|
extern void freq_ctr_init(void);
|
||||||
|
|
||||||
|
/* returns '1' in case we should break any endless loop */
|
||||||
|
static void check_exec_dbg_cmd(void)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
if (!UART_IsRxReady())
|
||||||
|
return;
|
||||||
|
|
||||||
|
ch = UART_GetChar();
|
||||||
|
|
||||||
|
board_exec_dbg_cmd(ch);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
extern int main(void)
|
||||||
|
{
|
||||||
|
led_init();
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
|
||||||
|
/* Enable watchdog for 2000 ms, with no window */
|
||||||
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
|
||||||
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
|
|
||||||
|
printf("\n\r\n\r"
|
||||||
|
"=============================================================================\n\r"
|
||||||
|
"Freq Ctr firmware " GIT_VERSION " (C) 2019 by Harald Welte\n\r"
|
||||||
|
"=============================================================================\n\r");
|
||||||
|
|
||||||
|
board_main_top();
|
||||||
|
|
||||||
|
TRACE_INFO("starting frequency counter...\n\r");
|
||||||
|
freq_ctr_init();
|
||||||
|
|
||||||
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
|
while (1) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
check_exec_dbg_cmd();
|
||||||
|
osmo_timers_prepare();
|
||||||
|
osmo_timers_update();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
10
firmware/apps/freq_ctr/usb_strings.txt
Normal file
10
firmware/apps/freq_ctr/usb_strings.txt
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
sysmocom - s.f.m.c. GmbH
|
||||||
|
SIMtrace 2 compatible device
|
||||||
|
SIMtrace Sniffer
|
||||||
|
SIMtrace CCID
|
||||||
|
SIMtrace Phone
|
||||||
|
SIMtrace MITM
|
||||||
|
CardEmulator Modem 1
|
||||||
|
CardEmulator Modem 2
|
||||||
|
CardEmulator Modem 3
|
||||||
|
CardEmulator Modem 4
|
||||||
3
firmware/apps/gpio_test/Makefile
Normal file
3
firmware/apps/gpio_test/Makefile
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
C_FILES += $(C_LIBUSB_RT)
|
||||||
|
|
||||||
|
C_FILES += gpio_test.c
|
||||||
8
firmware/apps/gpio_test/gpio_test.c
Normal file
8
firmware/apps/gpio_test/gpio_test.c
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
#include <stdint.h>
|
||||||
|
#include "utils.h"
|
||||||
|
#include "chip.h"
|
||||||
|
|
||||||
|
void gpio_test_init(void)
|
||||||
|
{
|
||||||
|
printf("FIXME run tests here\n\r");
|
||||||
|
}
|
||||||
54
firmware/apps/gpio_test/main.c
Normal file
54
firmware/apps/gpio_test/main.c
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "osmocom/core/timer.h"
|
||||||
|
|
||||||
|
extern void gpio_test_init(void);
|
||||||
|
|
||||||
|
/* returns '1' in case we should break any endless loop */
|
||||||
|
static void check_exec_dbg_cmd(void)
|
||||||
|
{
|
||||||
|
int ch;
|
||||||
|
|
||||||
|
if (!UART_IsRxReady())
|
||||||
|
return;
|
||||||
|
|
||||||
|
ch = UART_GetChar();
|
||||||
|
|
||||||
|
board_exec_dbg_cmd(ch);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
extern int main(void)
|
||||||
|
{
|
||||||
|
led_init();
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
|
||||||
|
/* Enable watchdog for 2000 ms, with no window */
|
||||||
|
WDT_Enable(WDT, WDT_MR_WDRSTEN | WDT_MR_WDDBGHLT | WDT_MR_WDIDLEHLT |
|
||||||
|
(WDT_GetPeriod(2000) << 16) | WDT_GetPeriod(2000));
|
||||||
|
|
||||||
|
PIO_InitializeInterrupts(0);
|
||||||
|
|
||||||
|
|
||||||
|
printf("\n\r\n\r"
|
||||||
|
"=============================================================================\n\r"
|
||||||
|
"GPIO Test firmware " GIT_VERSION " (C) 2019 Sysmocom GmbH\n\r"
|
||||||
|
"=============================================================================\n\r");
|
||||||
|
|
||||||
|
board_main_top();
|
||||||
|
|
||||||
|
TRACE_INFO("starting gpio test...\n\r");
|
||||||
|
gpio_test_init();
|
||||||
|
|
||||||
|
TRACE_INFO("entering main loop...\n\r");
|
||||||
|
while (1) {
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
check_exec_dbg_cmd();
|
||||||
|
osmo_timers_prepare();
|
||||||
|
osmo_timers_update();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
10
firmware/apps/gpio_test/usb_strings.txt
Normal file
10
firmware/apps/gpio_test/usb_strings.txt
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
sysmocom - s.f.m.c. GmbH
|
||||||
|
SIMtrace 2 compatible device
|
||||||
|
SIMtrace Sniffer
|
||||||
|
SIMtrace CCID
|
||||||
|
SIMtrace Phone
|
||||||
|
SIMtrace MITM
|
||||||
|
CardEmulator Modem 1
|
||||||
|
CardEmulator Modem 2
|
||||||
|
CardEmulator Modem 3
|
||||||
|
CardEmulator Modem 4
|
||||||
@@ -101,7 +101,7 @@ struct dfudata {
|
|||||||
extern struct dfudata _g_dfu;
|
extern struct dfudata _g_dfu;
|
||||||
extern struct dfudata *g_dfu;
|
extern struct dfudata *g_dfu;
|
||||||
|
|
||||||
void set_usb_serial_str(const uint8_t *serial_usbstr);
|
void set_usb_serial_str(void);
|
||||||
|
|
||||||
void DFURT_SwitchToDFU(void);
|
void DFURT_SwitchToDFU(void);
|
||||||
|
|
||||||
|
|||||||
@@ -13,14 +13,87 @@
|
|||||||
#include <usb/common/dfu/usb_dfu.h>
|
#include <usb/common/dfu/usb_dfu.h>
|
||||||
#include <usb/device/dfu/dfu.h>
|
#include <usb/device/dfu/dfu.h>
|
||||||
|
|
||||||
|
#include "usb_strings_generated.h"
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
STR_MANUF = 1,
|
STR_MANUF = 1,
|
||||||
STR_PROD,
|
STR_PROD,
|
||||||
STR_CONFIG,
|
STR_CONFIG,
|
||||||
_STR_FIRST_ALT,
|
_STR_FIRST_ALT,
|
||||||
|
// serial string
|
||||||
STR_SERIAL = (_STR_FIRST_ALT+BOARD_DFU_NUM_IF),
|
STR_SERIAL = (_STR_FIRST_ALT+BOARD_DFU_NUM_IF),
|
||||||
|
// version string (on additional interface)
|
||||||
|
VERSION_CONF_STR,
|
||||||
|
VERSION_STR,
|
||||||
|
// count
|
||||||
|
STRING_DESC_CNT,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* USB string for the serial (using 128-bit device ID) */
|
||||||
|
static unsigned char usb_string_serial[] = {
|
||||||
|
USBStringDescriptor_LENGTH(32),
|
||||||
|
USBGenericDescriptor_STRING,
|
||||||
|
USBStringDescriptor_UNICODE('0'),
|
||||||
|
USBStringDescriptor_UNICODE('0'),
|
||||||
|
USBStringDescriptor_UNICODE('1'),
|
||||||
|
USBStringDescriptor_UNICODE('1'),
|
||||||
|
USBStringDescriptor_UNICODE('2'),
|
||||||
|
USBStringDescriptor_UNICODE('2'),
|
||||||
|
USBStringDescriptor_UNICODE('3'),
|
||||||
|
USBStringDescriptor_UNICODE('3'),
|
||||||
|
USBStringDescriptor_UNICODE('4'),
|
||||||
|
USBStringDescriptor_UNICODE('4'),
|
||||||
|
USBStringDescriptor_UNICODE('5'),
|
||||||
|
USBStringDescriptor_UNICODE('5'),
|
||||||
|
USBStringDescriptor_UNICODE('6'),
|
||||||
|
USBStringDescriptor_UNICODE('6'),
|
||||||
|
USBStringDescriptor_UNICODE('7'),
|
||||||
|
USBStringDescriptor_UNICODE('7'),
|
||||||
|
USBStringDescriptor_UNICODE('8'),
|
||||||
|
USBStringDescriptor_UNICODE('8'),
|
||||||
|
USBStringDescriptor_UNICODE('9'),
|
||||||
|
USBStringDescriptor_UNICODE('9'),
|
||||||
|
USBStringDescriptor_UNICODE('a'),
|
||||||
|
USBStringDescriptor_UNICODE('a'),
|
||||||
|
USBStringDescriptor_UNICODE('b'),
|
||||||
|
USBStringDescriptor_UNICODE('b'),
|
||||||
|
USBStringDescriptor_UNICODE('c'),
|
||||||
|
USBStringDescriptor_UNICODE('c'),
|
||||||
|
USBStringDescriptor_UNICODE('d'),
|
||||||
|
USBStringDescriptor_UNICODE('d'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE('f'),
|
||||||
|
USBStringDescriptor_UNICODE('f'),
|
||||||
|
};
|
||||||
|
|
||||||
|
/* USB string for the version */
|
||||||
|
static const unsigned char usb_string_version_conf[] = {
|
||||||
|
USBStringDescriptor_LENGTH(16),
|
||||||
|
USBGenericDescriptor_STRING,
|
||||||
|
USBStringDescriptor_UNICODE('f'),
|
||||||
|
USBStringDescriptor_UNICODE('i'),
|
||||||
|
USBStringDescriptor_UNICODE('r'),
|
||||||
|
USBStringDescriptor_UNICODE('m'),
|
||||||
|
USBStringDescriptor_UNICODE('w'),
|
||||||
|
USBStringDescriptor_UNICODE('a'),
|
||||||
|
USBStringDescriptor_UNICODE('r'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE(' '),
|
||||||
|
USBStringDescriptor_UNICODE('v'),
|
||||||
|
USBStringDescriptor_UNICODE('e'),
|
||||||
|
USBStringDescriptor_UNICODE('r'),
|
||||||
|
USBStringDescriptor_UNICODE('s'),
|
||||||
|
USBStringDescriptor_UNICODE('i'),
|
||||||
|
USBStringDescriptor_UNICODE('o'),
|
||||||
|
USBStringDescriptor_UNICODE('n'),
|
||||||
|
};
|
||||||
|
|
||||||
|
static const char git_version[] = GIT_VERSION;
|
||||||
|
static unsigned char usb_string_version[2 + ARRAY_SIZE(git_version) * 2 - 2];
|
||||||
|
/** array of static (from usb_strings) and runtime (serial, version) USB strings */
|
||||||
|
static const unsigned char *usb_strings_extended[STRING_DESC_CNT];
|
||||||
|
|
||||||
static const USBDeviceDescriptor fsDevice = {
|
static const USBDeviceDescriptor fsDevice = {
|
||||||
.bLength = sizeof(USBDeviceDescriptor),
|
.bLength = sizeof(USBDeviceDescriptor),
|
||||||
.bDescriptorType = USBGenericDescriptor_DEVICE,
|
.bDescriptorType = USBGenericDescriptor_DEVICE,
|
||||||
@@ -34,12 +107,8 @@ 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,
|
||||||
#else
|
.bNumConfigurations = 2, // DFU + version configurations
|
||||||
.iSerialNumber = 0,
|
|
||||||
#endif
|
|
||||||
.bNumConfigurations = 1,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Alternate Interface Descriptor, we use one per partition/memory type */
|
/* Alternate Interface Descriptor, we use one per partition/memory type */
|
||||||
@@ -85,17 +154,74 @@ const struct dfu_desc dfu_cfg_descriptor = {
|
|||||||
.func_dfu = DFU_FUNC_DESC
|
.func_dfu = DFU_FUNC_DESC
|
||||||
};
|
};
|
||||||
|
|
||||||
#include "usb_strings_generated.h"
|
void set_usb_serial_str(void)
|
||||||
|
|
||||||
#if 0
|
|
||||||
void set_usb_serial_str(const uint8_t *serial_usbstr)
|
|
||||||
{
|
{
|
||||||
usb_strings[STR_SERIAL] = serial_usbstr;
|
unsigned int i;
|
||||||
|
|
||||||
|
// put device ID into USB serial number description
|
||||||
|
unsigned int device_id[4];
|
||||||
|
EEFC_ReadUniqueID(device_id);
|
||||||
|
char device_id_string[32 + 1];
|
||||||
|
snprintf(device_id_string, ARRAY_SIZE(device_id_string), "%08x%08x%08x%08x",
|
||||||
|
device_id[0], device_id[1], device_id[2], device_id[3]);
|
||||||
|
for (i = 0; i < ARRAY_SIZE(device_id_string) - 1; i++) {
|
||||||
|
usb_string_serial[2 + 2 * i] = device_id_string[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// put version into USB string
|
||||||
|
usb_string_version[0] = USBStringDescriptor_LENGTH(ARRAY_SIZE(git_version) - 1);
|
||||||
|
usb_string_version[1] = USBGenericDescriptor_STRING;
|
||||||
|
for (i = 0; i < ARRAY_SIZE(git_version) - 1; i++) {
|
||||||
|
usb_string_version[2 + i * 2 + 0] = git_version[i];
|
||||||
|
usb_string_version[2 + i * 2 + 1] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// fill extended USB strings
|
||||||
|
for (i = 0; i < ARRAY_SIZE(usb_strings) && i < ARRAY_SIZE(usb_strings_extended); i++) {
|
||||||
|
usb_strings_extended[i] = usb_strings[i];
|
||||||
|
}
|
||||||
|
usb_strings_extended[STR_SERIAL] = usb_string_serial;
|
||||||
|
usb_strings_extended[VERSION_CONF_STR] = usb_string_version_conf;
|
||||||
|
usb_strings_extended[VERSION_STR] = usb_string_version;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
/* USB descriptor just to show the version */
|
||||||
|
typedef struct _SIMTraceDriverConfigurationDescriptorVersion {
|
||||||
|
/** Standard configuration descriptor. */
|
||||||
|
USBConfigurationDescriptor configuration;
|
||||||
|
USBInterfaceDescriptor version;
|
||||||
|
} __attribute__ ((packed)) SIMTraceDriverConfigurationDescriptorVersion;
|
||||||
|
|
||||||
|
static const SIMTraceDriverConfigurationDescriptorVersion
|
||||||
|
configurationDescriptorVersion = {
|
||||||
|
/* Standard configuration descriptor for the interface descriptor*/
|
||||||
|
.configuration = {
|
||||||
|
.bLength = sizeof(USBConfigurationDescriptor),
|
||||||
|
.bDescriptorType = USBGenericDescriptor_CONFIGURATION,
|
||||||
|
.wTotalLength = sizeof(SIMTraceDriverConfigurationDescriptorVersion),
|
||||||
|
.bNumInterfaces = 1,
|
||||||
|
.bConfigurationValue = 2,
|
||||||
|
.iConfiguration = VERSION_CONF_STR,
|
||||||
|
.bmAttributes = USBD_BMATTRIBUTES,
|
||||||
|
.bMaxPower = USBConfigurationDescriptor_POWER(100),
|
||||||
|
},
|
||||||
|
/* Interface standard descriptor just holding the version information */
|
||||||
|
.version = {
|
||||||
|
.bLength = sizeof(USBInterfaceDescriptor),
|
||||||
|
.bDescriptorType = USBGenericDescriptor_INTERFACE,
|
||||||
|
.bInterfaceNumber = 0,
|
||||||
|
.bAlternateSetting = 0,
|
||||||
|
.bNumEndpoints = 0,
|
||||||
|
.bInterfaceClass = USB_CLASS_PROPRIETARY,
|
||||||
|
.bInterfaceSubClass = 0xff,
|
||||||
|
.bInterfaceProtocol = 0,
|
||||||
|
.iInterface = VERSION_STR,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
static const USBConfigurationDescriptor *conf_desc_arr[] = {
|
static const USBConfigurationDescriptor *conf_desc_arr[] = {
|
||||||
&dfu_cfg_descriptor.ucfg,
|
&dfu_cfg_descriptor.ucfg,
|
||||||
|
&configurationDescriptorVersion.configuration,
|
||||||
};
|
};
|
||||||
|
|
||||||
const USBDDriverDescriptors dfu_descriptors = {
|
const USBDDriverDescriptors dfu_descriptors = {
|
||||||
@@ -108,6 +234,6 @@ const USBDDriverDescriptors dfu_descriptors = {
|
|||||||
.pHsConfiguration = NULL,
|
.pHsConfiguration = NULL,
|
||||||
.pHsQualifier = NULL,
|
.pHsQualifier = NULL,
|
||||||
.pHsOtherSpeed = NULL,
|
.pHsOtherSpeed = NULL,
|
||||||
.pStrings = usb_strings,
|
.pStrings = usb_strings_extended,
|
||||||
.numStrings = ARRAY_SIZE(usb_strings),
|
.numStrings = ARRAY_SIZE(usb_strings_extended),
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -447,6 +447,7 @@ void USBDFU_Initialize(const USBDDriverDescriptors *pDescriptors)
|
|||||||
/* We already start in DFU idle mode */
|
/* We already start in DFU idle mode */
|
||||||
g_dfu->state = DFU_STATE_dfuIDLE;
|
g_dfu->state = DFU_STATE_dfuIDLE;
|
||||||
|
|
||||||
|
set_usb_serial_str();
|
||||||
USBDDriver_Initialize(&usbdDriver, pDescriptors, if_altsettings);
|
USBDDriver_Initialize(&usbdDriver, pDescriptors, if_altsettings);
|
||||||
USBD_Init();
|
USBD_Init();
|
||||||
USBD_Connect();
|
USBD_Connect();
|
||||||
|
|||||||
@@ -1,4 +1,7 @@
|
|||||||
/* SIMtrace 2 common board pin definitions
|
/* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -53,18 +56,12 @@
|
|||||||
/** Core definition */
|
/** Core definition */
|
||||||
#define cortexm3
|
#define cortexm3
|
||||||
|
|
||||||
#define BOARD_MCK 48000000
|
/* LEDs are used to indicate the status
|
||||||
|
* the LED definition is board specific
|
||||||
#define PIO_LED_RED PIO_PA17
|
* most boards have two LEDs, one green and one red
|
||||||
#define PIO_LED_GREEN PIO_PA18
|
* the red LED indicates of the main firmware is ready (on) or if there is an error (blinking)
|
||||||
|
* the green LED indicates if the firmware is idling (on) or if there is activity (blinking)
|
||||||
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
*/
|
||||||
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
|
||||||
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
|
||||||
|
|
||||||
#define LED_NUM_RED 0
|
|
||||||
#define LED_NUM_GREEN 1
|
|
||||||
|
|
||||||
/** USART0 pin RX */
|
/** USART0 pin RX */
|
||||||
#define PIN_USART0_RXD {PIO_PA9A_URXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
#define PIN_USART0_RXD {PIO_PA9A_URXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
/** USART0 pin TX */
|
/** USART0 pin TX */
|
||||||
@@ -79,8 +76,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 baudrate always using 115200. */
|
/** Console baud rate in bps */
|
||||||
#define CONSOLE_BAUDRATE 115200
|
#define CONSOLE_BAUDRATE 921600
|
||||||
/** 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). */
|
||||||
@@ -111,9 +108,6 @@
|
|||||||
/* 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
|
|
||||||
|
|
||||||
// Board has UDP controller
|
// Board has UDP controller
|
||||||
#define BOARD_USB_UDP
|
#define BOARD_USB_UDP
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
|
* Copyright (c) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
@@ -46,39 +47,39 @@
|
|||||||
#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 \
|
|
||||||
| CKGR_PLLAR_MULA(13-1) \
|
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
|
||||||
| CKGR_PLLAR_DIVA(5))
|
|
||||||
#elif (BOARD_MAINOSC == 12000000)
|
|
||||||
/* QMod has 12 MHz clock, so multply by 8 (96 MHz) and divide by 2 */
|
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
| CKGR_PLLAR_MULA(8-1) \
|
| CKGR_PLLAR_MULA(8-1) \
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_DIVA(2))
|
| CKGR_PLLAR_DIVA(2))
|
||||||
#else
|
#elif (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 58000000)
|
||||||
#error "Please define PLLA config for your MAINOSC frequency"
|
|
||||||
#endif /* MAINOSC */
|
|
||||||
#elif (BOARD_MCK == 64000000)
|
|
||||||
#if (BOARD_MAINOSC == 18432000)
|
|
||||||
/* Clock settings at 64MHz for 18 MHz crystal: 64.512 MHz */
|
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
| CKGR_PLLAR_MULA(7-1) \
|
| CKGR_PLLAR_MULA(29-1) \
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_DIVA(2))
|
| CKGR_PLLAR_DIVA(6))
|
||||||
#elif (BOARD_MAINOSC == 12000000)
|
#elif (BOARD_MAINOSC == 12000000) && (BOARD_MCK == 60000000)
|
||||||
/* QMod has 12 MHz clock, so multply by 10 / div by 2: 60 MHz */
|
|
||||||
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
| CKGR_PLLAR_MULA(10-1) \
|
| CKGR_PLLAR_MULA(10-1) \
|
||||||
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
| CKGR_PLLAR_DIVA(2))
|
| CKGR_PLLAR_DIVA(2))
|
||||||
#error "Please define PLLA config for your MAINOSC frequency"
|
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 47923200)
|
||||||
#endif /* MAINOSC */
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
|
| CKGR_PLLAR_MULA(13-1) \
|
||||||
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
|
| CKGR_PLLAR_DIVA(5))
|
||||||
|
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 58982400)
|
||||||
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
|
| CKGR_PLLAR_MULA(16-1) \
|
||||||
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
|
| CKGR_PLLAR_DIVA(5))
|
||||||
|
#elif (BOARD_MAINOSC == 18432000) && (BOARD_MCK == 64512000)
|
||||||
|
#define BOARD_PLLAR (CKGR_PLLAR_STUCKTO1 \
|
||||||
|
| CKGR_PLLAR_MULA(7-1) \
|
||||||
|
| CKGR_PLLAR_PLLACOUNT(0x1) \
|
||||||
|
| 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)
|
||||||
@@ -126,6 +127,9 @@ extern WEAK void LowLevelInit( void )
|
|||||||
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;
|
||||||
|
|
||||||
|
/* disable ERASE pin to prevent accidental flash erase */
|
||||||
|
MATRIX->CCFG_SYSIO |= CCFG_SYSIO_SYSIO12;
|
||||||
|
|
||||||
/* enable both LED and green LED */
|
/* enable both LED and green LED */
|
||||||
PIOA->PIO_PER |= PIO_LED_RED | PIO_LED_GREEN;
|
PIOA->PIO_PER |= PIO_LED_RED | PIO_LED_GREEN;
|
||||||
PIOA->PIO_OER |= PIO_LED_RED | PIO_LED_GREEN;
|
PIOA->PIO_OER |= PIO_LED_RED | PIO_LED_GREEN;
|
||||||
|
|||||||
@@ -85,7 +85,12 @@ extern void UART_Configure( uint32_t baudrate, uint32_t masterClock)
|
|||||||
|
|
||||||
/* 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) {
|
||||||
|
pUart->UART_BRGR = ( masterClock / baudrate) / 16 + 1;
|
||||||
|
} else {
|
||||||
|
pUart->UART_BRGR = ( masterClock / baudrate) / 16 + 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* Disable PDC channel */
|
/* Disable PDC channel */
|
||||||
pUart->UART_PTCR = UART_PTCR_RXTDIS | UART_PTCR_TXTDIS;
|
pUart->UART_PTCR = UART_PTCR_RXTDIS | UART_PTCR_TXTDIS;
|
||||||
|
|||||||
170
firmware/libboard/octsimtest/include/board.h
Normal file
170
firmware/libboard/octsimtest/include/board.h
Normal file
@@ -0,0 +1,170 @@
|
|||||||
|
/* octSIMtest with SAM3S board definition
|
||||||
|
*
|
||||||
|
* (C) 2019 by sysmocom -s.f.m.c. GmbH, Author:Joachim Steiger <jsteiger@sysmocom.de>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
|
/* Name of the board */
|
||||||
|
#define BOARD_NAME "OCTSIMTEST"
|
||||||
|
/* Board definition */
|
||||||
|
#define octsimtest
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
|
#define BOARD_MAINOSC 18432000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58982400 // 18.432 * 16 / 5
|
||||||
|
|
||||||
|
/** Pin configuration **/
|
||||||
|
|
||||||
|
/** there is no red LED, but the code needs this second LED, thus we provide an unused pin */
|
||||||
|
#define PIO_LED_RED PIO_PB13
|
||||||
|
/** MCU pin connected to green LED, which is actually amber, and the logic is inverted since it is connected to an NPN transistor (used as open drain) */
|
||||||
|
#define PIO_LED_GREEN PIO_PA4
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOB, ID_PIOB, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/* Button to force bootloader start (shorted to ground when pressed */
|
||||||
|
#define PIN_BOOTLOADER_SW {PIO_PA5, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP}
|
||||||
|
|
||||||
|
//FIXME SIM_PWEN_PIN collides with PA5/bootloader_sw on octsimtest
|
||||||
|
/* Enable powering the card using the second 3.3 V output of the LDO (active high) */
|
||||||
|
#define SIM_PWEN_PIN {PIO_PA12, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Enable powering the SIM card */
|
||||||
|
#define PWR_PINS SIM_PWEN_PIN
|
||||||
|
|
||||||
|
// FIXME PA8 is 32khz xtal on octsimtest
|
||||||
|
/* Card presence pin */
|
||||||
|
#define SW_SIM PIO_PA11
|
||||||
|
/* Pull card presence pin high (shorted to ground in card slot when card is present) */
|
||||||
|
#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_DEGLITCH | PIO_IT_EDGE }
|
||||||
|
|
||||||
|
/** Smart card connection **/
|
||||||
|
//FIXME
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_SIM_RST {PIO_PA13, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Card I/O data signal input/output (I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO {PIO_PA6A_TXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Card CLK clock input (CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK {PIO_PA2B_SCK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin to measure card I/O timing (to start measuring the ETU on I/O activity; connected I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO_INPUT {PIO_PA1B_TIOB0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
//FIXME PIO_PA4B_TCLK0 PA4 is LED on octsimtest
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK_INPUT {PIO_PA14, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pins used to measure ETU timing (using timer counter) */
|
||||||
|
#define PINS_TC PIN_SIM_IO_INPUT, PIN_SIM_CLK_INPUT
|
||||||
|
|
||||||
|
/** Phone connection **/
|
||||||
|
/* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_VCC {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
/* Phone USIM slot 1 RST pin (active low; RST_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_nRST {PIO_PA24, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_RISE_EDGE | PIO_DEGLITCH }
|
||||||
|
/* Phone I/O data signal input/output (I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO {PIO_PA22A_TXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Phone CLK clock input (CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK {PIO_PA23A_SCK1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used for phone USIM slot 1 communication */
|
||||||
|
#define PINS_USIM1 PIN_PHONE_IO, PIN_PHONE_CLK, PIN_PHONE_CLK_INPUT, PIN_USIM1_VCC, PIN_PHONE_IO_INPUT, PIN_USIM1_nRST
|
||||||
|
/* Phone I/O data signal input/output (unused USART RX input; connected to I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO_INPUT {PIO_PA21A_RXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK_INPUT {PIO_PA29B_TCLK2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Default pin configuration **/
|
||||||
|
/* Disconnect VPP, CLK, and RST lines between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_SC_SW_DEFAULT {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect I/O line between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_IO_SW_DEFAULT {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_DEFAULT PIN_SC_SW_DEFAULT, PIN_IO_SW_DEFAULT
|
||||||
|
|
||||||
|
/** Sniffer configuration **/
|
||||||
|
/* Connect VPP, CLK, and RST lines between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_SC_SW_SNIFF {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect I/O line between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_IO_SW_SNIFF {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_SNIFF PIN_SC_SW_SNIFF, PIN_IO_SW_SNIFF
|
||||||
|
/* Card RST reset signal input (use as input since the phone will drive it) */
|
||||||
|
#define PIN_SIM_RST_SNIFF {PIO_PA7, PIOA, ID_PIOA, PIO_INPUT, PIO_DEGLITCH | PIO_IT_EDGE}
|
||||||
|
/* Pins used to sniff phone-card communication */
|
||||||
|
#define PINS_SIM_SNIFF PIN_SIM_IO, PIN_SIM_CLK, PIN_SIM_RST_SNIFF
|
||||||
|
/* Disable power converter 4.5-6V to 3.3V (active high) */
|
||||||
|
#define PIN_SIM_PWEN_SNIFF {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Enable power switch to forward VCC_PHONE to VCC_SIM (active high) */
|
||||||
|
#define PIN_VCC_FWD_SNIFF {VCC_FWD, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Use phone VCC to power card */
|
||||||
|
#define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF
|
||||||
|
|
||||||
|
/** CCID configuration */
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_ISO7816_RSTMC {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* ISO7816-communication related pins */
|
||||||
|
#define PINS_ISO7816 PIN_SIM_IO, PIN_SIM_CLK, PIN_ISO7816_RSTMC // SIM_PWEN_PIN, PIN_SIM_IO2, PIN_SIM_CLK2
|
||||||
|
|
||||||
|
/** External SPI flash interface **/
|
||||||
|
/* SPI MISO pin definition */
|
||||||
|
#define PIN_SPI_MISO {PIO_PA12A_MISO, PIOA, PIOA, PIO_PERIPH_A, PIO_PULLUP}
|
||||||
|
/* SPI MOSI pin definition */
|
||||||
|
#define PIN_SPI_MOSI {PIO_PA13A_MOSI, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI SCK pin definition */
|
||||||
|
#define PIN_SPI_SCK {PIO_PA14A_SPCK, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI pins definition. Contains MISO, MOSI & SCK */
|
||||||
|
#define PINS_SPI PIN_SPI_MISO, PIN_SPI_MOSI, PIN_SPI_SCK
|
||||||
|
/* SPI chip select 0 pin definition */
|
||||||
|
#define PIN_SPI_NPCS0 {PIO_PA11A_NPCS0, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI flash write protect pin (active low, pulled low) */
|
||||||
|
#define PIN_SPI_WP {PA15, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Pin configuration to control USB pull-up on D+
|
||||||
|
* @details the USB pull-up on D+ is enable by default on the board but can be disabled by setting PA16 high
|
||||||
|
*/
|
||||||
|
#define PIN_USB_PULLUP {PIO_PA16, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** USB definitions */
|
||||||
|
/* OpenMoko SIMtrace 2 USB vendor ID */
|
||||||
|
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO
|
||||||
|
/* OpenMoko SIMtrace 2 USB product ID (main application/runtime mode) */
|
||||||
|
#define BOARD_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2
|
||||||
|
/* OpenMoko SIMtrace 2 DFU USB product ID (DFU bootloader/DFU mode) */
|
||||||
|
#define BOARD_DFU_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2_DFU
|
||||||
|
/* USB release number (bcdDevice, shown as 0.00) */
|
||||||
|
#define BOARD_USB_RELEASE 0x000
|
||||||
|
/* Indicate SIMtrace is bus power in USB attributes */
|
||||||
|
#define BOARD_USB_BMATTRIBUTES USBConfigurationDescriptor_BUSPOWERED_NORWAKEUP
|
||||||
|
|
||||||
|
/** Supported modes */
|
||||||
|
/* SIMtrace board supports sniffer mode */
|
||||||
|
//#define HAVE_SNIFFER
|
||||||
|
/* SIMtrace board supports CCID mode */
|
||||||
|
//#define HAVE_CCID
|
||||||
|
/* SIMtrace board supports card emulation mode */
|
||||||
|
//#define HAVE_CARDEM
|
||||||
|
/* SIMtrace board supports man-in-the-middle mode */
|
||||||
|
//#define HAVE_MITM
|
||||||
|
/* octsimtest board supports gpio_test mode */
|
||||||
|
#define HAVE_GPIO_TEST
|
||||||
28
firmware/libboard/octsimtest/include/i2c.h
Normal file
28
firmware/libboard/octsimtest/include/i2c.h
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
/* I2C EEPROM memory read and write utilities
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
void i2c_pin_init(void);
|
||||||
|
|
||||||
|
bool i2c_write_byte(bool send_start, bool send_stop, uint8_t byte);
|
||||||
|
uint8_t i2c_read_byte(bool nack, bool send_stop);
|
||||||
|
void i2c_stop_cond(void);
|
||||||
|
|
||||||
|
int eeprom_write_byte(uint8_t slave, uint8_t addr, uint8_t byte);
|
||||||
|
int eeprom_read_byte(uint8_t slave, uint8_t addr);
|
||||||
25
firmware/libboard/octsimtest/include/mcp23017.h
Normal file
25
firmware/libboard/octsimtest/include/mcp23017.h
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
/* mcp23017 i2c gpio expander read and write utilities
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#define MCP23017_ADDRESS 0x20
|
||||||
|
|
||||||
|
int mcp23017_init(uint8_t slave);
|
||||||
|
int mcp23017_test(uint8_t slave);
|
||||||
|
int mcp23017_toggle(uint8_t slave);
|
||||||
|
//int mcp23017_write_byte(uint8_t slave, uint8_t addr, uint8_t byte);
|
||||||
|
//int mcp23017_read_byte(uint8_t slave, uint8_t addr);
|
||||||
81
firmware/libboard/octsimtest/source/board_octsimtest.c
Normal file
81
firmware/libboard/octsimtest/source/board_octsimtest.c
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
/* SIMtrace with SAM3S specific application code
|
||||||
|
*
|
||||||
|
* (C) 2017 by Harald Welte <laforge@gnumonks.org>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#include "board.h"
|
||||||
|
#include "simtrace.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "sim_switch.h"
|
||||||
|
#include <osmocom/core/timer.h>
|
||||||
|
#include "usb_buf.h"
|
||||||
|
#include "i2c.h"
|
||||||
|
#include "mcp23017.h"
|
||||||
|
|
||||||
|
void board_exec_dbg_cmd(int ch)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case '?':
|
||||||
|
printf("\t?\thelp\n\r");
|
||||||
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
printf("\tm\trun mcp23017 test\n\r");
|
||||||
|
printf("\tR\ttoggle MSB of gpio on mcp23017\n\r");
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
printf("Asking NVIC to reset us\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
break;
|
||||||
|
case 'm':
|
||||||
|
mcp23017_test(MCP23017_ADDRESS);
|
||||||
|
break;
|
||||||
|
case 't':
|
||||||
|
mcp23017_toggle(MCP23017_ADDRESS);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void board_main_top(void)
|
||||||
|
{
|
||||||
|
#ifndef APPLICATION_dfu
|
||||||
|
usb_buf_init();
|
||||||
|
|
||||||
|
i2c_pin_init();
|
||||||
|
if (!mcp23017_init(MCP23017_ADDRESS))
|
||||||
|
printf("mcp23017 not found!\n\r");
|
||||||
|
/* Initialize checking for card insert/remove events */
|
||||||
|
//card_present_init();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
const Pin bl_sw_pin = PIN_BOOTLOADER_SW;
|
||||||
|
|
||||||
|
PIO_Configure(&bl_sw_pin, 1);
|
||||||
|
|
||||||
|
/* Enter DFU bootloader in case the respective button is pressed */
|
||||||
|
if (PIO_Get(&bl_sw_pin) == 0) {
|
||||||
|
/* do not print to early since the console is not initialized yet */
|
||||||
|
//printf("BOOTLOADER switch pressed -> Force DFU\n\r");
|
||||||
|
return 1;
|
||||||
|
} else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
225
firmware/libboard/octsimtest/source/i2c.c
Normal file
225
firmware/libboard/octsimtest/source/i2c.c
Normal file
@@ -0,0 +1,225 @@
|
|||||||
|
/* I2C EEPROM memory read and write utilities
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#include "board.h"
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
/* Low-Level I2C Routines */
|
||||||
|
|
||||||
|
static const Pin pin_sda = {PIO_PA30, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_OPENDRAIN };
|
||||||
|
static const Pin pin_sda_in = {PIO_PA30, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT };
|
||||||
|
static const Pin pin_scl = {PIO_PA31, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_OPENDRAIN };
|
||||||
|
|
||||||
|
static void i2c_delay()
|
||||||
|
{
|
||||||
|
volatile int v;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* 100 cycles results in SCL peak length of 44us, so it's about
|
||||||
|
* 440ns per cycle here */
|
||||||
|
for (i = 0; i < 14; i++) {
|
||||||
|
v = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void i2c_pin_init(void)
|
||||||
|
{
|
||||||
|
PIO_Configure(&pin_scl, PIO_LISTSIZE(pin_scl));
|
||||||
|
PIO_Configure(&pin_sda, PIO_LISTSIZE(pin_sda));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void set_scl(void)
|
||||||
|
{
|
||||||
|
PIO_Set(&pin_scl);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void set_sda(void)
|
||||||
|
{
|
||||||
|
PIO_Set(&pin_sda);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void clear_scl(void)
|
||||||
|
{
|
||||||
|
PIO_Clear(&pin_scl);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void clear_sda(void)
|
||||||
|
{
|
||||||
|
PIO_Clear(&pin_sda);
|
||||||
|
i2c_delay();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool read_sda(void)
|
||||||
|
{
|
||||||
|
bool ret;
|
||||||
|
|
||||||
|
PIO_Configure(&pin_sda_in, PIO_LISTSIZE(pin_sda_in));
|
||||||
|
if (PIO_Get(&pin_sda_in))
|
||||||
|
ret = true;
|
||||||
|
else
|
||||||
|
ret = false;
|
||||||
|
PIO_Configure(&pin_sda, PIO_LISTSIZE(pin_sda));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Core I2C Routines */
|
||||||
|
|
||||||
|
static bool i2c_started = false;
|
||||||
|
|
||||||
|
static void i2c_start_cond(void)
|
||||||
|
{
|
||||||
|
if (i2c_started) {
|
||||||
|
set_sda();
|
||||||
|
set_scl();
|
||||||
|
}
|
||||||
|
|
||||||
|
clear_sda();
|
||||||
|
i2c_delay();
|
||||||
|
clear_scl();
|
||||||
|
i2c_started = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void i2c_stop_cond(void)
|
||||||
|
{
|
||||||
|
clear_sda();
|
||||||
|
set_scl();
|
||||||
|
set_sda();
|
||||||
|
i2c_delay();
|
||||||
|
i2c_started = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void i2c_write_bit(bool bit)
|
||||||
|
{
|
||||||
|
if (bit)
|
||||||
|
set_sda();
|
||||||
|
else
|
||||||
|
clear_sda();
|
||||||
|
i2c_delay(); // ?
|
||||||
|
set_scl();
|
||||||
|
clear_scl();
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool i2c_read_bit(void)
|
||||||
|
{
|
||||||
|
bool bit;
|
||||||
|
|
||||||
|
set_sda();
|
||||||
|
set_scl();
|
||||||
|
bit = read_sda();
|
||||||
|
clear_scl();
|
||||||
|
|
||||||
|
return bit;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool i2c_write_byte(bool send_start, bool send_stop, uint8_t byte)
|
||||||
|
{
|
||||||
|
uint8_t bit;
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
if (send_start)
|
||||||
|
i2c_start_cond();
|
||||||
|
|
||||||
|
for (bit = 0; bit < 8; bit++) {
|
||||||
|
i2c_write_bit((byte & 0x80) != 0);
|
||||||
|
byte <<= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
nack = i2c_read_bit();
|
||||||
|
|
||||||
|
if (send_stop)
|
||||||
|
i2c_stop_cond();
|
||||||
|
|
||||||
|
return nack;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t i2c_read_byte(bool nack, bool send_stop)
|
||||||
|
{
|
||||||
|
uint8_t byte = 0;
|
||||||
|
uint8_t bit;
|
||||||
|
|
||||||
|
for (bit = 0; bit < 8; bit++) {
|
||||||
|
byte = (byte << 1) | i2c_read_bit();
|
||||||
|
}
|
||||||
|
|
||||||
|
i2c_write_bit(nack);
|
||||||
|
|
||||||
|
if (send_stop)
|
||||||
|
i2c_stop_cond();
|
||||||
|
|
||||||
|
return byte;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* EEPROM related code */
|
||||||
|
|
||||||
|
int eeprom_write_byte(uint8_t slave, uint8_t addr, uint8_t byte)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
/* Write slave address */
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, true, byte);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
/* Wait tWR time to ensure EEPROM is writing correctly (tWR = 5 ms for AT24C02) */
|
||||||
|
mdelay(5);
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int eeprom_read_byte(uint8_t slave, uint8_t addr)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
/* dummy write cycle */
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
/* Re-start with read */
|
||||||
|
nack = i2c_write_byte(true, false, (slave << 1) | 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
|
||||||
|
return i2c_read_byte(true, true);
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
140
firmware/libboard/octsimtest/source/mcp23017.c
Normal file
140
firmware/libboard/octsimtest/source/mcp23017.c
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
#include "board.h"
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include "i2c.h"
|
||||||
|
#include "mcp23017.h"
|
||||||
|
|
||||||
|
|
||||||
|
//defines from https://github.com/adafruit/Adafruit-MCP23017-Arduino-Library/blob/master/Adafruit_MCP23017.h under BSD license
|
||||||
|
|
||||||
|
// registers
|
||||||
|
#define MCP23017_IODIRA 0x00
|
||||||
|
#define MCP23017_IPOLA 0x02
|
||||||
|
#define MCP23017_GPINTENA 0x04
|
||||||
|
#define MCP23017_DEFVALA 0x06
|
||||||
|
#define MCP23017_INTCONA 0x08
|
||||||
|
#define MCP23017_IOCONA 0x0A
|
||||||
|
#define MCP23017_GPPUA 0x0C
|
||||||
|
#define MCP23017_INTFA 0x0E
|
||||||
|
#define MCP23017_INTCAPA 0x10
|
||||||
|
#define MCP23017_GPIOA 0x12
|
||||||
|
#define MCP23017_OLATA 0x14
|
||||||
|
|
||||||
|
|
||||||
|
#define MCP23017_IODIRB 0x01
|
||||||
|
#define MCP23017_IPOLB 0x03
|
||||||
|
#define MCP23017_GPINTENB 0x05
|
||||||
|
#define MCP23017_DEFVALB 0x07
|
||||||
|
#define MCP23017_INTCONB 0x09
|
||||||
|
#define MCP23017_IOCONB 0x0B
|
||||||
|
#define MCP23017_GPPUB 0x0D
|
||||||
|
#define MCP23017_INTFB 0x0F
|
||||||
|
#define MCP23017_INTCAPB 0x11
|
||||||
|
#define MCP23017_GPIOB 0x13
|
||||||
|
#define MCP23017_OLATB 0x15
|
||||||
|
|
||||||
|
#define MCP23017_INT_ERR 255
|
||||||
|
|
||||||
|
|
||||||
|
//bool i2c_write_byte(bool send_start, bool send_stop, uint8_t byte)
|
||||||
|
//uint8_t i2c_read_byte(bool nack, bool send_stop)
|
||||||
|
//static void i2c_stop_cond(void)
|
||||||
|
|
||||||
|
int mcp23017_write_byte(uint8_t slave, uint8_t addr, uint8_t byte)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
// Write slave address
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, true, byte);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_read_byte(uint8_t slave, uint8_t addr)
|
||||||
|
{
|
||||||
|
bool nack;
|
||||||
|
|
||||||
|
WDT_Restart(WDT);
|
||||||
|
|
||||||
|
// dummy write cycle
|
||||||
|
nack = i2c_write_byte(true, false, slave << 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
nack = i2c_write_byte(false, false, addr);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
// Re-start with read
|
||||||
|
nack = i2c_write_byte(true, false, (slave << 1) | 1);
|
||||||
|
if (nack)
|
||||||
|
goto out_stop;
|
||||||
|
|
||||||
|
return i2c_read_byte(true, true);
|
||||||
|
|
||||||
|
out_stop:
|
||||||
|
i2c_stop_cond();
|
||||||
|
if (nack)
|
||||||
|
return -1;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_init(uint8_t slave)
|
||||||
|
{
|
||||||
|
printf("mcp23017_init\n\r");
|
||||||
|
// all gpio input
|
||||||
|
if (mcp23017_write_byte(slave, MCP23017_IODIRA, 0xff))
|
||||||
|
return false;
|
||||||
|
// msb of portb output, rest input
|
||||||
|
if (mcp23017_write_byte(slave, MCP23017_IODIRB, 0x7f))
|
||||||
|
return false;
|
||||||
|
if (mcp23017_write_byte(slave, MCP23017_IOCONA, 0x20)) //disable SEQOP (autoinc addressing)
|
||||||
|
return false;
|
||||||
|
printf("mcp23017 found\n\r");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_test(uint8_t slave)
|
||||||
|
{
|
||||||
|
printf("mcp23017_test\n\r");
|
||||||
|
printf("GPIOA 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_GPIOA));
|
||||||
|
printf("GPIOB 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_GPIOB));
|
||||||
|
printf("IODIRA 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IODIRA));
|
||||||
|
printf("IODIRB 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IODIRB));
|
||||||
|
printf("IOCONA 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IOCONA));
|
||||||
|
printf("IOCONB 0x%x\n\r", mcp23017_read_byte(slave,MCP23017_IOCONB));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mcp23017_toggle(uint8_t slave)
|
||||||
|
{
|
||||||
|
// example writing MSB of gpio
|
||||||
|
static bool foo=false;
|
||||||
|
if (foo)
|
||||||
|
{
|
||||||
|
printf("+\n\r");
|
||||||
|
mcp23017_write_byte(slave, MCP23017_OLATB, 0x80);
|
||||||
|
foo=false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf("-\n\r");
|
||||||
|
mcp23017_write_byte(slave, MCP23017_OLATB, 0x00);
|
||||||
|
foo=true;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
/* OWHW board definition
|
/* OWHW board definition
|
||||||
*
|
*
|
||||||
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
* (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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -25,7 +26,28 @@
|
|||||||
/** Board definition */
|
/** Board definition */
|
||||||
#define owhw
|
#define owhw
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
#define BOARD_MAINOSC 18432000
|
#define BOARD_MAINOSC 18432000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58982400 // 18.432 * 16 / 5
|
||||||
|
|
||||||
|
/** MCU pin connected to red LED */
|
||||||
|
#define PIO_LED_RED PIO_PA17
|
||||||
|
/** MCU pin connected to green LED */
|
||||||
|
#define PIO_LED_GREEN PIO_PA18
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/* pin connected to the SIMTRACE_BOOTLOADER signal. set high to force DFU bootloader start */
|
||||||
|
#define PIN_BOOTLOADER {PIO_PA31, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
|
||||||
/* USIM 2 interface (USART) */
|
/* USIM 2 interface (USART) */
|
||||||
#define PIN_USIM2_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
#define PIN_USIM2_CLK {PIO_PA2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
/* Card simulator specific functions
|
/* Card simulator specific functions
|
||||||
*
|
*
|
||||||
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
* (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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -20,9 +21,35 @@
|
|||||||
#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)
|
||||||
@@ -38,3 +65,16 @@ 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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -20,15 +20,34 @@
|
|||||||
#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}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
/* sysmocom quad-modem sysmoQMOD application code
|
/* sysmocom quad-modem sysmoQMOD application code
|
||||||
*
|
*
|
||||||
* (C) 2016-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
* (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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -19,6 +20,7 @@
|
|||||||
#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"
|
||||||
@@ -26,6 +28,7 @@
|
|||||||
#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};
|
||||||
@@ -44,6 +47,7 @@ static int qmod_sam3_is_12(void)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
const unsigned char __eeprom_bin[256] = {
|
const unsigned char __eeprom_bin[256] = {
|
||||||
USB_VENDOR_OPENMOKO & 0xff,
|
USB_VENDOR_OPENMOKO & 0xff,
|
||||||
USB_VENDOR_OPENMOKO >> 8,
|
USB_VENDOR_OPENMOKO >> 8,
|
||||||
@@ -67,7 +71,6 @@ 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)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
@@ -90,11 +93,12 @@ static int write_hub_eeprom(void)
|
|||||||
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 < ARRAY_SIZE(__eeprom_bin); i++) {
|
||||||
int byte = eeprom_read_byte(0x50, i);
|
int byte = eeprom_read_byte(0x50, i);
|
||||||
TRACE_INFO("0x%02x: %02x\n\r", i, byte);
|
TRACE_DEBUG("0x%02x: %02x\n\r", i, byte);
|
||||||
if (byte != __eeprom_bin[i])
|
if (byte != __eeprom_bin[i])
|
||||||
TRACE_ERROR("Byte %u is wrong, expected 0x%02x, found 0x%02x\n\r",
|
TRACE_ERROR("Byte %u is wrong, expected 0x%02x, found 0x%02x\n\r",
|
||||||
i, __eeprom_bin[i], byte);
|
i, __eeprom_bin[i], byte);
|
||||||
}
|
}
|
||||||
|
TRACE_INFO("EEPROM written\n\r");
|
||||||
|
|
||||||
/* FIXME: Release PIN_PRTPWR_OVERRIDE after we know the hub is
|
/* FIXME: Release PIN_PRTPWR_OVERRIDE after we know the hub is
|
||||||
* again powering us up */
|
* again powering us up */
|
||||||
@@ -119,10 +123,11 @@ static int erase_hub_eeprom(void)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
TRACE_INFO("EEPROM erased\n\r");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
#endif /* ALLOW_PEER_ERASE */
|
||||||
|
|
||||||
static void board_exec_dbg_cmd_st12only(int ch)
|
static void board_exec_dbg_cmd_st12only(int ch)
|
||||||
{
|
{
|
||||||
@@ -133,12 +138,14 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
switch (ch) {
|
switch (ch) {
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
case 'E':
|
case 'E':
|
||||||
write_hub_eeprom();
|
write_hub_eeprom();
|
||||||
break;
|
break;
|
||||||
case 'e':
|
case 'e':
|
||||||
erase_hub_eeprom();
|
erase_hub_eeprom();
|
||||||
break;
|
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);
|
||||||
@@ -147,6 +154,7 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
printf("Clearing PRTPWR_OVERRIDE\n\r");
|
printf("Clearing PRTPWR_OVERRIDE\n\r");
|
||||||
PIO_Clear(&pin_hubpwr_override);
|
PIO_Clear(&pin_hubpwr_override);
|
||||||
break;
|
break;
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
case 'H':
|
case 'H':
|
||||||
printf("Clearing _HUB_RESET -> HUB_RESET high (inactive)\n\r");
|
printf("Clearing _HUB_RESET -> HUB_RESET high (inactive)\n\r");
|
||||||
PIO_Clear(&pin_hub_rst);
|
PIO_Clear(&pin_hub_rst);
|
||||||
@@ -166,6 +174,7 @@ static void board_exec_dbg_cmd_st12only(int ch)
|
|||||||
printf("Writing value 0x%02lx to EEPROM offset 0x%02lx\n\r", val, addr);
|
printf("Writing value 0x%02lx to EEPROM offset 0x%02lx\n\r", val, addr);
|
||||||
eeprom_write_byte(0x50, addr, val);
|
eeprom_write_byte(0x50, addr, val);
|
||||||
break;
|
break;
|
||||||
|
#endif /* ALLOW_PEER_ERASE */
|
||||||
case 'r':
|
case 'r':
|
||||||
printf("Please enter EEPROM offset:\n\r");
|
printf("Please enter EEPROM offset:\n\r");
|
||||||
UART_GetIntegerMinMax(&addr, 0, 255);
|
UART_GetIntegerMinMax(&addr, 0, 255);
|
||||||
@@ -180,10 +189,21 @@ 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("\te\tErase EEPROM\n\r");
|
||||||
@@ -196,8 +216,11 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
}
|
}
|
||||||
printf("\tX\tRelease peer SAM3 from reset\n\r");
|
printf("\tX\tRelease peer SAM3 from reset\n\r");
|
||||||
printf("\tx\tAssert peer SAM3 reset\n\r");
|
printf("\tx\tAssert peer SAM3 reset\n\r");
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
printf("\tY\tRelease peer SAM3 ERASE signal\n\r");
|
printf("\tY\tRelease peer SAM3 ERASE signal\n\r");
|
||||||
|
printf("\ta\tAllow asserting peer SAM3 ERASE signal\n\r");
|
||||||
printf("\ty\tAssert peer SAM3 ERASE signal\n\r");
|
printf("\ty\tAssert peer SAM3 ERASE signal\n\r");
|
||||||
|
#endif
|
||||||
printf("\tU\tProceed to USB Initialization\n\r");
|
printf("\tU\tProceed to USB Initialization\n\r");
|
||||||
printf("\t1\tGenerate 1ms reset pulse on WWAN1\n\r");
|
printf("\t1\tGenerate 1ms reset pulse on WWAN1\n\r");
|
||||||
printf("\t2\tGenerate 1ms reset pulse on WWAN2\n\r");
|
printf("\t2\tGenerate 1ms reset pulse on WWAN2\n\r");
|
||||||
@@ -207,6 +230,22 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
USBD_Disconnect();
|
USBD_Disconnect();
|
||||||
NVIC_SystemReset();
|
NVIC_SystemReset();
|
||||||
break;
|
break;
|
||||||
|
case 'l':
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_OFF);
|
||||||
|
printf("LED 1 switched off\n\r");
|
||||||
|
break;
|
||||||
|
case 'L':
|
||||||
|
led_blink(LED_GREEN, BLINK_ALWAYS_ON);
|
||||||
|
printf("LED 1 switched on\n\r");
|
||||||
|
break;
|
||||||
|
case 'g':
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_OFF);
|
||||||
|
printf("LED 2 switched off\n\r");
|
||||||
|
break;
|
||||||
|
case 'G':
|
||||||
|
led_blink(LED_RED, BLINK_ALWAYS_ON);
|
||||||
|
printf("LED 2 switched on\n\r");
|
||||||
|
break;
|
||||||
case 'X':
|
case 'X':
|
||||||
printf("Clearing _SIMTRACExx_RST -> SIMTRACExx_RST high (inactive)\n\r");
|
printf("Clearing _SIMTRACExx_RST -> SIMTRACExx_RST high (inactive)\n\r");
|
||||||
PIO_Clear(&pin_peer_rst);
|
PIO_Clear(&pin_peer_rst);
|
||||||
@@ -215,14 +254,24 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
printf("Setting _SIMTRACExx_RST -> SIMTRACExx_RST low (active)\n\r");
|
printf("Setting _SIMTRACExx_RST -> SIMTRACExx_RST low (active)\n\r");
|
||||||
PIO_Set(&pin_peer_rst);
|
PIO_Set(&pin_peer_rst);
|
||||||
break;
|
break;
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
case 'Y':
|
case 'Y':
|
||||||
printf("Clearing SIMTRACExx_ERASE (inactive)\n\r");
|
printf("Clearing SIMTRACExx_ERASE (inactive)\n\r");
|
||||||
PIO_Clear(&pin_peer_erase);
|
PIO_Clear(&pin_peer_erase);
|
||||||
break;
|
break;
|
||||||
case 'y':
|
case 'a':
|
||||||
printf("Seetting SIMTRACExx_ERASE (active)\n\r");
|
printf("Asserting SIMTRACExx_ERASE allowed on next command\n\r");
|
||||||
PIO_Set(&pin_peer_erase);
|
allow_erase = true;
|
||||||
break;
|
break;
|
||||||
|
case 'y':
|
||||||
|
if (allow_erase) {
|
||||||
|
printf("Setting SIMTRACExx_ERASE (active)\n\r");
|
||||||
|
PIO_Set(&pin_peer_erase);
|
||||||
|
} else {
|
||||||
|
printf("Please first allow setting SIMTRACExx_ERASE\n\r");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
case '1':
|
case '1':
|
||||||
printf("Resetting Modem 1 (of this SAM3)\n\r");
|
printf("Resetting Modem 1 (of this SAM3)\n\r");
|
||||||
wwan_perst_do_reset_pulse(0, 300);
|
wwan_perst_do_reset_pulse(0, 300);
|
||||||
@@ -244,6 +293,13 @@ void board_exec_dbg_cmd(int ch)
|
|||||||
board_exec_dbg_cmd_st12only(ch);
|
board_exec_dbg_cmd_st12only(ch);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if (ALLOW_PEER_ERASE > 0)
|
||||||
|
// set protection back so it can only run for one command
|
||||||
|
if ('a' != ch) {
|
||||||
|
allow_erase = false;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void board_main_top(void)
|
void board_main_top(void)
|
||||||
|
|||||||
163
firmware/libboard/sam3p256/include/board.h
Normal file
163
firmware/libboard/sam3p256/include/board.h
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
/* Olimiex SAM3S-P256 board definition
|
||||||
|
*
|
||||||
|
* (C) 2019 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
#include "board_common.h"
|
||||||
|
#include "simtrace_usb.h"
|
||||||
|
|
||||||
|
/* Name of the board */
|
||||||
|
#define BOARD_NAME "SAM3S-P256"
|
||||||
|
/* Board definition */
|
||||||
|
#define simtrace
|
||||||
|
|
||||||
|
/** oscillator used as main clock source (in Hz) */
|
||||||
|
#define BOARD_MAINOSC 12000000
|
||||||
|
/** desired main clock frequency (in Hz, based on BOARD_MAINOSC) */
|
||||||
|
#define BOARD_MCK 58000000
|
||||||
|
|
||||||
|
/** MCU pin connected to yellow LED2 */
|
||||||
|
#define PIO_LED_RED PIO_PA17
|
||||||
|
/** MCU pin connected to green LED1 */
|
||||||
|
#define PIO_LED_GREEN PIO_PA18
|
||||||
|
/** red LED pin definition */
|
||||||
|
#define PIN_LED_RED {PIO_LED_RED, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** green LED pin definition */
|
||||||
|
#define PIN_LED_GREEN {PIO_LED_GREEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/** LEDs pin definition */
|
||||||
|
#define PINS_LEDS PIN_LED_RED, PIN_LED_GREEN
|
||||||
|
/** index for red LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_RED 0
|
||||||
|
/** index for green LED in LEDs pin definition array */
|
||||||
|
#define LED_NUM_GREEN 1
|
||||||
|
|
||||||
|
/** Pin configuration **/
|
||||||
|
/* Button to force bootloader start (shorted to ground when pressed */
|
||||||
|
#define PIN_BOOTLOADER_SW {PIO_PA20, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
#if 0
|
||||||
|
/* Enable powering the card using the second 3.3 V output of the LDO (active high) */
|
||||||
|
#define SIM_PWEN_PIN {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Enable powering the SIM card */
|
||||||
|
#define PWR_PINS SIM_PWEN_PIN
|
||||||
|
/* Card presence pin */
|
||||||
|
#define SW_SIM PIO_PA8
|
||||||
|
/* Pull card presence pin high (shorted to ground in card slot when card is present) */
|
||||||
|
#define SMARTCARD_CONNECT_PIN {SW_SIM, PIOA, ID_PIOA, PIO_INPUT, PIO_PULLUP | PIO_DEBOUNCE | PIO_DEGLITCH | PIO_IT_EDGE }
|
||||||
|
|
||||||
|
/** Smart card connection **/
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_SIM_RST {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Card I/O data signal input/output (I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO {PIO_PA6A_TXD0, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Card CLK clock input (CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK {PIO_PA2B_SCK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin to measure card I/O timing (to start measuring the ETU on I/O activity; connected I/O_SIM in schematic) */
|
||||||
|
#define PIN_SIM_IO_INPUT {PIO_PA1B_TIOB0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_SIM in schematic) */
|
||||||
|
#define PIN_SIM_CLK_INPUT {PIO_PA4B_TCLK0, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
/* Pins used to measure ETU timing (using timer counter) */
|
||||||
|
#define PINS_TC PIN_SIM_IO_INPUT, PIN_SIM_CLK_INPUT
|
||||||
|
|
||||||
|
/** Phone connection **/
|
||||||
|
/* Phone USIM slot 1 VCC pin (VCC_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_VCC {PIO_PA25, PIOA, ID_PIOA, PIO_INPUT, PIO_DEFAULT}
|
||||||
|
/* Phone USIM slot 1 RST pin (active low; RST_PHONE in schematic) */
|
||||||
|
#define PIN_USIM1_nRST {PIO_PA24, PIOA, ID_PIOA, PIO_INPUT, PIO_IT_RISE_EDGE | PIO_DEGLITCH }
|
||||||
|
/* Phone I/O data signal input/output (I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO {PIO_PA22A_TXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Phone CLK clock input (CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK {PIO_PA23A_SCK1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used for phone USIM slot 1 communication */
|
||||||
|
#define PINS_USIM1 PIN_PHONE_IO, PIN_PHONE_CLK, PIN_PHONE_CLK_INPUT, PIN_USIM1_VCC, PIN_PHONE_IO_INPUT, PIN_USIM1_nRST
|
||||||
|
/* Phone I/O data signal input/output (unused USART RX input; connected to I/O_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_IO_INPUT {PIO_PA21A_RXD1, PIOA, ID_PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* Pin used as clock input (to measure the ETU duration; connected to CLK_PHONE in schematic) */
|
||||||
|
#define PIN_PHONE_CLK_INPUT {PIO_PA29B_TCLK2, PIOA, ID_PIOA, PIO_PERIPH_B, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** Default pin configuration **/
|
||||||
|
/* Disconnect VPP, CLK, and RST lines between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_SC_SW_DEFAULT {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect I/O line between card and phone using bus switch (high sets bus switch to high-impedance) */
|
||||||
|
#define PIN_IO_SW_DEFAULT {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Disconnect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_DEFAULT PIN_SC_SW_DEFAULT, PIN_IO_SW_DEFAULT
|
||||||
|
|
||||||
|
/** Sniffer configuration **/
|
||||||
|
/* Connect VPP, CLK, and RST lines between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_SC_SW_SNIFF {PIO_PA20, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect I/O line between card and phone using bus switch (low connects signals on bus switch) */
|
||||||
|
#define PIN_IO_SW_SNIFF {PIO_PA19, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Connect all lines (VPP, CLK, RST, and I/O) between card and phone */
|
||||||
|
#define PINS_BUS_SNIFF PIN_SC_SW_SNIFF, PIN_IO_SW_SNIFF
|
||||||
|
/* Card RST reset signal input (use as input since the phone will drive it) */
|
||||||
|
#define PIN_SIM_RST_SNIFF {PIO_PA7, PIOA, ID_PIOA, PIO_INPUT, PIO_DEGLITCH | PIO_IT_EDGE}
|
||||||
|
/* Pins used to sniff phone-card communication */
|
||||||
|
#define PINS_SIM_SNIFF PIN_SIM_IO, PIN_SIM_CLK, PIN_SIM_RST_SNIFF
|
||||||
|
/* Disable power converter 4.5-6V to 3.3V (active high) */
|
||||||
|
#define PIN_SIM_PWEN_SNIFF {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* Enable power switch to forward VCC_PHONE to VCC_SIM (active high) */
|
||||||
|
#define PIN_VCC_FWD_SNIFF {VCC_FWD, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
||||||
|
/* Use phone VCC to power card */
|
||||||
|
#define PINS_PWR_SNIFF PIN_SIM_PWEN_SNIFF, PIN_VCC_FWD_SNIFF
|
||||||
|
|
||||||
|
/** CCID configuration */
|
||||||
|
/* Card RST reset signal input (active low; RST_SIM in schematic) */
|
||||||
|
#define PIN_ISO7816_RSTMC {PIO_PA7, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
/* ISO7816-communication related pins */
|
||||||
|
#define PINS_ISO7816 PIN_SIM_IO, PIN_SIM_CLK, PIN_ISO7816_RSTMC // SIM_PWEN_PIN, PIN_SIM_IO2, PIN_SIM_CLK2
|
||||||
|
|
||||||
|
/** External SPI flash interface **/
|
||||||
|
/* SPI MISO pin definition */
|
||||||
|
#define PIN_SPI_MISO {PIO_PA12A_MISO, PIOA, PIOA, PIO_PERIPH_A, PIO_PULLUP}
|
||||||
|
/* SPI MOSI pin definition */
|
||||||
|
#define PIN_SPI_MOSI {PIO_PA13A_MOSI, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI SCK pin definition */
|
||||||
|
#define PIN_SPI_SCK {PIO_PA14A_SPCK, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI pins definition. Contains MISO, MOSI & SCK */
|
||||||
|
#define PINS_SPI PIN_SPI_MISO, PIN_SPI_MOSI, PIN_SPI_SCK
|
||||||
|
/* SPI chip select 0 pin definition */
|
||||||
|
#define PIN_SPI_NPCS0 {PIO_PA11A_NPCS0, PIOA, PIOA, PIO_PERIPH_A, PIO_DEFAULT}
|
||||||
|
/* SPI flash write protect pin (active low, pulled low) */
|
||||||
|
#define PIN_SPI_WP {PA15, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/** Pin configuration to control USB pull-up on D+
|
||||||
|
* @details the USB pull-up on D+ is enable by default on the board but can be disabled by setting PA16 high
|
||||||
|
*/
|
||||||
|
#define PIN_USB_PULLUP {PIO_PA16, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
|
|
||||||
|
/** USB definitions */
|
||||||
|
/* OpenMoko SIMtrace 2 USB vendor ID */
|
||||||
|
#define BOARD_USB_VENDOR_ID USB_VENDOR_OPENMOKO
|
||||||
|
/* OpenMoko SIMtrace 2 USB product ID (main application/runtime mode) */
|
||||||
|
#define BOARD_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2
|
||||||
|
/* OpenMoko SIMtrace 2 DFU USB product ID (DFU bootloader/DFU mode) */
|
||||||
|
#define BOARD_DFU_USB_PRODUCT_ID USB_PRODUCT_SIMTRACE2_DFU
|
||||||
|
/* USB release number (bcdDevice, shown as 0.00) */
|
||||||
|
#define BOARD_USB_RELEASE 0x000
|
||||||
|
/* Indicate SIMtrace is bus power in USB attributes */
|
||||||
|
#define BOARD_USB_BMATTRIBUTES USBConfigurationDescriptor_BUSPOWERED_NORWAKEUP
|
||||||
|
|
||||||
|
/** Supported modes */
|
||||||
|
/* SIMtrace board supports sniffer mode */
|
||||||
|
#define HAVE_SNIFFER
|
||||||
|
/* SIMtrace board supports CCID mode */
|
||||||
|
//#define HAVE_CCID
|
||||||
|
/* SIMtrace board supports card emulation mode */
|
||||||
|
//#define HAVE_CARDEM
|
||||||
|
/* SIMtrace board supports man-in-the-middle mode */
|
||||||
|
//#define HAVE_MITM
|
||||||
68
firmware/libboard/sam3p256/source/board_sam3p256.c
Normal file
68
firmware/libboard/sam3p256/source/board_sam3p256.c
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/* Olimex SAM3S-P256 specific application code
|
||||||
|
*
|
||||||
|
* (C) 2017,2019 by Harald Welte <laforge@gnumonks.org>
|
||||||
|
* (C) 2018 by sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
|
||||||
|
*/
|
||||||
|
#include "board.h"
|
||||||
|
#include "simtrace.h"
|
||||||
|
#include "utils.h"
|
||||||
|
#include "sim_switch.h"
|
||||||
|
#include <osmocom/core/timer.h>
|
||||||
|
#include "usb_buf.h"
|
||||||
|
|
||||||
|
void board_exec_dbg_cmd(int ch)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case '?':
|
||||||
|
printf("\t?\thelp\n\r");
|
||||||
|
printf("\tR\treset SAM3\n\r");
|
||||||
|
break;
|
||||||
|
case 'R':
|
||||||
|
printf("Asking NVIC to reset us\n\r");
|
||||||
|
USBD_Disconnect();
|
||||||
|
NVIC_SystemReset();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("Unknown command '%c'\n\r", ch);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void board_main_top(void)
|
||||||
|
{
|
||||||
|
#ifndef APPLICATION_dfu
|
||||||
|
usb_buf_init();
|
||||||
|
|
||||||
|
/* Initialize checking for card insert/remove events */
|
||||||
|
//card_present_init();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
int board_override_enter_dfu(void)
|
||||||
|
{
|
||||||
|
const Pin bl_sw_pin = PIN_BOOTLOADER_SW;
|
||||||
|
|
||||||
|
PIO_Configure(&bl_sw_pin, 1);
|
||||||
|
|
||||||
|
/* Enter DFU bootloader in case the respective button is pressed */
|
||||||
|
if (PIO_Get(&bl_sw_pin) == 0) {
|
||||||
|
/* do not print to early since the console is not initialized yet */
|
||||||
|
//printf("BOOTLOADER switch pressed -> Force DFU\n\r");
|
||||||
|
return 1;
|
||||||
|
} else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -26,14 +26,31 @@
|
|||||||
/* Board definition */
|
/* Board definition */
|
||||||
#define simtrace
|
#define simtrace
|
||||||
|
|
||||||
/* Board main oscillator frequency (in Hz) */
|
/** 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 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 {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_1, PIO_DEFAULT}
|
#define SIM_PWEN_PIN {PIO_PA5, 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 */
|
||||||
@@ -91,9 +108,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 {SIM_PWEN, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
#define PIN_SIM_PWEN_SNIFF {PIO_PA5, PIOA, ID_PIOA, PIO_OUTPUT_0, PIO_DEFAULT}
|
||||||
/* Enable power switch to forward VCC_PHONE to VCC_SIM (active high) */
|
/* 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}
|
#define PIN_VCC_FWD_SNIFF {PIO_PA26, 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
|
||||||
|
|
||||||
|
|||||||
@@ -29,8 +29,18 @@ enum card_io {
|
|||||||
CARD_IO_CLK,
|
CARD_IO_CLK,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan,
|
/** initialise card slot
|
||||||
uint8_t in_ep, uint8_t irq_ep);
|
* @param[in] slot_num slot number (arbitrary number)
|
||||||
|
* @param[in] tc_chan timer counter channel (to measure the ETU)
|
||||||
|
* @param[in] uart_chan UART peripheral channel
|
||||||
|
* @param[in] in_ep USB IN end point number
|
||||||
|
* @param[in] irq_ep USB INTerrupt end point number
|
||||||
|
* @param[in] vcc_active initial VCC signal state (true = on)
|
||||||
|
* @param[in] in_reset initial RST signal state (true = reset asserted)
|
||||||
|
* @param[in] clocked initial CLK signat state (true = active)
|
||||||
|
* @return main card handle reference
|
||||||
|
*/
|
||||||
|
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan, uint8_t in_ep, uint8_t irq_ep, bool vcc_active, bool in_reset, bool clocked);
|
||||||
|
|
||||||
/* process a single byte received from the reader */
|
/* process a single byte received from the reader */
|
||||||
void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte);
|
void card_emu_process_rx_byte(struct card_handle *ch, uint8_t byte);
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
/* SIMtrace 2 mode definitions
|
/* SIMtrace 2 mode definitions
|
||||||
*
|
*
|
||||||
* (C) 2015-2017 by Harald Welte <hwelte@hmw-consulting.de>
|
* 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
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -57,6 +58,7 @@ enum confNum {
|
|||||||
#ifdef HAVE_MITM
|
#ifdef HAVE_MITM
|
||||||
CFG_NUM_MITM,
|
CFG_NUM_MITM,
|
||||||
#endif
|
#endif
|
||||||
|
CFG_NUM_VERSION,
|
||||||
NUM_CONF
|
NUM_CONF
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -55,14 +55,38 @@ enum iso7816_3_card_state {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const struct value_string iso7816_3_card_state_names[] = {
|
const struct value_string iso7816_3_card_state_names[] = {
|
||||||
OSMO_VALUE_STRING(ISO_S_WAIT_POWER),
|
{
|
||||||
OSMO_VALUE_STRING(ISO_S_WAIT_CLK),
|
.value = ISO_S_WAIT_POWER,
|
||||||
OSMO_VALUE_STRING(ISO_S_WAIT_RST),
|
.str = "WAIT_POWER",
|
||||||
OSMO_VALUE_STRING(ISO_S_WAIT_ATR),
|
},
|
||||||
OSMO_VALUE_STRING(ISO_S_IN_ATR),
|
{
|
||||||
OSMO_VALUE_STRING(ISO_S_IN_PTS),
|
.value = ISO_S_WAIT_CLK,
|
||||||
OSMO_VALUE_STRING(ISO_S_WAIT_TPDU),
|
.str = "WAIT_CLK",
|
||||||
OSMO_VALUE_STRING(ISO_S_IN_TPDU),
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_RST,
|
||||||
|
.str = "WAIT_RST",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_ATR,
|
||||||
|
.str = "WAIT_ATR",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_IN_ATR,
|
||||||
|
.str = "IN_ATR",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_IN_PTS,
|
||||||
|
.str = "IN_PTS",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_WAIT_TPDU,
|
||||||
|
.str = "WAIT_TPDU",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = ISO_S_IN_TPDU,
|
||||||
|
.str = "IN_TPDU",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
.value = 0,
|
.value = 0,
|
||||||
.str = NULL,
|
.str = NULL,
|
||||||
@@ -86,6 +110,7 @@ enum pts_state {
|
|||||||
PTS_S_WAIT_RESP_PCK = PTS_S_WAIT_REQ_PCK | 0x10,
|
PTS_S_WAIT_RESP_PCK = PTS_S_WAIT_REQ_PCK | 0x10,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* PTS field byte index */
|
||||||
#define _PTSS 0
|
#define _PTSS 0
|
||||||
#define _PTS0 1
|
#define _PTS0 1
|
||||||
#define _PTS1 2
|
#define _PTS1 2
|
||||||
@@ -105,6 +130,46 @@ enum tpdu_state {
|
|||||||
TPDU_S_WAIT_TX, /* waiting for more data to reader */
|
TPDU_S_WAIT_TX, /* waiting for more data to reader */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const struct value_string tpdu_state_names[] = {
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_CLA,
|
||||||
|
.str = "WAIT_CLA",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_INS,
|
||||||
|
.str = "WAIT_INS",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_P1,
|
||||||
|
.str = "WAIT_P1",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_P2,
|
||||||
|
.str = "WAIT_P2",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_P3,
|
||||||
|
.str = "WAIT_P3",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_PB,
|
||||||
|
.str = "WAIT_PB",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_RX,
|
||||||
|
.str = "WAIT_RX",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = TPDU_S_WAIT_TX,
|
||||||
|
.str = "WAIT_TX",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.value = 0,
|
||||||
|
.str = NULL,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/* TPDU field byte index */
|
||||||
#define _CLA 0
|
#define _CLA 0
|
||||||
#define _INS 1
|
#define _INS 1
|
||||||
#define _P1 2
|
#define _P1 2
|
||||||
@@ -117,9 +182,9 @@ struct card_handle {
|
|||||||
enum iso7816_3_card_state state;
|
enum iso7816_3_card_state state;
|
||||||
|
|
||||||
/* signal levels */
|
/* signal levels */
|
||||||
uint8_t vcc_active; /* 1 = on, 0 = off */
|
bool vcc_active; /*< if VCC is active (true = active/ON) */
|
||||||
uint8_t in_reset; /* 1 = RST low, 0 = RST high */
|
bool in_reset; /*< if card is in reset (true = RST low/asserted, false = RST high/ released) */
|
||||||
uint8_t clocked; /* 1 = active, 0 = inactive */
|
bool clocked; /*< if clock is active ( true = active, false = inactive) */
|
||||||
|
|
||||||
/* timing parameters, from PTS */
|
/* timing parameters, from PTS */
|
||||||
uint8_t fi;
|
uint8_t fi;
|
||||||
@@ -189,12 +254,35 @@ void usb_buf_upd_len_and_submit(struct msgb *msg)
|
|||||||
/* Allocate USB buffer and push + initialize simtrace_msg_hdr */
|
/* Allocate USB buffer and push + initialize simtrace_msg_hdr */
|
||||||
struct msgb *usb_buf_alloc_st(uint8_t ep, uint8_t msg_class, uint8_t msg_type)
|
struct msgb *usb_buf_alloc_st(uint8_t ep, uint8_t msg_class, uint8_t msg_type)
|
||||||
{
|
{
|
||||||
struct msgb *msg;
|
struct msgb *msg = NULL;
|
||||||
struct simtrace_msg_hdr *sh;
|
struct simtrace_msg_hdr *sh;
|
||||||
|
|
||||||
msg = usb_buf_alloc(ep);
|
while (!msg) {
|
||||||
if (!msg)
|
msg = usb_buf_alloc(ep); // try to allocate some memory
|
||||||
return NULL;
|
if (!msg) { // allocation failed, we might be out of memory
|
||||||
|
struct llist_head *queue = usb_get_queue(ep);
|
||||||
|
if (!queue) {
|
||||||
|
TRACE_ERROR("ep %u: %s queue does not exist\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
if (llist_empty(queue)) {
|
||||||
|
TRACE_ERROR("ep %u: %s EOMEM (queue already empty)\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
msg = msgb_dequeue(queue);
|
||||||
|
if (!msg) {
|
||||||
|
TRACE_ERROR("ep %u: %s no msg in non-empty queue\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
usb_buf_free(msg);
|
||||||
|
msg = NULL;
|
||||||
|
TRACE_DEBUG("ep %u: %s queue msg dropped\n\r",
|
||||||
|
ep, __func__);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
msg->l1h = msgb_put(msg, sizeof(*sh));
|
msg->l1h = msgb_put(msg, sizeof(*sh));
|
||||||
sh = (struct simtrace_msg_hdr *) msg->l1h;
|
sh = (struct simtrace_msg_hdr *) msg->l1h;
|
||||||
@@ -304,9 +392,9 @@ static void card_set_state(struct card_handle *ch,
|
|||||||
if (ch->state == new_state)
|
if (ch->state == new_state)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
TRACE_DEBUG("%u: 7816 card state %u (%s) -> %u (%s)\r\n", ch->num,
|
TRACE_DEBUG("%u: 7816 card state %s -> %s\r\n", ch->num,
|
||||||
ch->state, get_value_string(iso7816_3_card_state_names, ch->state),
|
get_value_string(iso7816_3_card_state_names, ch->state),
|
||||||
new_state, get_value_string(iso7816_3_card_state_names, new_state));
|
get_value_string(iso7816_3_card_state_names, new_state));
|
||||||
ch->state = new_state;
|
ch->state = new_state;
|
||||||
|
|
||||||
switch (new_state) {
|
switch (new_state) {
|
||||||
@@ -315,6 +403,13 @@ static void card_set_state(struct card_handle *ch,
|
|||||||
case ISO_S_WAIT_RST:
|
case ISO_S_WAIT_RST:
|
||||||
/* disable Rx and Tx of UART */
|
/* disable Rx and Tx of UART */
|
||||||
card_emu_uart_enable(ch->uart_chan, 0);
|
card_emu_uart_enable(ch->uart_chan, 0);
|
||||||
|
/* check end activation state (only necessary if the reader to not respect the activation sequence) */
|
||||||
|
if (ch->vcc_active && ch->clocked && !ch->in_reset) {
|
||||||
|
/* enable the TC/ETU counter once reset has been released */
|
||||||
|
tc_etu_enable(ch->tc_chan);
|
||||||
|
/* prepare to send the ATR */
|
||||||
|
card_set_state(ch, ISO_S_WAIT_ATR);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case ISO_S_WAIT_ATR:
|
case ISO_S_WAIT_ATR:
|
||||||
/* Reset to initial Fi / Di ratio */
|
/* Reset to initial Fi / Di ratio */
|
||||||
@@ -651,9 +746,9 @@ static void set_tpdu_state(struct card_handle *ch, enum tpdu_state new_ts)
|
|||||||
if (ch->tpdu.state == new_ts)
|
if (ch->tpdu.state == new_ts)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
TRACE_DEBUG("%u: 7816 TPDU state %u -> %u\r\n", ch->num,
|
TRACE_DEBUG("%u: 7816 TPDU state %s -> %s\r\n", ch->num,
|
||||||
ch->tpdu.state, new_ts);
|
get_value_string(tpdu_state_names, ch->tpdu.state),
|
||||||
|
get_value_string(tpdu_state_names, new_ts));
|
||||||
ch->tpdu.state = new_ts;
|
ch->tpdu.state = new_ts;
|
||||||
|
|
||||||
switch (new_ts) {
|
switch (new_ts) {
|
||||||
@@ -1061,8 +1156,7 @@ static const uint8_t default_atr[] = { 0x3B, 0x02, 0x14, 0x50 };
|
|||||||
|
|
||||||
static struct card_handle card_handles[NUM_SLOTS];
|
static struct card_handle card_handles[NUM_SLOTS];
|
||||||
|
|
||||||
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan,
|
struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uart_chan, uint8_t in_ep, uint8_t irq_ep, bool vcc_active, bool in_reset, bool clocked)
|
||||||
uint8_t in_ep, uint8_t irq_ep)
|
|
||||||
{
|
{
|
||||||
struct card_handle *ch;
|
struct card_handle *ch;
|
||||||
|
|
||||||
@@ -1075,14 +1169,13 @@ struct card_handle *card_emu_init(uint8_t slot_num, uint8_t tc_chan, uint8_t uar
|
|||||||
|
|
||||||
INIT_LLIST_HEAD(&ch->uart_tx_queue);
|
INIT_LLIST_HEAD(&ch->uart_tx_queue);
|
||||||
|
|
||||||
/* initialize the card_handle with reasonable defaults */
|
|
||||||
ch->num = slot_num;
|
ch->num = slot_num;
|
||||||
ch->irq_ep = irq_ep;
|
ch->irq_ep = irq_ep;
|
||||||
ch->in_ep = in_ep;
|
ch->in_ep = in_ep;
|
||||||
ch->state = ISO_S_WAIT_POWER;
|
ch->state = ISO_S_WAIT_POWER;
|
||||||
ch->vcc_active = 0;
|
ch->vcc_active = vcc_active;
|
||||||
ch->in_reset = 1;
|
ch->in_reset = in_reset;
|
||||||
ch->clocked = 0;
|
ch->clocked = clocked;
|
||||||
|
|
||||||
ch->fi = 0;
|
ch->fi = 0;
|
||||||
ch->di = 1;
|
ch->di = 1;
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ static const Pin pin_usim1_rst = PIN_USIM1_nRST;
|
|||||||
static const Pin pin_usim1_vcc = PIN_USIM1_VCC;
|
static const Pin pin_usim1_vcc = PIN_USIM1_VCC;
|
||||||
|
|
||||||
#ifdef CARDEMU_SECOND_UART
|
#ifdef CARDEMU_SECOND_UART
|
||||||
static const Pin pins_usim2[] = {PINS_USIM2};
|
static const Pin pins_usim2[] = {PINS_USIM2};
|
||||||
static const Pin pin_usim2_rst = PIN_USIM2_nRST;
|
static const Pin pin_usim2_rst = PIN_USIM2_nRST;
|
||||||
static const Pin pin_usim2_vcc = PIN_USIM2_VCC;
|
static const Pin pin_usim2_vcc = PIN_USIM2_VCC;
|
||||||
#endif
|
#endif
|
||||||
@@ -309,7 +309,7 @@ static int card_vcc_adc_init(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#define VCC_UV_THRESH_1V8 1500000
|
#define VCC_UV_THRESH_1V8 1500000
|
||||||
#define VCC_UV_THRESH_3V 2800000
|
#define VCC_UV_THRESH_3V 2500000
|
||||||
|
|
||||||
static void process_vcc_adc(struct cardem_inst *ci)
|
static void process_vcc_adc(struct cardem_inst *ci)
|
||||||
{
|
{
|
||||||
@@ -357,14 +357,14 @@ void ADC_IrqHandler(void)
|
|||||||
|
|
||||||
static void usim1_rst_irqhandler(const Pin *pPin)
|
static void usim1_rst_irqhandler(const Pin *pPin)
|
||||||
{
|
{
|
||||||
int active = PIO_Get(&pin_usim1_rst) ? 0 : 1;
|
bool active = PIO_Get(&pin_usim1_rst) ? false : true;
|
||||||
card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_RST, active);
|
card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_RST, active);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef DETECT_VCC_BY_ADC
|
#ifndef DETECT_VCC_BY_ADC
|
||||||
static void usim1_vcc_irqhandler(const Pin *pPin)
|
static void usim1_vcc_irqhandler(const Pin *pPin)
|
||||||
{
|
{
|
||||||
int active = PIO_Get(&pin_usim1_vcc) ? 1 : 0;
|
bool active = PIO_Get(&pin_usim1_vcc) ? true : false;
|
||||||
card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_VCC, active);
|
card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_VCC, active);
|
||||||
/* FIXME do this for real */
|
/* FIXME do this for real */
|
||||||
card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_CLK, active);
|
card_emu_io_statechg(cardem_inst[0].ch, CARD_IO_CLK, active);
|
||||||
@@ -374,14 +374,14 @@ static void usim1_vcc_irqhandler(const Pin *pPin)
|
|||||||
#ifdef CARDEMU_SECOND_UART
|
#ifdef CARDEMU_SECOND_UART
|
||||||
static void usim2_rst_irqhandler(const Pin *pPin)
|
static void usim2_rst_irqhandler(const Pin *pPin)
|
||||||
{
|
{
|
||||||
int active = PIO_Get(&pin_usim2_rst) ? 0 : 1;
|
bool active = PIO_Get(&pin_usim2_rst) ? false : true;
|
||||||
card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_RST, active);
|
card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_RST, active);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef DETECT_VCC_BY_ADC
|
#ifndef DETECT_VCC_BY_ADC
|
||||||
static void usim2_vcc_irqhandler(const Pin *pPin)
|
static void usim2_vcc_irqhandler(const Pin *pPin)
|
||||||
{
|
{
|
||||||
int active = PIO_Get(&pin_usim2_vcc) ? 1 : 0;
|
bool active = PIO_Get(&pin_usim2_vcc) ? true : false;
|
||||||
card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_VCC, active);
|
card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_VCC, active);
|
||||||
/* FIXME do this for real */
|
/* FIXME do this for real */
|
||||||
card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_CLK, active);
|
card_emu_io_statechg(cardem_inst[1].ch, CARD_IO_CLK, active);
|
||||||
@@ -420,7 +420,7 @@ void mode_cardemu_init(void)
|
|||||||
PIO_ConfigureIt(&pin_usim1_vcc, usim1_vcc_irqhandler);
|
PIO_ConfigureIt(&pin_usim1_vcc, usim1_vcc_irqhandler);
|
||||||
PIO_EnableIt(&pin_usim1_vcc);
|
PIO_EnableIt(&pin_usim1_vcc);
|
||||||
#endif /* DETECT_VCC_BY_ADC */
|
#endif /* DETECT_VCC_BY_ADC */
|
||||||
cardem_inst[0].ch = card_emu_init(0, 2, 0, SIMTRACE_CARDEM_USB_EP_USIM1_DATAIN, SIMTRACE_CARDEM_USB_EP_USIM1_INT);
|
cardem_inst[0].ch = card_emu_init(0, 2, 0, SIMTRACE_CARDEM_USB_EP_USIM1_DATAIN, SIMTRACE_CARDEM_USB_EP_USIM1_INT, PIO_Get(&pin_usim1_vcc) ? true : false, PIO_Get(&pin_usim1_rst) ? false : true, PIO_Get(&pin_usim1_vcc) ? true : false);
|
||||||
sim_switch_use_physical(0, 1);
|
sim_switch_use_physical(0, 1);
|
||||||
|
|
||||||
#ifdef CARDEMU_SECOND_UART
|
#ifdef CARDEMU_SECOND_UART
|
||||||
@@ -435,10 +435,9 @@ void mode_cardemu_init(void)
|
|||||||
PIO_ConfigureIt(&pin_usim2_vcc, usim2_vcc_irqhandler);
|
PIO_ConfigureIt(&pin_usim2_vcc, usim2_vcc_irqhandler);
|
||||||
PIO_EnableIt(&pin_usim2_vcc);
|
PIO_EnableIt(&pin_usim2_vcc);
|
||||||
#endif /* DETECT_VCC_BY_ADC */
|
#endif /* DETECT_VCC_BY_ADC */
|
||||||
cardem_inst[1].ch = card_emu_init(1, 0, 1, SIMTRACE_CARDEM_USB_EP_USIM2_DATAIN, SIMTRACE_CARDEM_USB_EP_USIM2_INT);
|
cardem_inst[1].ch = card_emu_init(1, 0, 1, SIMTRACE_CARDEM_USB_EP_USIM2_DATAIN, SIMTRACE_CARDEM_USB_EP_USIM2_INT, PIO_Get(&pin_usim2_vcc) ? true : false, PIO_Get(&pin_usim2_rst) ? false : true, PIO_Get(&pin_usim2_vcc) ? true : false);
|
||||||
sim_switch_use_physical(1, 1);
|
sim_switch_use_physical(1, 1);
|
||||||
#endif /* CARDEMU_SECOND_UART */
|
#endif /* CARDEMU_SECOND_UART */
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* called if config is deactivated */
|
/* called if config is deactivated */
|
||||||
|
|||||||
@@ -350,19 +350,22 @@ signed int vsnprintf(char *pStr, size_t length, const char *pFormat, va_list ap)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Parse type
|
// Parse type
|
||||||
switch (*pFormat) {
|
do {
|
||||||
case 'd':
|
num = 0;
|
||||||
case 'i': num = PutSignedInt(pStr, fill, width, va_arg(ap, signed int)); break;
|
switch (*pFormat) {
|
||||||
case 'u': num = PutUnsignedInt(pStr, fill, width, va_arg(ap, unsigned int)); break;
|
case 'l': num = -1; break; // ignore long qualifier since int == long (and long long is not supported)
|
||||||
case 'x': num = PutHexa(pStr, fill, width, 0, va_arg(ap, unsigned int)); break;
|
case 'd':
|
||||||
case 'X': num = PutHexa(pStr, fill, width, 1, va_arg(ap, unsigned int)); break;
|
case 'i': num = PutSignedInt(pStr, fill, width, va_arg(ap, signed int)); break;
|
||||||
case 's': num = PutString(pStr, va_arg(ap, char *)); break;
|
case 'u': num = PutUnsignedInt(pStr, fill, width, va_arg(ap, unsigned int)); break;
|
||||||
case 'c': num = PutChar(pStr, va_arg(ap, unsigned int)); break;
|
case 'x': num = PutHexa(pStr, fill, width, 0, va_arg(ap, unsigned int)); break;
|
||||||
default:
|
case 'X': num = PutHexa(pStr, fill, width, 1, va_arg(ap, unsigned int)); break;
|
||||||
return EOF;
|
case 's': num = PutString(pStr, va_arg(ap, char *)); break;
|
||||||
}
|
case 'c': num = PutChar(pStr, va_arg(ap, unsigned int)); break;
|
||||||
|
default:
|
||||||
pFormat++;
|
return EOF;
|
||||||
|
}
|
||||||
|
pFormat++;
|
||||||
|
} while (num < 0);
|
||||||
pStr += num;
|
pStr += num;
|
||||||
size += num;
|
size += num;
|
||||||
}
|
}
|
||||||
@@ -426,12 +429,17 @@ signed int vsprintf(char *pString, const char *pFormat, va_list ap)
|
|||||||
signed int vfprintf(FILE *pStream, const char *pFormat, va_list ap)
|
signed int vfprintf(FILE *pStream, const char *pFormat, va_list ap)
|
||||||
{
|
{
|
||||||
char pStr[MAX_STRING_SIZE];
|
char pStr[MAX_STRING_SIZE];
|
||||||
char pError[] = "stdio.c: increase MAX_STRING_SIZE\n\r";
|
|
||||||
|
|
||||||
// Write formatted string in buffer
|
// Write formatted string in buffer
|
||||||
if (vsprintf(pStr, pFormat, ap) >= MAX_STRING_SIZE) {
|
int rc = vsprintf(pStr, pFormat, ap);
|
||||||
|
if (rc < 0) {
|
||||||
fputs(pError, stderr);
|
fputs("format string error in ", stderr);
|
||||||
|
fputs(pFormat, stderr);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
if (rc >= MAX_STRING_SIZE) {
|
||||||
|
fputs("stdio.c: increase MAX_STRING_SIZE\n\r", stderr);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display string
|
// Display string
|
||||||
@@ -451,12 +459,17 @@ signed int vfprintf(FILE *pStream, const char *pFormat, va_list ap)
|
|||||||
signed int vfprintf_sync(FILE *pStream, const char *pFormat, va_list ap)
|
signed int vfprintf_sync(FILE *pStream, const char *pFormat, va_list ap)
|
||||||
{
|
{
|
||||||
char pStr[MAX_STRING_SIZE];
|
char pStr[MAX_STRING_SIZE];
|
||||||
char pError[] = "stdio.c: increase MAX_STRING_SIZE\n\r";
|
|
||||||
|
|
||||||
// Write formatted string in buffer
|
// Write formatted string in buffer
|
||||||
if (vsprintf(pStr, pFormat, ap) >= MAX_STRING_SIZE) {
|
int rc = vsprintf(pStr, pFormat, ap);
|
||||||
|
if (rc < 0) {
|
||||||
fputs_sync(pError, stderr);
|
fputs_sync("format string error in ", stderr);
|
||||||
|
fputs_sync(pFormat, stderr);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
if (rc >= MAX_STRING_SIZE) {
|
||||||
|
fputs_sync("stdio.c: increase MAX_STRING_SIZE\n\r", stderr);
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display string
|
// Display string
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
* ATMEL Microcontroller Software Support
|
* ATMEL Microcontroller Software Support
|
||||||
* ----------------------------------------------------------------------------
|
* ----------------------------------------------------------------------------
|
||||||
* Copyright (c) 2009, Atmel Corporation
|
* Copyright (c) 2009, Atmel Corporation
|
||||||
* Copyright (c) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
* Copyright (c) 2018-2019, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>
|
||||||
*
|
*
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
@@ -46,7 +46,10 @@
|
|||||||
* USB String descriptors
|
* USB String descriptors
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
#include "usb_strings_generated.h"
|
#include "usb_strings_generated.h"
|
||||||
|
|
||||||
|
// the index of the strings (must match the order in usb_strings.txt)
|
||||||
enum strDescNum {
|
enum strDescNum {
|
||||||
|
// static strings from usb_strings
|
||||||
MANUF_STR = 1,
|
MANUF_STR = 1,
|
||||||
PRODUCT_STRING,
|
PRODUCT_STRING,
|
||||||
SNIFFER_CONF_STR,
|
SNIFFER_CONF_STR,
|
||||||
@@ -55,9 +58,82 @@ enum strDescNum {
|
|||||||
MITM_CONF_STR,
|
MITM_CONF_STR,
|
||||||
CARDEM_USIM1_INTF_STR,
|
CARDEM_USIM1_INTF_STR,
|
||||||
CARDEM_USIM2_INTF_STR,
|
CARDEM_USIM2_INTF_STR,
|
||||||
|
CARDEM_USIM3_INTF_STR,
|
||||||
|
CARDEM_USIM4_INTF_STR,
|
||||||
|
// runtime strings
|
||||||
|
SERIAL_STR,
|
||||||
|
VERSION_CONF_STR,
|
||||||
|
VERSION_STR,
|
||||||
|
// count
|
||||||
STRING_DESC_CNT
|
STRING_DESC_CNT
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** array of static (from usb_strings) and runtime (serial, version) USB strings
|
||||||
|
*/
|
||||||
|
static const unsigned char *usb_strings_extended[ARRAY_SIZE(usb_strings) + 3];
|
||||||
|
|
||||||
|
/* 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];
|
||||||
|
|
||||||
/*------------------------------------------------------------------------------
|
/*------------------------------------------------------------------------------
|
||||||
* USB Device descriptors
|
* USB Device descriptors
|
||||||
*------------------------------------------------------------------------------*/
|
*------------------------------------------------------------------------------*/
|
||||||
@@ -523,6 +599,40 @@ static const SIMTraceDriverConfigurationDescriptorMITM
|
|||||||
};
|
};
|
||||||
#endif /* HAVE_CARDEM */
|
#endif /* HAVE_CARDEM */
|
||||||
|
|
||||||
|
/* 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 = CFG_NUM_VERSION,
|
||||||
|
.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,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
const USBConfigurationDescriptor *configurationDescriptorsArr[] = {
|
const USBConfigurationDescriptor *configurationDescriptorsArr[] = {
|
||||||
#ifdef HAVE_SNIFFER
|
#ifdef HAVE_SNIFFER
|
||||||
&configurationDescriptorSniffer.configuration,
|
&configurationDescriptorSniffer.configuration,
|
||||||
@@ -536,6 +646,7 @@ const USBConfigurationDescriptor *configurationDescriptorsArr[] = {
|
|||||||
#ifdef HAVE_MITM
|
#ifdef HAVE_MITM
|
||||||
&configurationDescriptorMITM.configuration,
|
&configurationDescriptorMITM.configuration,
|
||||||
#endif
|
#endif
|
||||||
|
&configurationDescriptorVersion.configuration,
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Standard USB device descriptor for the CDC serial driver */
|
/** Standard USB device descriptor for the CDC serial driver */
|
||||||
@@ -552,7 +663,7 @@ const USBDeviceDescriptor deviceDescriptor = {
|
|||||||
.bcdDevice = 2, /* Release number */
|
.bcdDevice = 2, /* Release number */
|
||||||
.iManufacturer = MANUF_STR,
|
.iManufacturer = MANUF_STR,
|
||||||
.iProduct = PRODUCT_STRING,
|
.iProduct = PRODUCT_STRING,
|
||||||
.iSerialNumber = 0,
|
.iSerialNumber = SERIAL_STR,
|
||||||
.bNumConfigurations = ARRAY_SIZE(configurationDescriptorsArr),
|
.bNumConfigurations = ARRAY_SIZE(configurationDescriptorsArr),
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -566,8 +677,8 @@ static const USBDDriverDescriptors driverDescriptors = {
|
|||||||
0, /* No high-speed configuration descriptor */
|
0, /* No high-speed configuration descriptor */
|
||||||
0, /* No high-speed device qualifier descriptor */
|
0, /* No high-speed device qualifier descriptor */
|
||||||
0, /* No high-speed other speed configuration descriptor */
|
0, /* No high-speed other speed configuration descriptor */
|
||||||
usb_strings,
|
usb_strings_extended,
|
||||||
ARRAY_SIZE(usb_strings),/* cnt string descriptors in list */
|
ARRAY_SIZE(usb_strings_extended),/* cnt string descriptors in list */
|
||||||
};
|
};
|
||||||
|
|
||||||
/*----------------------------------------------------------------------------
|
/*----------------------------------------------------------------------------
|
||||||
@@ -576,7 +687,7 @@ static const USBDDriverDescriptors driverDescriptors = {
|
|||||||
|
|
||||||
void SIMtrace_USB_Initialize(void)
|
void SIMtrace_USB_Initialize(void)
|
||||||
{
|
{
|
||||||
|
unsigned int i;
|
||||||
/* 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 */
|
||||||
#ifdef PIN_USB_PULLUP
|
#ifdef PIN_USB_PULLUP
|
||||||
const Pin usb_dp_pullup = PIN_USB_PULLUP;
|
const Pin usb_dp_pullup = PIN_USB_PULLUP;
|
||||||
@@ -593,6 +704,32 @@ void SIMtrace_USB_Initialize(void)
|
|||||||
// Get std USB driver
|
// Get std USB driver
|
||||||
USBDDriver *pUsbd = USBD_GetDriver();
|
USBDDriver *pUsbd = USBD_GetDriver();
|
||||||
|
|
||||||
|
// 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[SERIAL_STR] = usb_string_serial;
|
||||||
|
usb_strings_extended[VERSION_CONF_STR] = usb_string_version_conf;
|
||||||
|
usb_strings_extended[VERSION_STR] = usb_string_version;
|
||||||
|
|
||||||
// Initialize standard USB driver
|
// Initialize standard USB driver
|
||||||
USBDDriver_Initialize(pUsbd, &driverDescriptors, 0); // Multiple interface settings not supported
|
USBDDriver_Initialize(pUsbd, &driverDescriptors, 0); // Multiple interface settings not supported
|
||||||
USBD_Init();
|
USBD_Init();
|
||||||
|
|||||||
@@ -11,12 +11,12 @@ CFLAGS=-g -Wall $(LIBOSMOCORE_CFLAGS) \
|
|||||||
-I../libboard/common/include \
|
-I../libboard/common/include \
|
||||||
-I../libboard/simtrace/include \
|
-I../libboard/simtrace/include \
|
||||||
-I.
|
-I.
|
||||||
LDFLAGS=$(LIBOSMOCORE_LIBS)
|
LIBS=$(LIBOSMOCORE_LIBS)
|
||||||
|
|
||||||
VPATH=../src_simtrace ../libcommon/source
|
VPATH=../src_simtrace ../libcommon/source
|
||||||
|
|
||||||
card_emu_test: card_emu_tests.hobj card_emu.hobj usb_buf.hobj iso7816_fidi.hobj
|
card_emu_test: card_emu_tests.hobj card_emu.hobj usb_buf.hobj iso7816_fidi.hobj
|
||||||
$(CC) $(LDFLAGS) -o $@ $^
|
$(CC) $(LDFLAGS) -o $@ $^ $(LIBS)
|
||||||
|
|
||||||
%.hobj: %.c
|
%.hobj: %.c
|
||||||
$(CC) $(CFLAGS) -o $@ -c $^
|
$(CC) $(CFLAGS) -o $@ -c $^
|
||||||
|
|||||||
@@ -13,7 +13,9 @@
|
|||||||
#define PHONE_INT 2
|
#define PHONE_INT 2
|
||||||
#define PHONE_DATAOUT 3
|
#define PHONE_DATAOUT 3
|
||||||
|
|
||||||
/* stub functions required by card_emu.c */
|
/***********************************************************************
|
||||||
|
* stub functions required by card_emu.c
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
void card_emu_uart_wait_tx_idle(uint8_t uart_chan)
|
void card_emu_uart_wait_tx_idle(uint8_t uart_chan)
|
||||||
{
|
{
|
||||||
@@ -30,6 +32,7 @@ int card_emu_uart_update_fidi(uint8_t uart_chan, unsigned int fidi)
|
|||||||
static uint8_t tx_debug_buf[1024];
|
static uint8_t tx_debug_buf[1024];
|
||||||
static unsigned int tx_debug_buf_idx;
|
static unsigned int tx_debug_buf_idx;
|
||||||
|
|
||||||
|
/* the card emulator wants to send some data to the host [reader] */
|
||||||
int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte)
|
int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte)
|
||||||
{
|
{
|
||||||
printf("UART_TX(%02x)\n", byte);
|
printf("UART_TX(%02x)\n", byte);
|
||||||
@@ -37,13 +40,6 @@ int card_emu_uart_tx(uint8_t uart_chan, uint8_t byte)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void reader_check_and_clear(const uint8_t *data, unsigned int len)
|
|
||||||
{
|
|
||||||
assert(len == tx_debug_buf_idx);
|
|
||||||
assert(!memcmp(tx_debug_buf, data, len));
|
|
||||||
tx_debug_buf_idx = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx)
|
void card_emu_uart_enable(uint8_t uart_chan, uint8_t rxtx)
|
||||||
{
|
{
|
||||||
char *rts;
|
char *rts;
|
||||||
@@ -95,7 +91,21 @@ void tc_etu_disable(uint8_t chan_nr)
|
|||||||
printf("tc_etu_disable(tc_chan=%u)\n", chan_nr);
|
printf("tc_etu_disable(tc_chan=%u)\n", chan_nr);
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint8_t atr[] = { 0x3b, 0x02, 0x14, 0x50 };
|
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
* test helper functions
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
|
|
||||||
|
static void reader_check_and_clear(const uint8_t *data, unsigned int len)
|
||||||
|
{
|
||||||
|
assert(len == tx_debug_buf_idx);
|
||||||
|
assert(!memcmp(tx_debug_buf, data, len));
|
||||||
|
tx_debug_buf_idx = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const uint8_t atr[] = { 0x3b, 0x02, 0x14, 0x50 };
|
||||||
|
|
||||||
static int verify_atr(struct card_handle *ch)
|
static int verify_atr(struct card_handle *ch)
|
||||||
{
|
{
|
||||||
@@ -130,6 +140,7 @@ static void io_start_card(struct card_handle *ch)
|
|||||||
verify_atr(ch);
|
verify_atr(ch);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* emulate the host/reader sending some bytes to the [emulated] card */
|
||||||
static void reader_send_bytes(struct card_handle *ch, const uint8_t *bytes, unsigned int len)
|
static void reader_send_bytes(struct card_handle *ch, const uint8_t *bytes, unsigned int len)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
@@ -397,7 +408,7 @@ int main(int argc, char **argv)
|
|||||||
struct card_handle *ch;
|
struct card_handle *ch;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
ch = card_emu_init(0, 23, 42, PHONE_DATAIN, PHONE_INT);
|
ch = card_emu_init(0, 23, 42, PHONE_DATAIN, PHONE_INT, false, true, false);
|
||||||
assert(ch);
|
assert(ch);
|
||||||
|
|
||||||
usb_buf_init();
|
usb_buf_init();
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<gnm:Workbook xmlns:gnm="http://www.gnumeric.org/v10.dtd" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.gnumeric.org/v9.xsd">
|
<gnm:Workbook xmlns:gnm="http://www.gnumeric.org/v10.dtd" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.gnumeric.org/v9.xsd">
|
||||||
<gnm:Version Epoch="1" Major="12" Minor="32" Full="1.12.32"/>
|
<gnm:Version Epoch="1" Major="12" Minor="44" Full="1.12.44"/>
|
||||||
<gnm:Attributes>
|
<gnm:Attributes>
|
||||||
<gnm:Attribute>
|
<gnm:Attribute>
|
||||||
<gnm:name>WorkbookView::show_horizontal_scrollbar</gnm:name>
|
<gnm:name>WorkbookView::show_horizontal_scrollbar</gnm:name>
|
||||||
@@ -25,7 +25,7 @@
|
|||||||
</gnm:Attributes>
|
</gnm:Attributes>
|
||||||
<office:document-meta xmlns:office="urn:oasis:names:tc:opendocument:xmlns:office:1.0" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:meta="urn:oasis:names:tc:opendocument:xmlns:meta:1.0" xmlns:ooo="http://openoffice.org/2004/office" office:version="1.2">
|
<office:document-meta xmlns:office="urn:oasis:names:tc:opendocument:xmlns:office:1.0" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:meta="urn:oasis:names:tc:opendocument:xmlns:meta:1.0" xmlns:ooo="http://openoffice.org/2004/office" office:version="1.2">
|
||||||
<office:meta>
|
<office:meta>
|
||||||
<dc:date>2017-05-05T14:31:27Z</dc:date>
|
<dc:date>2019-02-28T17:12:35Z</dc:date>
|
||||||
<meta:creation-date>2017-03-05T18:42:38Z</meta:creation-date>
|
<meta:creation-date>2017-03-05T18:42:38Z</meta:creation-date>
|
||||||
</office:meta>
|
</office:meta>
|
||||||
</office:document-meta>
|
</office:document-meta>
|
||||||
@@ -35,11 +35,11 @@
|
|||||||
<gnm:SheetName gnm:Cols="256" gnm:Rows="65536">Sheet2</gnm:SheetName>
|
<gnm:SheetName gnm:Cols="256" gnm:Rows="65536">Sheet2</gnm:SheetName>
|
||||||
<gnm:SheetName gnm:Cols="256" gnm:Rows="65536">Sheet3</gnm:SheetName>
|
<gnm:SheetName gnm:Cols="256" gnm:Rows="65536">Sheet3</gnm:SheetName>
|
||||||
</gnm:SheetNameIndex>
|
</gnm:SheetNameIndex>
|
||||||
<gnm:Geometry Width="1304" Height="896"/>
|
<gnm:Geometry Width="1918" Height="2001"/>
|
||||||
<gnm:Sheets>
|
<gnm:Sheets>
|
||||||
<gnm:Sheet DisplayFormulas="0" HideZero="0" HideGrid="0" HideColHeader="0" HideRowHeader="0" DisplayOutlines="1" OutlineSymbolsBelow="1" OutlineSymbolsRight="1" Visibility="GNM_SHEET_VISIBILITY_VISIBLE" GridColor="0:0:0">
|
<gnm:Sheet DisplayFormulas="0" HideZero="0" HideGrid="0" HideColHeader="0" HideRowHeader="0" DisplayOutlines="1" OutlineSymbolsBelow="1" OutlineSymbolsRight="1" Visibility="GNM_SHEET_VISIBILITY_VISIBLE" GridColor="0:0:0">
|
||||||
<gnm:Name>Sheet1</gnm:Name>
|
<gnm:Name>Sheet1</gnm:Name>
|
||||||
<gnm:MaxCol>7</gnm:MaxCol>
|
<gnm:MaxCol>9</gnm:MaxCol>
|
||||||
<gnm:MaxRow>67</gnm:MaxRow>
|
<gnm:MaxRow>67</gnm:MaxRow>
|
||||||
<gnm:Zoom>1</gnm:Zoom>
|
<gnm:Zoom>1</gnm:Zoom>
|
||||||
<gnm:Names>
|
<gnm:Names>
|
||||||
@@ -1057,127 +1057,452 @@
|
|||||||
</gnm:StyleBorder>
|
</gnm:StyleBorder>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="0" endCol="255" endRow="5">
|
<gnm:StyleRegion startCol="7" startRow="0" endCol="7" endRow="5">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="6" endCol="255" endRow="9">
|
<gnm:StyleRegion startCol="7" startRow="6" endCol="7" endRow="9">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="10" endCol="255" endRow="11">
|
<gnm:StyleRegion startCol="7" startRow="10" endCol="7" endRow="11">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="12" endCol="255" endRow="14">
|
<gnm:StyleRegion startCol="7" startRow="12" endCol="7" endRow="14">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="15" endCol="255" endRow="15">
|
<gnm:StyleRegion startCol="7" startRow="15" endCol="7" endRow="15">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="16" endCol="255" endRow="19">
|
<gnm:StyleRegion startCol="7" startRow="16" endCol="7" endRow="19">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="20" endCol="255" endRow="21">
|
<gnm:StyleRegion startCol="7" startRow="20" endCol="7" endRow="21">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="22" endCol="255" endRow="26">
|
<gnm:StyleRegion startCol="7" startRow="22" endCol="7" endRow="26">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="27" endCol="255" endRow="27">
|
<gnm:StyleRegion startCol="7" startRow="27" endCol="7" endRow="27">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="28" endCol="255" endRow="35">
|
<gnm:StyleRegion startCol="7" startRow="28" endCol="7" endRow="35">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="36" endCol="255" endRow="36">
|
<gnm:StyleRegion startCol="7" startRow="36" endCol="7" endRow="36">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="37" endCol="255" endRow="41">
|
<gnm:StyleRegion startCol="7" startRow="37" endCol="7" endRow="41">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="42" endCol="255" endRow="43">
|
<gnm:StyleRegion startCol="7" startRow="42" endCol="7" endRow="43">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="44" endCol="255" endRow="47">
|
<gnm:StyleRegion startCol="7" startRow="44" endCol="7" endRow="47">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="48" endCol="255" endRow="49">
|
<gnm:StyleRegion startCol="7" startRow="48" endCol="7" endRow="49">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="50" endCol="255" endRow="51">
|
<gnm:StyleRegion startCol="7" startRow="50" endCol="7" endRow="51">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="52" endCol="255" endRow="54">
|
<gnm:StyleRegion startCol="7" startRow="52" endCol="7" endRow="54">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="55" endCol="255" endRow="55">
|
<gnm:StyleRegion startCol="7" startRow="55" endCol="7" endRow="55">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="56" endCol="255" endRow="57">
|
<gnm:StyleRegion startCol="7" startRow="56" endCol="7" endRow="57">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="58" endCol="255" endRow="58">
|
<gnm:StyleRegion startCol="7" startRow="58" endCol="7" endRow="58">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="59" endCol="255" endRow="61">
|
<gnm:StyleRegion startCol="7" startRow="59" endCol="7" endRow="61">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="62" endCol="255" endRow="62">
|
<gnm:StyleRegion startCol="7" startRow="62" endCol="7" endRow="62">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="63" endCol="255" endRow="63">
|
<gnm:StyleRegion startCol="7" startRow="63" endCol="7" endRow="63">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="64" endCol="255" endRow="66">
|
<gnm:StyleRegion startCol="7" startRow="64" endCol="7" endRow="66">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
</gnm:StyleRegion>
|
</gnm:StyleRegion>
|
||||||
<gnm:StyleRegion startCol="7" startRow="67" endCol="255" endRow="65535">
|
<gnm:StyleRegion startCol="7" startRow="67" endCol="7" endRow="65535">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="0" endCol="8" endRow="5">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="6" endCol="8" endRow="9">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="10" endCol="8" endRow="11">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="12" endCol="8" endRow="14">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="15" endCol="8" endRow="15">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="16" endCol="8" endRow="19">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="20" endCol="8" endRow="21">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="22" endCol="8" endRow="26">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="27" endCol="8" endRow="27">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="28" endCol="8" endRow="35">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="36" endCol="8" endRow="36">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="37" endCol="8" endRow="41">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="42" endCol="8" endRow="43">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="44" endCol="8" endRow="47">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="48" endCol="8" endRow="49">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="50" endCol="8" endRow="51">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="52" endCol="8" endRow="54">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="55" endCol="8" endRow="55">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="56" endCol="8" endRow="57">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="58" endCol="8" endRow="58">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="59" endCol="8" endRow="61">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="62" endCol="8" endRow="62">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="63" endCol="8" endRow="63">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="64" endCol="8" endRow="66">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="8" startRow="67" endCol="8" endRow="65535">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
<gnm:StyleBorder>
|
||||||
|
<gnm:Left Style="1" Color="0:0:0"/>
|
||||||
|
</gnm:StyleBorder>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="0" endCol="255" endRow="5">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="6" endCol="255" endRow="9">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="10" endCol="255" endRow="11">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="12" endCol="255" endRow="14">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="15" endCol="255" endRow="15">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="16" endCol="255" endRow="19">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="20" endCol="255" endRow="21">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="22" endCol="255" endRow="26">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="27" endCol="255" endRow="27">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="28" endCol="255" endRow="35">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="36" endCol="255" endRow="36">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="37" endCol="255" endRow="41">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="42" endCol="255" endRow="43">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="44" endCol="255" endRow="47">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="48" endCol="255" endRow="49">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="50" endCol="255" endRow="51">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="52" endCol="255" endRow="54">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="55" endCol="255" endRow="55">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="56" endCol="255" endRow="57">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="58" endCol="255" endRow="58">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="59" endCol="255" endRow="61">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="62" endCol="255" endRow="62">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="63" endCol="255" endRow="63">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="64" endCol="255" endRow="66">
|
||||||
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="1" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:BABA" PatternColor="0:0:0" Format="General">
|
||||||
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
|
</gnm:Style>
|
||||||
|
</gnm:StyleRegion>
|
||||||
|
<gnm:StyleRegion startCol="9" startRow="67" endCol="255" endRow="65535">
|
||||||
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
<gnm:Style HAlign="GNM_HALIGN_GENERAL" VAlign="GNM_VALIGN_BOTTOM" WrapText="0" ShrinkToFit="0" Rotation="0" Shade="0" Indent="0" Locked="1" Hidden="0" Fore="0:0:0" Back="FFFF:FFFF:FFFF" PatternColor="0:0:0" Format="General">
|
||||||
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
<gnm:Font Unit="10" Bold="0" Italic="0" Underline="0" StrikeThrough="0" Script="0">Sans</gnm:Font>
|
||||||
</gnm:Style>
|
</gnm:Style>
|
||||||
@@ -1190,15 +1515,27 @@
|
|||||||
<gnm:ColInfo No="5" Unit="120" HardSize="1"/>
|
<gnm:ColInfo No="5" Unit="120" HardSize="1"/>
|
||||||
<gnm:ColInfo No="7" Unit="161.2" HardSize="1"/>
|
<gnm:ColInfo No="7" Unit="161.2" HardSize="1"/>
|
||||||
</gnm:Cols>
|
</gnm:Cols>
|
||||||
<gnm:Rows DefaultSizePts="12.75"/>
|
<gnm:Rows DefaultSizePts="12.75">
|
||||||
<gnm:Selections CursorCol="5" CursorRow="34">
|
<gnm:RowInfo No="2" Unit="13.5" Count="2"/>
|
||||||
<gnm:Selection startCol="5" startRow="34" endCol="5" endRow="34"/>
|
<gnm:RowInfo No="12" Unit="13.5" Count="3"/>
|
||||||
|
<gnm:RowInfo No="16" Unit="13.5" Count="2"/>
|
||||||
|
<gnm:RowInfo No="19" Unit="13.5"/>
|
||||||
|
<gnm:RowInfo No="22" Unit="13.5"/>
|
||||||
|
<gnm:RowInfo No="24" Unit="13.5" Count="2"/>
|
||||||
|
<gnm:RowInfo No="30" Unit="13.5" Count="5"/>
|
||||||
|
<gnm:RowInfo No="39" Unit="13.5"/>
|
||||||
|
<gnm:RowInfo No="46" Unit="13.5"/>
|
||||||
|
<gnm:RowInfo No="55" Unit="13.5"/>
|
||||||
|
</gnm:Rows>
|
||||||
|
<gnm:Selections CursorCol="3" CursorRow="0">
|
||||||
|
<gnm:Selection startCol="3" startRow="0" endCol="3" endRow="0"/>
|
||||||
</gnm:Selections>
|
</gnm:Selections>
|
||||||
<gnm:Cells>
|
<gnm:Cells>
|
||||||
<gnm:Cell Row="0" Col="0" ValueType="60">Pins / Ios of SAM3 based devices</gnm:Cell>
|
<gnm:Cell Row="0" Col="0" ValueType="60">Pins / Ios of SAM3 based devices</gnm:Cell>
|
||||||
<gnm:Cell Row="2" Col="2" ValueType="60">SIMtrace v1.x</gnm:Cell>
|
<gnm:Cell Row="2" Col="2" ValueType="60">SIMtrace v1.x</gnm:Cell>
|
||||||
<gnm:Cell Row="2" Col="4" ValueType="60">Quad Modem</gnm:Cell>
|
<gnm:Cell Row="2" Col="4" ValueType="60">Quad Modem</gnm:Cell>
|
||||||
<gnm:Cell Row="2" Col="6" ValueType="60">OWHW</gnm:Cell>
|
<gnm:Cell Row="2" Col="6" ValueType="60">OWHW</gnm:Cell>
|
||||||
|
<gnm:Cell Row="2" Col="8" ValueType="60">SAM3-P256</gnm:Cell>
|
||||||
<gnm:Cell Row="3" Col="0" ValueType="60">Pin Number</gnm:Cell>
|
<gnm:Cell Row="3" Col="0" ValueType="60">Pin Number</gnm:Cell>
|
||||||
<gnm:Cell Row="3" Col="1" ValueType="60">Pin Name</gnm:Cell>
|
<gnm:Cell Row="3" Col="1" ValueType="60">Pin Name</gnm:Cell>
|
||||||
<gnm:Cell Row="3" Col="2" ValueType="60">Used Function</gnm:Cell>
|
<gnm:Cell Row="3" Col="2" ValueType="60">Used Function</gnm:Cell>
|
||||||
@@ -1207,6 +1544,8 @@
|
|||||||
<gnm:Cell Row="3" Col="5" ValueType="60">Signal Name</gnm:Cell>
|
<gnm:Cell Row="3" Col="5" ValueType="60">Signal Name</gnm:Cell>
|
||||||
<gnm:Cell Row="3" Col="6" ValueType="60">Used Function</gnm:Cell>
|
<gnm:Cell Row="3" Col="6" ValueType="60">Used Function</gnm:Cell>
|
||||||
<gnm:Cell Row="3" Col="7" ValueType="60">Signal Name</gnm:Cell>
|
<gnm:Cell Row="3" Col="7" ValueType="60">Signal Name</gnm:Cell>
|
||||||
|
<gnm:Cell Row="3" Col="8" ValueType="60">Used Function</gnm:Cell>
|
||||||
|
<gnm:Cell Row="3" Col="9" ValueType="60">Signal Name</gnm:Cell>
|
||||||
<gnm:Cell Row="4" Col="0" ValueType="40">1</gnm:Cell>
|
<gnm:Cell Row="4" Col="0" ValueType="40">1</gnm:Cell>
|
||||||
<gnm:Cell Row="4" Col="1" ValueType="60">ADVREF</gnm:Cell>
|
<gnm:Cell Row="4" Col="1" ValueType="60">ADVREF</gnm:Cell>
|
||||||
<gnm:Cell Row="4" Col="4" ValueType="60">ADVREF</gnm:Cell>
|
<gnm:Cell Row="4" Col="4" ValueType="60">ADVREF</gnm:Cell>
|
||||||
@@ -1245,6 +1584,7 @@
|
|||||||
<gnm:Cell Row="12" Col="5" ValueType="60">LED_RED</gnm:Cell>
|
<gnm:Cell Row="12" Col="5" ValueType="60">LED_RED</gnm:Cell>
|
||||||
<gnm:Cell Row="12" Col="6" ValueType="60">PA17</gnm:Cell>
|
<gnm:Cell Row="12" Col="6" ValueType="60">PA17</gnm:Cell>
|
||||||
<gnm:Cell Row="12" Col="7" ValueType="60">LED_RED</gnm:Cell>
|
<gnm:Cell Row="12" Col="7" ValueType="60">LED_RED</gnm:Cell>
|
||||||
|
<gnm:Cell Row="12" Col="8" ValueType="60">LED2</gnm:Cell>
|
||||||
<gnm:Cell Row="13" Col="0" ValueType="40">10</gnm:Cell>
|
<gnm:Cell Row="13" Col="0" ValueType="40">10</gnm:Cell>
|
||||||
<gnm:Cell Row="13" Col="1" ValueType="60">PA18</gnm:Cell>
|
<gnm:Cell Row="13" Col="1" ValueType="60">PA18</gnm:Cell>
|
||||||
<gnm:Cell Row="13" Col="2" ValueType="60">PA18</gnm:Cell>
|
<gnm:Cell Row="13" Col="2" ValueType="60">PA18</gnm:Cell>
|
||||||
@@ -1253,6 +1593,7 @@
|
|||||||
<gnm:Cell Row="13" Col="5" ValueType="60">LED_GREEN</gnm:Cell>
|
<gnm:Cell Row="13" Col="5" ValueType="60">LED_GREEN</gnm:Cell>
|
||||||
<gnm:Cell Row="13" Col="6" ValueType="60">PA18</gnm:Cell>
|
<gnm:Cell Row="13" Col="6" ValueType="60">PA18</gnm:Cell>
|
||||||
<gnm:Cell Row="13" Col="7" ValueType="60">LED_GREEN</gnm:Cell>
|
<gnm:Cell Row="13" Col="7" ValueType="60">LED_GREEN</gnm:Cell>
|
||||||
|
<gnm:Cell Row="13" Col="8" ValueType="60">LED1</gnm:Cell>
|
||||||
<gnm:Cell Row="14" Col="0" ValueType="40">11</gnm:Cell>
|
<gnm:Cell Row="14" Col="0" ValueType="40">11</gnm:Cell>
|
||||||
<gnm:Cell Row="14" Col="1" ValueType="60">PA21/RXD1/PCK1</gnm:Cell>
|
<gnm:Cell Row="14" Col="1" ValueType="60">PA21/RXD1/PCK1</gnm:Cell>
|
||||||
<gnm:Cell Row="14" Col="2" ValueType="60">RXD1</gnm:Cell>
|
<gnm:Cell Row="14" Col="2" ValueType="60">RXD1</gnm:Cell>
|
||||||
@@ -1261,6 +1602,7 @@
|
|||||||
<gnm:Cell Row="14" Col="5" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
<gnm:Cell Row="14" Col="5" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
||||||
<gnm:Cell Row="14" Col="6" ValueType="60">RXD1</gnm:Cell>
|
<gnm:Cell Row="14" Col="6" ValueType="60">RXD1</gnm:Cell>
|
||||||
<gnm:Cell Row="14" Col="7" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
<gnm:Cell Row="14" Col="7" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
||||||
|
<gnm:Cell Row="14" Col="8" ValueType="60">RXD1</gnm:Cell>
|
||||||
<gnm:Cell Row="15" Col="0" ValueType="40">12</gnm:Cell>
|
<gnm:Cell Row="15" Col="0" ValueType="40">12</gnm:Cell>
|
||||||
<gnm:Cell Row="15" Col="1" ValueType="60">VDDCORE</gnm:Cell>
|
<gnm:Cell Row="15" Col="1" ValueType="60">VDDCORE</gnm:Cell>
|
||||||
<gnm:Cell Row="16" Col="0" ValueType="40">13</gnm:Cell>
|
<gnm:Cell Row="16" Col="0" ValueType="40">13</gnm:Cell>
|
||||||
@@ -1270,6 +1612,7 @@
|
|||||||
<gnm:Cell Row="16" Col="4" ValueType="60">AD2</gnm:Cell>
|
<gnm:Cell Row="16" Col="4" ValueType="60">AD2</gnm:Cell>
|
||||||
<gnm:Cell Row="16" Col="5" ValueType="60">VERSION_DETECT12</gnm:Cell>
|
<gnm:Cell Row="16" Col="5" ValueType="60">VERSION_DETECT12</gnm:Cell>
|
||||||
<gnm:Cell Row="16" Col="7" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="16" Col="7" ValueType="60">NC</gnm:Cell>
|
||||||
|
<gnm:Cell Row="16" Col="8" ValueType="60">B1</gnm:Cell>
|
||||||
<gnm:Cell Row="17" Col="0" ValueType="40">14</gnm:Cell>
|
<gnm:Cell Row="17" Col="0" ValueType="40">14</gnm:Cell>
|
||||||
<gnm:Cell Row="17" Col="1" ValueType="60">PA22/TXD1/NPCS3</gnm:Cell>
|
<gnm:Cell Row="17" Col="1" ValueType="60">PA22/TXD1/NPCS3</gnm:Cell>
|
||||||
<gnm:Cell Row="17" Col="2" ValueType="60">TXD1</gnm:Cell>
|
<gnm:Cell Row="17" Col="2" ValueType="60">TXD1</gnm:Cell>
|
||||||
@@ -1278,6 +1621,7 @@
|
|||||||
<gnm:Cell Row="17" Col="5" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
<gnm:Cell Row="17" Col="5" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
||||||
<gnm:Cell Row="17" Col="6" ValueType="60">TXD1</gnm:Cell>
|
<gnm:Cell Row="17" Col="6" ValueType="60">TXD1</gnm:Cell>
|
||||||
<gnm:Cell Row="17" Col="7" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
<gnm:Cell Row="17" Col="7" ValueType="60">ST_USIM1_IO</gnm:Cell>
|
||||||
|
<gnm:Cell Row="17" Col="8" ValueType="60">TXD1</gnm:Cell>
|
||||||
<gnm:Cell Row="18" Col="0" ValueType="40">15</gnm:Cell>
|
<gnm:Cell Row="18" Col="0" ValueType="40">15</gnm:Cell>
|
||||||
<gnm:Cell Row="18" Col="1" ValueType="60">PA23/SCK1/PWM0</gnm:Cell>
|
<gnm:Cell Row="18" Col="1" ValueType="60">PA23/SCK1/PWM0</gnm:Cell>
|
||||||
<gnm:Cell Row="18" Col="2" ValueType="60">SCK1</gnm:Cell>
|
<gnm:Cell Row="18" Col="2" ValueType="60">SCK1</gnm:Cell>
|
||||||
@@ -1293,6 +1637,7 @@
|
|||||||
<gnm:Cell Row="19" Col="4" ValueType="60">PA20</gnm:Cell>
|
<gnm:Cell Row="19" Col="4" ValueType="60">PA20</gnm:Cell>
|
||||||
<gnm:Cell Row="19" Col="5" ValueType="60">!CONNECT_ST_USIM1</gnm:Cell>
|
<gnm:Cell Row="19" Col="5" ValueType="60">!CONNECT_ST_USIM1</gnm:Cell>
|
||||||
<gnm:Cell Row="19" Col="7" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="19" Col="7" ValueType="60">NC</gnm:Cell>
|
||||||
|
<gnm:Cell Row="19" Col="8" ValueType="60">B2</gnm:Cell>
|
||||||
<gnm:Cell Row="20" Col="0" ValueType="40">17</gnm:Cell>
|
<gnm:Cell Row="20" Col="0" ValueType="40">17</gnm:Cell>
|
||||||
<gnm:Cell Row="20" Col="1" ValueType="60">GND</gnm:Cell>
|
<gnm:Cell Row="20" Col="1" ValueType="60">GND</gnm:Cell>
|
||||||
<gnm:Cell Row="21" Col="0" ValueType="40">18</gnm:Cell>
|
<gnm:Cell Row="21" Col="0" ValueType="40">18</gnm:Cell>
|
||||||
@@ -1305,6 +1650,7 @@
|
|||||||
<gnm:Cell Row="22" Col="5" ValueType="60">WWAN2</gnm:Cell>
|
<gnm:Cell Row="22" Col="5" ValueType="60">WWAN2</gnm:Cell>
|
||||||
<gnm:Cell Row="22" Col="6" ValueType="60">PA16</gnm:Cell>
|
<gnm:Cell Row="22" Col="6" ValueType="60">PA16</gnm:Cell>
|
||||||
<gnm:Cell Row="22" Col="7" ValueType="60">UDP_PUP_CTL</gnm:Cell>
|
<gnm:Cell Row="22" Col="7" ValueType="60">UDP_PUP_CTL</gnm:Cell>
|
||||||
|
<gnm:Cell Row="22" Col="8" ValueType="60">DP_PUP</gnm:Cell>
|
||||||
<gnm:Cell Row="23" Col="0" ValueType="40">20</gnm:Cell>
|
<gnm:Cell Row="23" Col="0" ValueType="40">20</gnm:Cell>
|
||||||
<gnm:Cell Row="23" Col="1" ValueType="60">PA15/TF/TIOA1</gnm:Cell>
|
<gnm:Cell Row="23" Col="1" ValueType="60">PA15/TF/TIOA1</gnm:Cell>
|
||||||
<gnm:Cell Row="23" Col="2" ValueType="60">PA15</gnm:Cell>
|
<gnm:Cell Row="23" Col="2" ValueType="60">PA15</gnm:Cell>
|
||||||
@@ -1320,6 +1666,7 @@
|
|||||||
<gnm:Cell Row="24" Col="5" ValueType="60">ST12_ST34_SELECT</gnm:Cell>
|
<gnm:Cell Row="24" Col="5" ValueType="60">ST12_ST34_SELECT</gnm:Cell>
|
||||||
<gnm:Cell Row="24" Col="6" ValueType="60">PA14</gnm:Cell>
|
<gnm:Cell Row="24" Col="6" ValueType="60">PA14</gnm:Cell>
|
||||||
<gnm:Cell Row="24" Col="7" ValueType="60">SET_USIM2_PRES</gnm:Cell>
|
<gnm:Cell Row="24" Col="7" ValueType="60">SET_USIM2_PRES</gnm:Cell>
|
||||||
|
<gnm:Cell Row="24" Col="8" ValueType="60">SPCK</gnm:Cell>
|
||||||
<gnm:Cell Row="25" Col="0" ValueType="40">22</gnm:Cell>
|
<gnm:Cell Row="25" Col="0" ValueType="40">22</gnm:Cell>
|
||||||
<gnm:Cell Row="25" Col="1" ValueType="60">PA13/MOSI/PWM2</gnm:Cell>
|
<gnm:Cell Row="25" Col="1" ValueType="60">PA13/MOSI/PWM2</gnm:Cell>
|
||||||
<gnm:Cell Row="25" Col="2" ValueType="60">MOSI</gnm:Cell>
|
<gnm:Cell Row="25" Col="2" ValueType="60">MOSI</gnm:Cell>
|
||||||
@@ -1327,6 +1674,7 @@
|
|||||||
<gnm:Cell Row="25" Col="4" ValueType="60">PA13</gnm:Cell>
|
<gnm:Cell Row="25" Col="4" ValueType="60">PA13</gnm:Cell>
|
||||||
<gnm:Cell Row="25" Col="5" ValueType="60">HUB_RESET</gnm:Cell>
|
<gnm:Cell Row="25" Col="5" ValueType="60">HUB_RESET</gnm:Cell>
|
||||||
<gnm:Cell Row="25" Col="7" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="25" Col="7" ValueType="60">NC</gnm:Cell>
|
||||||
|
<gnm:Cell Row="25" Col="8" ValueType="60">MOSI</gnm:Cell>
|
||||||
<gnm:Cell Row="26" Col="0" ValueType="40">23</gnm:Cell>
|
<gnm:Cell Row="26" Col="0" ValueType="40">23</gnm:Cell>
|
||||||
<gnm:Cell Row="26" Col="1" ValueType="60">PA24/RTS1/PWM1</gnm:Cell>
|
<gnm:Cell Row="26" Col="1" ValueType="60">PA24/RTS1/PWM1</gnm:Cell>
|
||||||
<gnm:Cell Row="26" Col="2" ValueType="60">PA24</gnm:Cell>
|
<gnm:Cell Row="26" Col="2" ValueType="60">PA24</gnm:Cell>
|
||||||
@@ -1359,6 +1707,7 @@
|
|||||||
<gnm:Cell Row="30" Col="5" ValueType="60">SIMPRES1</gnm:Cell>
|
<gnm:Cell Row="30" Col="5" ValueType="60">SIMPRES1</gnm:Cell>
|
||||||
<gnm:Cell Row="30" Col="6" ValueType="60">PA12</gnm:Cell>
|
<gnm:Cell Row="30" Col="6" ValueType="60">PA12</gnm:Cell>
|
||||||
<gnm:Cell Row="30" Col="7" ValueType="60">SET_USIM1_PRES</gnm:Cell>
|
<gnm:Cell Row="30" Col="7" ValueType="60">SET_USIM1_PRES</gnm:Cell>
|
||||||
|
<gnm:Cell Row="30" Col="8" ValueType="60">MISO</gnm:Cell>
|
||||||
<gnm:Cell Row="31" Col="0" ValueType="40">28</gnm:Cell>
|
<gnm:Cell Row="31" Col="0" ValueType="40">28</gnm:Cell>
|
||||||
<gnm:Cell Row="31" Col="1" ValueType="60">PA11/NPCS0/PWM0</gnm:Cell>
|
<gnm:Cell Row="31" Col="1" ValueType="60">PA11/NPCS0/PWM0</gnm:Cell>
|
||||||
<gnm:Cell Row="31" Col="2" ValueType="60">NPCS0</gnm:Cell>
|
<gnm:Cell Row="31" Col="2" ValueType="60">NPCS0</gnm:Cell>
|
||||||
@@ -1366,6 +1715,7 @@
|
|||||||
<gnm:Cell Row="31" Col="4" ValueType="60">PA11</gnm:Cell>
|
<gnm:Cell Row="31" Col="4" ValueType="60">PA11</gnm:Cell>
|
||||||
<gnm:Cell Row="31" Col="5" ValueType="60">_SIMTRACE34_ERASE</gnm:Cell>
|
<gnm:Cell Row="31" Col="5" ValueType="60">_SIMTRACE34_ERASE</gnm:Cell>
|
||||||
<gnm:Cell Row="31" Col="7" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="31" Col="7" ValueType="60">NC</gnm:Cell>
|
||||||
|
<gnm:Cell Row="31" Col="8" ValueType="60">NPCS0</gnm:Cell>
|
||||||
<gnm:Cell Row="32" Col="0" ValueType="40">29</gnm:Cell>
|
<gnm:Cell Row="32" Col="0" ValueType="40">29</gnm:Cell>
|
||||||
<gnm:Cell Row="32" Col="1" ValueType="60">PA10/DTXD/NCPS2</gnm:Cell>
|
<gnm:Cell Row="32" Col="1" ValueType="60">PA10/DTXD/NCPS2</gnm:Cell>
|
||||||
<gnm:Cell Row="32" Col="2" ValueType="60">DTXD</gnm:Cell>
|
<gnm:Cell Row="32" Col="2" ValueType="60">DTXD</gnm:Cell>
|
||||||
@@ -1374,6 +1724,7 @@
|
|||||||
<gnm:Cell Row="32" Col="5" ValueType="60">SIMTRACE1_DTXD</gnm:Cell>
|
<gnm:Cell Row="32" Col="5" ValueType="60">SIMTRACE1_DTXD</gnm:Cell>
|
||||||
<gnm:Cell Row="32" Col="6" ValueType="60">DTXD</gnm:Cell>
|
<gnm:Cell Row="32" Col="6" ValueType="60">DTXD</gnm:Cell>
|
||||||
<gnm:Cell Row="32" Col="7" ValueType="60">SIMTRACE_DTXD</gnm:Cell>
|
<gnm:Cell Row="32" Col="7" ValueType="60">SIMTRACE_DTXD</gnm:Cell>
|
||||||
|
<gnm:Cell Row="32" Col="8" ValueType="60">DTXD</gnm:Cell>
|
||||||
<gnm:Cell Row="33" Col="0" ValueType="40">30</gnm:Cell>
|
<gnm:Cell Row="33" Col="0" ValueType="40">30</gnm:Cell>
|
||||||
<gnm:Cell Row="33" Col="1" ValueType="60">PA9/DRXDNPCS1</gnm:Cell>
|
<gnm:Cell Row="33" Col="1" ValueType="60">PA9/DRXDNPCS1</gnm:Cell>
|
||||||
<gnm:Cell Row="33" Col="2" ValueType="60">DRXD</gnm:Cell>
|
<gnm:Cell Row="33" Col="2" ValueType="60">DRXD</gnm:Cell>
|
||||||
@@ -1382,6 +1733,7 @@
|
|||||||
<gnm:Cell Row="33" Col="5" ValueType="60">SIMTRACE1_DRXD</gnm:Cell>
|
<gnm:Cell Row="33" Col="5" ValueType="60">SIMTRACE1_DRXD</gnm:Cell>
|
||||||
<gnm:Cell Row="33" Col="6" ValueType="60">DRXD</gnm:Cell>
|
<gnm:Cell Row="33" Col="6" ValueType="60">DRXD</gnm:Cell>
|
||||||
<gnm:Cell Row="33" Col="7" ValueType="60">SIMTRACE_DRXD</gnm:Cell>
|
<gnm:Cell Row="33" Col="7" ValueType="60">SIMTRACE_DRXD</gnm:Cell>
|
||||||
|
<gnm:Cell Row="33" Col="8" ValueType="60">DRXD</gnm:Cell>
|
||||||
<gnm:Cell Row="34" Col="0" ValueType="40">31</gnm:Cell>
|
<gnm:Cell Row="34" Col="0" ValueType="40">31</gnm:Cell>
|
||||||
<gnm:Cell Row="34" Col="1" ValueType="60">PA8/CTS0/ADTRG</gnm:Cell>
|
<gnm:Cell Row="34" Col="1" ValueType="60">PA8/CTS0/ADTRG</gnm:Cell>
|
||||||
<gnm:Cell Row="34" Col="2" ValueType="60">PA8</gnm:Cell>
|
<gnm:Cell Row="34" Col="2" ValueType="60">PA8</gnm:Cell>
|
||||||
@@ -1389,6 +1741,7 @@
|
|||||||
<gnm:Cell Row="34" Col="4" ValueType="60">PA8</gnm:Cell>
|
<gnm:Cell Row="34" Col="4" ValueType="60">PA8</gnm:Cell>
|
||||||
<gnm:Cell Row="34" Col="5" ValueType="60">SIMPRES2 / ST12_PRTPWR-OVERRIDE</gnm:Cell>
|
<gnm:Cell Row="34" Col="5" ValueType="60">SIMPRES2 / ST12_PRTPWR-OVERRIDE</gnm:Cell>
|
||||||
<gnm:Cell Row="34" Col="7" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="34" Col="7" ValueType="60">NC</gnm:Cell>
|
||||||
|
<gnm:Cell Row="34" Col="8" ValueType="60">DM_PUP</gnm:Cell>
|
||||||
<gnm:Cell Row="35" Col="0" ValueType="40">32</gnm:Cell>
|
<gnm:Cell Row="35" Col="0" ValueType="40">32</gnm:Cell>
|
||||||
<gnm:Cell Row="35" Col="1" ValueType="60">PA7/RTS0/PWM3</gnm:Cell>
|
<gnm:Cell Row="35" Col="1" ValueType="60">PA7/RTS0/PWM3</gnm:Cell>
|
||||||
<gnm:Cell Row="35" Col="2" ValueType="60">PA7</gnm:Cell>
|
<gnm:Cell Row="35" Col="2" ValueType="60">PA7</gnm:Cell>
|
||||||
@@ -1424,6 +1777,7 @@
|
|||||||
<gnm:Cell Row="39" Col="5" ValueType="60">ST_USIM2_CLK</gnm:Cell>
|
<gnm:Cell Row="39" Col="5" ValueType="60">ST_USIM2_CLK</gnm:Cell>
|
||||||
<gnm:Cell Row="39" Col="6" ValueType="60">TCLK0</gnm:Cell>
|
<gnm:Cell Row="39" Col="6" ValueType="60">TCLK0</gnm:Cell>
|
||||||
<gnm:Cell Row="39" Col="7" ValueType="60">ST_USIM2_CLK</gnm:Cell>
|
<gnm:Cell Row="39" Col="7" ValueType="60">ST_USIM2_CLK</gnm:Cell>
|
||||||
|
<gnm:Cell Row="39" Col="8" ValueType="60">TWCK</gnm:Cell>
|
||||||
<gnm:Cell Row="40" Col="0" ValueType="40">37</gnm:Cell>
|
<gnm:Cell Row="40" Col="0" ValueType="40">37</gnm:Cell>
|
||||||
<gnm:Cell Row="40" Col="1" ValueType="60">PA27/DTR1/TIOB2</gnm:Cell>
|
<gnm:Cell Row="40" Col="1" ValueType="60">PA27/DTR1/TIOB2</gnm:Cell>
|
||||||
<gnm:Cell Row="40" Col="2" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="40" Col="2" ValueType="60">NC</gnm:Cell>
|
||||||
@@ -1470,6 +1824,7 @@
|
|||||||
<gnm:Cell Row="46" Col="4" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="46" Col="4" ValueType="60">NC</gnm:Cell>
|
||||||
<gnm:Cell Row="46" Col="5" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="46" Col="5" ValueType="60">NC</gnm:Cell>
|
||||||
<gnm:Cell Row="46" Col="7" ValueType="60">NC</gnm:Cell>
|
<gnm:Cell Row="46" Col="7" ValueType="60">NC</gnm:Cell>
|
||||||
|
<gnm:Cell Row="46" Col="8" ValueType="60">TWD</gnm:Cell>
|
||||||
<gnm:Cell Row="47" Col="0" ValueType="40">44</gnm:Cell>
|
<gnm:Cell Row="47" Col="0" ValueType="40">44</gnm:Cell>
|
||||||
<gnm:Cell Row="47" Col="1" ValueType="60">PA2/PWM2/SCK0</gnm:Cell>
|
<gnm:Cell Row="47" Col="1" ValueType="60">PA2/PWM2/SCK0</gnm:Cell>
|
||||||
<gnm:Cell Row="47" Col="2" ValueType="60">SCK0</gnm:Cell>
|
<gnm:Cell Row="47" Col="2" ValueType="60">SCK0</gnm:Cell>
|
||||||
@@ -1522,6 +1877,7 @@
|
|||||||
<gnm:Cell Row="55" Col="3" ValueType="60">BOTLOADER_SW</gnm:Cell>
|
<gnm:Cell Row="55" Col="3" ValueType="60">BOTLOADER_SW</gnm:Cell>
|
||||||
<gnm:Cell Row="55" Col="4" ValueType="60">PA31</gnm:Cell>
|
<gnm:Cell Row="55" Col="4" ValueType="60">PA31</gnm:Cell>
|
||||||
<gnm:Cell Row="55" Col="5" ValueType="60">SCL</gnm:Cell>
|
<gnm:Cell Row="55" Col="5" ValueType="60">SCL</gnm:Cell>
|
||||||
|
<gnm:Cell Row="55" Col="8" ValueType="60">NPCS1</gnm:Cell>
|
||||||
<gnm:Cell Row="56" Col="0" ValueType="40">53</gnm:Cell>
|
<gnm:Cell Row="56" Col="0" ValueType="40">53</gnm:Cell>
|
||||||
<gnm:Cell Row="56" Col="1" ValueType="60">TCK</gnm:Cell>
|
<gnm:Cell Row="56" Col="1" ValueType="60">TCK</gnm:Cell>
|
||||||
<gnm:Cell Row="56" Col="2" ValueType="60">TCK</gnm:Cell>
|
<gnm:Cell Row="56" Col="2" ValueType="60">TCK</gnm:Cell>
|
||||||
@@ -1565,7 +1921,7 @@
|
|||||||
<gnm:Cell Row="67" Col="0" ValueType="40">64</gnm:Cell>
|
<gnm:Cell Row="67" Col="0" ValueType="40">64</gnm:Cell>
|
||||||
<gnm:Cell Row="67" Col="1" ValueType="60">VDDPLL</gnm:Cell>
|
<gnm:Cell Row="67" Col="1" ValueType="60">VDDPLL</gnm:Cell>
|
||||||
</gnm:Cells>
|
</gnm:Cells>
|
||||||
<gnm:SheetLayout TopLeft="A13"/>
|
<gnm:SheetLayout TopLeft="A1"/>
|
||||||
<gnm:Solver ModelType="0" ProblemType="0" MaxTime="60" MaxIter="1000" NonNeg="1" Discr="0" AutoScale="0" ProgramR="0" SensitivityR="0"/>
|
<gnm:Solver ModelType="0" ProblemType="0" MaxTime="60" MaxIter="1000" NonNeg="1" Discr="0" AutoScale="0" ProgramR="0" SensitivityR="0"/>
|
||||||
</gnm:Sheet>
|
</gnm:Sheet>
|
||||||
<gnm:Sheet DisplayFormulas="0" HideZero="0" HideGrid="0" HideColHeader="0" HideRowHeader="0" DisplayOutlines="1" OutlineSymbolsBelow="1" OutlineSymbolsRight="1" Visibility="GNM_SHEET_VISIBILITY_VISIBLE" GridColor="0:0:0">
|
<gnm:Sheet DisplayFormulas="0" HideZero="0" HideGrid="0" HideColHeader="0" HideRowHeader="0" DisplayOutlines="1" OutlineSymbolsBelow="1" OutlineSymbolsRight="1" Visibility="GNM_SHEET_VISIBILITY_VISIBLE" GridColor="0:0:0">
|
||||||
|
|||||||
32
host/.gitignore
vendored
Normal file
32
host/.gitignore
vendored
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
.o
|
||||||
|
*.a
|
||||||
|
*.lo
|
||||||
|
*.la
|
||||||
|
.deps
|
||||||
|
Makefile
|
||||||
|
Makefile.in
|
||||||
|
|
||||||
|
#configure
|
||||||
|
aclocal.m4
|
||||||
|
autom4te.cache/
|
||||||
|
compile
|
||||||
|
config.guess
|
||||||
|
config.log
|
||||||
|
config.status
|
||||||
|
config.sub
|
||||||
|
configure
|
||||||
|
configure.lineno
|
||||||
|
depcomp
|
||||||
|
install-sh
|
||||||
|
missing
|
||||||
|
stamp-h1
|
||||||
|
|
||||||
|
#libtool
|
||||||
|
ltmain.sh
|
||||||
|
libtool
|
||||||
|
.libs
|
||||||
|
|
||||||
|
.tarball-version
|
||||||
|
.version
|
||||||
|
|
||||||
|
*.pc
|
||||||
14
host/Makefile.am
Normal file
14
host/Makefile.am
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
AUTOMAKE_OPTIONS = foreign dist-bzip2 1.6
|
||||||
|
|
||||||
|
AM_CPPFLAGS = $(all_includes) -I$(top_srcdir)/include
|
||||||
|
SUBDIRS = include lib src contrib #tests examples doc
|
||||||
|
|
||||||
|
EXTRA_DIST = .version git-version-gen
|
||||||
|
|
||||||
|
@RELMAKE@
|
||||||
|
|
||||||
|
BUILT_SOURCES = $(top_srcdir)/.version
|
||||||
|
$(top_srcdir)/.version:
|
||||||
|
echo $(VERSION) > $@-t && mv $@-t $@
|
||||||
|
dist-hook:
|
||||||
|
echo $(VERSION) > $(distdir)/.tarball-version
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
LDFLAGS=`pkg-config --libs libusb-1.0 libosmocore` -losmocore
|
LDFLAGS+=`pkg-config --libs libusb-1.0 libosmocore` -pthread
|
||||||
CFLAGS=-Wall -g
|
CFLAGS=-Wall -g
|
||||||
|
|
||||||
APPS=simtrace2-remsim simtrace2-remsim-usb2udp simtrace2-list simtrace2-sniff
|
APPS=simtrace2-remsim simtrace2-remsim-usb2udp simtrace2-list simtrace2-sniff
|
||||||
|
|
||||||
all: $(APPS)
|
all: $(APPS)
|
||||||
|
|
||||||
simtrace2-remsim: simtrace2-remsim.o apdu_dispatch.o simtrace2-discovery.o libusb_util.o
|
simtrace2-remsim: simtrace2-remsim.o apdu_dispatch.o simtrace2-discovery.o simtrace2_api.o libusb_util.o
|
||||||
$(CC) -o $@ $^ $(LDFLAGS) -losmosim
|
$(CC) -o $@ $^ $(LDFLAGS) `pkg-config --libs libosmosim libpcsclite`
|
||||||
|
|
||||||
simtrace2-remsim-usb2udp: usb2udp.o simtrace2-discovery.o
|
simtrace2-remsim-usb2udp: usb2udp.o simtrace2-discovery.o
|
||||||
$(CC) -o $@ $^ $(LDFLAGS)
|
$(CC) -o $@ $^ $(LDFLAGS)
|
||||||
102
host/configure.ac
Normal file
102
host/configure.ac
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
AC_INIT([simtrace2],
|
||||||
|
m4_esyscmd([./git-version-gen .tarball-version]),
|
||||||
|
[simtrace@lists.osmocom.org])
|
||||||
|
|
||||||
|
dnl *This* is the root dir, even if an install-sh exists in ../ or ../../
|
||||||
|
AC_CONFIG_AUX_DIR([.])
|
||||||
|
|
||||||
|
AM_INIT_AUTOMAKE([foreign dist-bzip2 no-dist-gzip 1.6 subdir-objects])
|
||||||
|
AC_CONFIG_TESTDIR(tests)
|
||||||
|
|
||||||
|
dnl kernel style compile messages
|
||||||
|
m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])])
|
||||||
|
|
||||||
|
dnl include release helper
|
||||||
|
RELMAKE='-include osmo-release.mk'
|
||||||
|
AC_SUBST([RELMAKE])
|
||||||
|
|
||||||
|
dnl checks for programs
|
||||||
|
AC_PROG_MAKE_SET
|
||||||
|
AC_PROG_CC
|
||||||
|
AC_PROG_INSTALL
|
||||||
|
LT_INIT([pic-only])
|
||||||
|
|
||||||
|
dnl check for pkg-config (explained in detail in libosmocore/configure.ac)
|
||||||
|
AC_PATH_PROG(PKG_CONFIG_INSTALLED, pkg-config, no)
|
||||||
|
if test "x$PKG_CONFIG_INSTALLED" = "xno"; then
|
||||||
|
AC_MSG_WARN([You need to install pkg-config])
|
||||||
|
fi
|
||||||
|
PKG_PROG_PKG_CONFIG([0.20])
|
||||||
|
|
||||||
|
AC_CONFIG_MACRO_DIR([m4])
|
||||||
|
|
||||||
|
CFLAGS="$CFLAGS -Wall"
|
||||||
|
CPPFLAGS="$CPPFLAGS -Wall"
|
||||||
|
|
||||||
|
AC_ARG_ENABLE(sanitize,
|
||||||
|
[AS_HELP_STRING(
|
||||||
|
[--enable-sanitize],
|
||||||
|
[Compile with address sanitizer enabled],
|
||||||
|
)],
|
||||||
|
[sanitize=$enableval], [sanitize="no"])
|
||||||
|
if test x"$sanitize" = x"yes"
|
||||||
|
then
|
||||||
|
CFLAGS="$CFLAGS -fsanitize=address -fsanitize=undefined"
|
||||||
|
CPPFLAGS="$CPPFLAGS -fsanitize=address -fsanitize=undefined"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# The following test is taken from WebKit's webkit.m4
|
||||||
|
saved_CFLAGS="$CFLAGS"
|
||||||
|
CFLAGS="$CFLAGS -fvisibility=hidden "
|
||||||
|
AC_MSG_CHECKING([if ${CC} supports -fvisibility=hidden])
|
||||||
|
AC_COMPILE_IFELSE([AC_LANG_SOURCE([char foo;])],
|
||||||
|
[ AC_MSG_RESULT([yes])
|
||||||
|
SYMBOL_VISIBILITY="-fvisibility=hidden"],
|
||||||
|
AC_MSG_RESULT([no]))
|
||||||
|
CFLAGS="$saved_CFLAGS"
|
||||||
|
AC_SUBST(SYMBOL_VISIBILITY)
|
||||||
|
|
||||||
|
PKG_CHECK_MODULES(LIBOSMOCORE, libosmocore >= 1.0.0)
|
||||||
|
PKG_CHECK_MODULES(LIBOSMOSIM, libosmosim >= 1.0.0)
|
||||||
|
PKG_CHECK_MODULES(LIBUSB, libusb-1.0)
|
||||||
|
|
||||||
|
AC_ARG_ENABLE(sanitize,
|
||||||
|
[AS_HELP_STRING(
|
||||||
|
[--enable-sanitize],
|
||||||
|
[Compile with address sanitizer enabled],
|
||||||
|
)],
|
||||||
|
[sanitize=$enableval], [sanitize="no"])
|
||||||
|
if test x"$sanitize" = x"yes"
|
||||||
|
then
|
||||||
|
CFLAGS="$CFLAGS -fsanitize=address -fsanitize=undefined"
|
||||||
|
CPPFLAGS="$CPPFLAGS -fsanitize=address -fsanitize=undefined"
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_ARG_ENABLE(werror,
|
||||||
|
[AS_HELP_STRING(
|
||||||
|
[--enable-werror],
|
||||||
|
[Turn all compiler warnings into errors, with exceptions:
|
||||||
|
a) deprecation (allow upstream to mark deprecation without breaking builds);
|
||||||
|
b) "#warning" pragmas (allow to remind ourselves of errors without breaking builds)
|
||||||
|
]
|
||||||
|
)],
|
||||||
|
[werror=$enableval], [werror="no"])
|
||||||
|
if test x"$werror" = x"yes"
|
||||||
|
then
|
||||||
|
WERROR_FLAGS="-Werror"
|
||||||
|
WERROR_FLAGS+=" -Wno-error=deprecated -Wno-error=deprecated-declarations"
|
||||||
|
WERROR_FLAGS+=" -Wno-error=cpp" # "#warning"
|
||||||
|
CFLAGS="$CFLAGS $WERROR_FLAGS"
|
||||||
|
CPPFLAGS="$CPPFLAGS $WERROR_FLAGS"
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_MSG_RESULT([CFLAGS="$CFLAGS"])
|
||||||
|
AC_MSG_RESULT([CPPFLAGS="$CPPFLAGS"])
|
||||||
|
|
||||||
|
AC_OUTPUT(
|
||||||
|
libosmo-simtrace2.pc
|
||||||
|
include/Makefile
|
||||||
|
src/Makefile
|
||||||
|
lib/Makefile
|
||||||
|
contrib/Makefile
|
||||||
|
Makefile)
|
||||||
1
host/contrib/Makefile.am
Normal file
1
host/contrib/Makefile.am
Normal file
@@ -0,0 +1 @@
|
|||||||
|
EXTRA_DIST = 99-simtrace2.rules
|
||||||
151
host/git-version-gen
Executable file
151
host/git-version-gen
Executable file
@@ -0,0 +1,151 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# Print a version string.
|
||||||
|
scriptversion=2010-01-28.01
|
||||||
|
|
||||||
|
# Copyright (C) 2007-2010 Free Software Foundation, Inc.
|
||||||
|
#
|
||||||
|
# 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 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
# This script is derived from GIT-VERSION-GEN from GIT: http://git.or.cz/.
|
||||||
|
# It may be run two ways:
|
||||||
|
# - from a git repository in which the "git describe" command below
|
||||||
|
# produces useful output (thus requiring at least one signed tag)
|
||||||
|
# - from a non-git-repo directory containing a .tarball-version file, which
|
||||||
|
# presumes this script is invoked like "./git-version-gen .tarball-version".
|
||||||
|
|
||||||
|
# In order to use intra-version strings in your project, you will need two
|
||||||
|
# separate generated version string files:
|
||||||
|
#
|
||||||
|
# .tarball-version - present only in a distribution tarball, and not in
|
||||||
|
# a checked-out repository. Created with contents that were learned at
|
||||||
|
# the last time autoconf was run, and used by git-version-gen. Must not
|
||||||
|
# be present in either $(srcdir) or $(builddir) for git-version-gen to
|
||||||
|
# give accurate answers during normal development with a checked out tree,
|
||||||
|
# but must be present in a tarball when there is no version control system.
|
||||||
|
# Therefore, it cannot be used in any dependencies. GNUmakefile has
|
||||||
|
# hooks to force a reconfigure at distribution time to get the value
|
||||||
|
# correct, without penalizing normal development with extra reconfigures.
|
||||||
|
#
|
||||||
|
# .version - present in a checked-out repository and in a distribution
|
||||||
|
# tarball. Usable in dependencies, particularly for files that don't
|
||||||
|
# want to depend on config.h but do want to track version changes.
|
||||||
|
# Delete this file prior to any autoconf run where you want to rebuild
|
||||||
|
# files to pick up a version string change; and leave it stale to
|
||||||
|
# minimize rebuild time after unrelated changes to configure sources.
|
||||||
|
#
|
||||||
|
# It is probably wise to add these two files to .gitignore, so that you
|
||||||
|
# don't accidentally commit either generated file.
|
||||||
|
#
|
||||||
|
# Use the following line in your configure.ac, so that $(VERSION) will
|
||||||
|
# automatically be up-to-date each time configure is run (and note that
|
||||||
|
# since configure.ac no longer includes a version string, Makefile rules
|
||||||
|
# should not depend on configure.ac for version updates).
|
||||||
|
#
|
||||||
|
# AC_INIT([GNU project],
|
||||||
|
# m4_esyscmd([build-aux/git-version-gen .tarball-version]),
|
||||||
|
# [bug-project@example])
|
||||||
|
#
|
||||||
|
# Then use the following lines in your Makefile.am, so that .version
|
||||||
|
# will be present for dependencies, and so that .tarball-version will
|
||||||
|
# exist in distribution tarballs.
|
||||||
|
#
|
||||||
|
# BUILT_SOURCES = $(top_srcdir)/.version
|
||||||
|
# $(top_srcdir)/.version:
|
||||||
|
# echo $(VERSION) > $@-t && mv $@-t $@
|
||||||
|
# dist-hook:
|
||||||
|
# echo $(VERSION) > $(distdir)/.tarball-version
|
||||||
|
|
||||||
|
case $# in
|
||||||
|
1) ;;
|
||||||
|
*) echo 1>&2 "Usage: $0 \$srcdir/.tarball-version"; exit 1;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
tarball_version_file=$1
|
||||||
|
nl='
|
||||||
|
'
|
||||||
|
|
||||||
|
# First see if there is a tarball-only version file.
|
||||||
|
# then try "git describe", then default.
|
||||||
|
if test -f $tarball_version_file
|
||||||
|
then
|
||||||
|
v=`cat $tarball_version_file` || exit 1
|
||||||
|
case $v in
|
||||||
|
*$nl*) v= ;; # reject multi-line output
|
||||||
|
[0-9]*) ;;
|
||||||
|
*) v= ;;
|
||||||
|
esac
|
||||||
|
test -z "$v" \
|
||||||
|
&& echo "$0: WARNING: $tarball_version_file seems to be damaged" 1>&2
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test -n "$v"
|
||||||
|
then
|
||||||
|
: # use $v
|
||||||
|
elif
|
||||||
|
v=`git describe --abbrev=4 --match='v*' HEAD 2>/dev/null \
|
||||||
|
|| git describe --abbrev=4 HEAD 2>/dev/null` \
|
||||||
|
&& case $v in
|
||||||
|
[0-9]*) ;;
|
||||||
|
v[0-9]*) ;;
|
||||||
|
*) (exit 1) ;;
|
||||||
|
esac
|
||||||
|
then
|
||||||
|
# Is this a new git that lists number of commits since the last
|
||||||
|
# tag or the previous older version that did not?
|
||||||
|
# Newer: v6.10-77-g0f8faeb
|
||||||
|
# Older: v6.10-g0f8faeb
|
||||||
|
case $v in
|
||||||
|
*-*-*) : git describe is okay three part flavor ;;
|
||||||
|
*-*)
|
||||||
|
: git describe is older two part flavor
|
||||||
|
# Recreate the number of commits and rewrite such that the
|
||||||
|
# result is the same as if we were using the newer version
|
||||||
|
# of git describe.
|
||||||
|
vtag=`echo "$v" | sed 's/-.*//'`
|
||||||
|
numcommits=`git rev-list "$vtag"..HEAD | wc -l`
|
||||||
|
v=`echo "$v" | sed "s/\(.*\)-\(.*\)/\1-$numcommits-\2/"`;
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
# Change the first '-' to a '.', so version-comparing tools work properly.
|
||||||
|
# Remove the "g" in git describe's output string, to save a byte.
|
||||||
|
v=`echo "$v" | sed 's/-/./;s/\(.*\)-g/\1-/'`;
|
||||||
|
else
|
||||||
|
v=UNKNOWN
|
||||||
|
fi
|
||||||
|
|
||||||
|
v=`echo "$v" |sed 's/^v//'`
|
||||||
|
|
||||||
|
# Don't declare a version "dirty" merely because a time stamp has changed.
|
||||||
|
git status > /dev/null 2>&1
|
||||||
|
|
||||||
|
dirty=`sh -c 'git diff-index --name-only HEAD' 2>/dev/null` || dirty=
|
||||||
|
case "$dirty" in
|
||||||
|
'') ;;
|
||||||
|
*) # Append the suffix only if there isn't one already.
|
||||||
|
case $v in
|
||||||
|
*-dirty) ;;
|
||||||
|
*) v="$v-dirty" ;;
|
||||||
|
esac ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
# Omit the trailing newline, so that m4_esyscmd can use the result directly.
|
||||||
|
echo "$v" | tr -d '\012'
|
||||||
|
|
||||||
|
# Local variables:
|
||||||
|
# eval: (add-hook 'write-file-hooks 'time-stamp)
|
||||||
|
# time-stamp-start: "scriptversion="
|
||||||
|
# time-stamp-format: "%:y-%02m-%02d.%02H"
|
||||||
|
# time-stamp-end: "$"
|
||||||
|
# End:
|
||||||
7
host/include/Makefile.am
Normal file
7
host/include/Makefile.am
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
nobase_include_HEADERS = \
|
||||||
|
osmocom/simtrace2/apdu_dispatch.h \
|
||||||
|
osmocom/simtrace2/libusb_util.h \
|
||||||
|
osmocom/simtrace2/simtrace2_api.h \
|
||||||
|
osmocom/simtrace2/simtrace_usb.h \
|
||||||
|
osmocom/simtrace2/simtrace_prot.h \
|
||||||
|
osmocom/simtrace2/gsmtap.h
|
||||||
@@ -23,7 +23,7 @@
|
|||||||
|
|
||||||
#include <osmocom/sim/sim.h>
|
#include <osmocom/sim/sim.h>
|
||||||
|
|
||||||
struct apdu_context {
|
struct osmo_apdu_context {
|
||||||
struct osim_apdu_cmd_hdr hdr;
|
struct osim_apdu_cmd_hdr hdr;
|
||||||
uint8_t dc[256];
|
uint8_t dc[256];
|
||||||
uint8_t de[256];
|
uint8_t de[256];
|
||||||
@@ -39,11 +39,11 @@ struct apdu_context {
|
|||||||
} le;
|
} le;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum apdu_action {
|
enum osmo_apdu_action {
|
||||||
APDU_ACT_TX_CAPDU_TO_CARD = 0x0001,
|
APDU_ACT_TX_CAPDU_TO_CARD = 0x0001,
|
||||||
APDU_ACT_RX_MORE_CAPDU_FROM_READER = 0x0002,
|
APDU_ACT_RX_MORE_CAPDU_FROM_READER = 0x0002,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
int apdu_segment_in(struct apdu_context *ac, const uint8_t *apdu_buf,
|
int osmo_apdu_segment_in(struct osmo_apdu_context *ac, const uint8_t *apdu_buf,
|
||||||
unsigned int apdu_len, bool new_apdu);
|
unsigned int apdu_len, bool new_apdu);
|
||||||
6
host/include/osmocom/simtrace2/gsmtap.h
Normal file
6
host/include/osmocom/simtrace2/gsmtap.h
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
#pragma once
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <osmocom/core/gsmtap.h>
|
||||||
|
|
||||||
|
int osmo_st2_gsmtap_init(const char *gsmtap_host);
|
||||||
|
int osmo_st2_gsmtap_send_apdu(uint8_t sub_type, const uint8_t *apdu, unsigned int len);
|
||||||
@@ -68,3 +68,6 @@ int usb_match_interfaces(libusb_context *ctx, const struct dev_id *dev_ids,
|
|||||||
|
|
||||||
libusb_device_handle *usb_open_claim_interface(libusb_context *ctx,
|
libusb_device_handle *usb_open_claim_interface(libusb_context *ctx,
|
||||||
const struct usb_interface_match *ifm);
|
const struct usb_interface_match *ifm);
|
||||||
|
|
||||||
|
int get_usb_ep_addrs(libusb_device_handle *devh, unsigned int if_num,
|
||||||
|
uint8_t *out, uint8_t *in, uint8_t *irq);
|
||||||
58
host/include/osmocom/simtrace2/simtrace2_api.h
Normal file
58
host/include/osmocom/simtrace2/simtrace2_api.h
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <osmocom/sim/sim.h>
|
||||||
|
|
||||||
|
/* transport to a SIMtrace device */
|
||||||
|
struct osmo_st2_transport {
|
||||||
|
/* USB */
|
||||||
|
struct libusb_device_handle *usb_devh;
|
||||||
|
struct {
|
||||||
|
uint8_t in;
|
||||||
|
uint8_t out;
|
||||||
|
uint8_t irq_in;
|
||||||
|
} usb_ep;
|
||||||
|
|
||||||
|
/* UDP */
|
||||||
|
int udp_fd;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* a SIMtrace slot; communicates over a transport */
|
||||||
|
struct osmo_st2_slot {
|
||||||
|
/* transport through which the slot can be reached */
|
||||||
|
struct osmo_st2_transport *transp;
|
||||||
|
/* number of the slot within the transport */
|
||||||
|
uint8_t slot_nr;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* One istance of card emulation */
|
||||||
|
struct osmo_st2_cardem_inst {
|
||||||
|
/* slot on which this card emulation instance runs */
|
||||||
|
struct osmo_st2_slot *slot;
|
||||||
|
/* libosmosim SIM card profile */
|
||||||
|
const struct osim_cla_ins_card_profile *card_prof;
|
||||||
|
/* libosmosim SIM card channel */
|
||||||
|
struct osim_chan_hdl *chan;
|
||||||
|
};
|
||||||
|
|
||||||
|
int osmo_st2_transp_tx_msg(struct osmo_st2_transport *transp, struct msgb *msg);
|
||||||
|
|
||||||
|
int osmo_st2_slot_tx_msg(struct osmo_st2_slot *slot, struct msgb *msg,
|
||||||
|
uint8_t msg_class, uint8_t msg_type);
|
||||||
|
|
||||||
|
|
||||||
|
int osmo_st2_cardem_request_card_insert(struct osmo_st2_cardem_inst *ci, bool inserted);
|
||||||
|
int osmo_st2_cardem_request_pb_and_rx(struct osmo_st2_cardem_inst *ci, uint8_t pb, uint8_t le);
|
||||||
|
int osmo_st2_cardem_request_pb_and_tx(struct osmo_st2_cardem_inst *ci, uint8_t pb,
|
||||||
|
const uint8_t *data, uint16_t data_len_in);
|
||||||
|
int osmo_st2_cardem_request_sw_tx(struct osmo_st2_cardem_inst *ci, const uint8_t *sw);
|
||||||
|
int osmo_st2_cardem_request_set_atr(struct osmo_st2_cardem_inst *ci, const uint8_t *atr,
|
||||||
|
unsigned int atr_len);
|
||||||
|
|
||||||
|
|
||||||
|
int osmo_st2_modem_reset_pulse(struct osmo_st2_slot *slot, uint16_t duration_ms);
|
||||||
|
int osmo_st2_modem_reset_active(struct osmo_st2_slot *slot);
|
||||||
|
int osmo_st2_modem_reset_inactive(struct osmo_st2_slot *slot);
|
||||||
|
int osmo_st2_modem_sim_select_local(struct osmo_st2_slot *slot);
|
||||||
|
int osmo_st2_modem_sim_select_remote(struct osmo_st2_slot *slot);
|
||||||
|
int osmo_st2_modem_get_status(struct osmo_st2_slot *slot);
|
||||||
1
host/include/osmocom/simtrace2/simtrace_prot.h
Symbolic link
1
host/include/osmocom/simtrace2/simtrace_prot.h
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../../../../firmware/libcommon/include/simtrace_prot.h
|
||||||
1
host/include/osmocom/simtrace2/simtrace_usb.h
Symbolic link
1
host/include/osmocom/simtrace2/simtrace_usb.h
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../../../../firmware/libcommon/include/simtrace_usb.h
|
||||||
19
host/lib/Makefile.am
Normal file
19
host/lib/Makefile.am
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
# This is _NOT_ the library release version, it's an API version.
|
||||||
|
# Please read chapter "Library interface versions" of the libtool documentation
|
||||||
|
# before making any modifications: https://www.gnu.org/software/libtool/manual/html_node/Versioning.html
|
||||||
|
ST2_LIBVERSION=0:0:0
|
||||||
|
|
||||||
|
AM_CPPFLAGS = $(all_includes) -I$(top_srcdir)/include -I$(top_builddir)
|
||||||
|
AM_CFLAGS= -Wall $(LIBOSMOCORE_CFLAGS) $(LIBOSMOSIM_CFLAGS) $(LIBUSB_CFLAGS) $(COVERAGE_CFLAGS)
|
||||||
|
AM_LDFLAGS = $(COVERAGE_LDFLAGS)
|
||||||
|
COMMONLIBS = $(LIBOSMOCORE_LIBS) $(LIBOSMOSIM_LIBS) $(LIBUSB_LIBS)
|
||||||
|
|
||||||
|
lib_LTLIBRARIES = libosmo-simtrace2.la
|
||||||
|
|
||||||
|
libosmo_simtrace2_la_LDFLAGS = $(AM_LDFLAGS) -version-info $(ST2_LIBVERSION)
|
||||||
|
libosmo_simtrace2_la_LIBADD = $(COMMONLIBS)
|
||||||
|
libosmo_simtrace2_la_SOURCES = \
|
||||||
|
apdu_dispatch.c \
|
||||||
|
gsmtap.c \
|
||||||
|
libusb_util.c \
|
||||||
|
simtrace2_api.c
|
||||||
@@ -27,16 +27,16 @@
|
|||||||
#include <osmocom/sim/sim.h>
|
#include <osmocom/sim/sim.h>
|
||||||
#include <osmocom/sim/class_tables.h>
|
#include <osmocom/sim/class_tables.h>
|
||||||
|
|
||||||
#include "apdu_dispatch.h"
|
#include <osmocom/simtrace2/apdu_dispatch.h>
|
||||||
|
|
||||||
/*! \brief Has the command-data phase been completed yet? */
|
/*! \brief Has the command-data phase been completed yet? */
|
||||||
static inline bool is_dc_complete(struct apdu_context *ac)
|
static inline bool is_dc_complete(struct osmo_apdu_context *ac)
|
||||||
{
|
{
|
||||||
return (ac->lc.tot == ac->lc.cur);
|
return (ac->lc.tot == ac->lc.cur);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Has the expected-data phase been completed yet? */
|
/*! \brief Has the expected-data phase been completed yet? */
|
||||||
static inline bool is_de_complete(struct apdu_context *ac)
|
static inline bool is_de_complete(struct osmo_apdu_context *ac)
|
||||||
{
|
{
|
||||||
return (ac->le.tot == ac->le.cur);
|
return (ac->le.tot == ac->le.cur);
|
||||||
}
|
}
|
||||||
@@ -50,7 +50,7 @@ static const char *dump_apdu_hdr(const struct osim_apdu_cmd_hdr *h)
|
|||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dump_apdu_ctx(const struct apdu_context *ac)
|
static void dump_apdu_ctx(const struct osmo_apdu_context *ac)
|
||||||
{
|
{
|
||||||
printf("%s; case=%d, lc=%d(%d), le=%d(%d)\n",
|
printf("%s; case=%d, lc=%d(%d), le=%d(%d)\n",
|
||||||
dump_apdu_hdr(&ac->hdr), ac->apdu_case,
|
dump_apdu_hdr(&ac->hdr), ac->apdu_case,
|
||||||
@@ -71,8 +71,8 @@ static void dump_apdu_ctx(const struct apdu_context *ac)
|
|||||||
* The function retunrs APDU_ACT_RX_MORE_CAPDU_FROM_READER when there
|
* The function retunrs APDU_ACT_RX_MORE_CAPDU_FROM_READER when there
|
||||||
* is more data to be received from the card reader (GSM Phone).
|
* is more data to be received from the card reader (GSM Phone).
|
||||||
*/
|
*/
|
||||||
int apdu_segment_in(struct apdu_context *ac, const uint8_t *apdu_buf,
|
int osmo_apdu_segment_in(struct osmo_apdu_context *ac, const uint8_t *apdu_buf,
|
||||||
unsigned int apdu_len, bool new_apdu)
|
unsigned int apdu_len, bool new_apdu)
|
||||||
{
|
{
|
||||||
int rc = 0;
|
int rc = 0;
|
||||||
|
|
||||||
63
host/lib/gsmtap.c
Normal file
63
host/lib/gsmtap.c
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
#include <osmocom/simtrace2/gsmtap.h>
|
||||||
|
|
||||||
|
#include <osmocom/core/gsmtap.h>
|
||||||
|
#include <osmocom/core/gsmtap_util.h>
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
/*! global GSMTAP instance */
|
||||||
|
static struct gsmtap_inst *g_gti;
|
||||||
|
|
||||||
|
/*! initialize the global GSMTAP instance for SIM traces */
|
||||||
|
int osmo_st2_gsmtap_init(const char *gsmtap_host)
|
||||||
|
{
|
||||||
|
if (g_gti)
|
||||||
|
return -EEXIST;
|
||||||
|
|
||||||
|
g_gti = gsmtap_source_init(gsmtap_host, GSMTAP_UDP_PORT, 0);
|
||||||
|
if (!g_gti) {
|
||||||
|
perror("unable to open GSMTAP");
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
gsmtap_source_add_sink(g_gti);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! log one APDU via the global GSMTAP instance.
|
||||||
|
* \param[in] sub_type GSMTAP sub-type (GSMTAP_SIM_* constant)
|
||||||
|
* \param[in] apdu User-provided buffer with APDU to log
|
||||||
|
* \param[in] len Length of apdu in bytes
|
||||||
|
*/
|
||||||
|
int osmo_st2_gsmtap_send_apdu(uint8_t sub_type, const uint8_t *apdu, unsigned int len)
|
||||||
|
{
|
||||||
|
struct gsmtap_hdr *gh;
|
||||||
|
unsigned int gross_len = len + sizeof(*gh);
|
||||||
|
uint8_t *buf = malloc(gross_len);
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
if (!buf)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
memset(buf, 0, sizeof(*gh));
|
||||||
|
gh = (struct gsmtap_hdr *) buf;
|
||||||
|
gh->version = GSMTAP_VERSION;
|
||||||
|
gh->hdr_len = sizeof(*gh)/4;
|
||||||
|
gh->type = GSMTAP_TYPE_SIM;
|
||||||
|
gh->sub_type = sub_type;
|
||||||
|
|
||||||
|
memcpy(buf + sizeof(*gh), apdu, len);
|
||||||
|
|
||||||
|
rc = write(gsmtap_inst_fd(g_gti), buf, gross_len);
|
||||||
|
if (rc < 0) {
|
||||||
|
perror("write gsmtap");
|
||||||
|
free(buf);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(buf);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
/* libisb utilities
|
/* libusb utilities
|
||||||
*
|
*
|
||||||
* (C) 2010-2016 by Harald Welte <hwelte@hmw-consulting.de>
|
* (C) 2010-2019 by Harald Welte <hwelte@hmw-consulting.de>
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or
|
* This program is free software; you can redistribute it and/or
|
||||||
* modify it under the terms of the GNU General Public License
|
* modify it under the terms of the GNU General Public License
|
||||||
@@ -25,7 +25,7 @@
|
|||||||
|
|
||||||
#include <libusb.h>
|
#include <libusb.h>
|
||||||
|
|
||||||
#include "libusb_util.h"
|
#include <osmocom/simtrace2/libusb_util.h>
|
||||||
|
|
||||||
static char path_buf[USB_MAX_PATH_LEN];
|
static char path_buf[USB_MAX_PATH_LEN];
|
||||||
|
|
||||||
@@ -159,7 +159,7 @@ int dev_find_matching_interfaces(libusb_device *dev, int class, int sub_class, i
|
|||||||
out[out_idx].vendor = dev_desc.idVendor;
|
out[out_idx].vendor = dev_desc.idVendor;
|
||||||
out[out_idx].product = dev_desc.idProduct;
|
out[out_idx].product = dev_desc.idProduct;
|
||||||
out[out_idx].addr = addr;
|
out[out_idx].addr = addr;
|
||||||
strncpy(out[out_idx].path, path, sizeof(out[out_idx].path));
|
strncpy(out[out_idx].path, path, sizeof(out[out_idx].path)-1);
|
||||||
out[out_idx].path[sizeof(out[out_idx].path)-1] = '\0';
|
out[out_idx].path[sizeof(out[out_idx].path)-1] = '\0';
|
||||||
out[out_idx].configuration = conf_desc->bConfigurationValue;
|
out[out_idx].configuration = conf_desc->bConfigurationValue;
|
||||||
out[out_idx].interface = if_desc->bInterfaceNumber;
|
out[out_idx].interface = if_desc->bInterfaceNumber;
|
||||||
@@ -295,3 +295,44 @@ libusb_device_handle *usb_open_claim_interface(libusb_context *ctx,
|
|||||||
|
|
||||||
return usb_devh;
|
return usb_devh;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*! \brief obtain the endpoint addresses for a given USB interface */
|
||||||
|
int get_usb_ep_addrs(libusb_device_handle *devh, unsigned int if_num,
|
||||||
|
uint8_t *out, uint8_t *in, uint8_t *irq)
|
||||||
|
{
|
||||||
|
libusb_device *dev = libusb_get_device(devh);
|
||||||
|
struct libusb_config_descriptor *cdesc;
|
||||||
|
const struct libusb_interface_descriptor *idesc;
|
||||||
|
const struct libusb_interface *iface;
|
||||||
|
int rc, l;
|
||||||
|
|
||||||
|
rc = libusb_get_active_config_descriptor(dev, &cdesc);
|
||||||
|
if (rc < 0)
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
iface = &cdesc->interface[if_num];
|
||||||
|
/* FIXME: we assume there's no altsetting */
|
||||||
|
idesc = &iface->altsetting[0];
|
||||||
|
|
||||||
|
for (l = 0; l < idesc->bNumEndpoints; l++) {
|
||||||
|
const struct libusb_endpoint_descriptor *edesc = &idesc->endpoint[l];
|
||||||
|
switch (edesc->bmAttributes & 3) {
|
||||||
|
case LIBUSB_TRANSFER_TYPE_BULK:
|
||||||
|
if (edesc->bEndpointAddress & 0x80) {
|
||||||
|
if (in)
|
||||||
|
*in = edesc->bEndpointAddress;
|
||||||
|
} else {
|
||||||
|
if (out)
|
||||||
|
*out = edesc->bEndpointAddress;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case LIBUSB_TRANSFER_TYPE_INTERRUPT:
|
||||||
|
if (irq)
|
||||||
|
*irq = edesc->bEndpointAddress;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
281
host/lib/simtrace2_api.c
Normal file
281
host/lib/simtrace2_api.c
Normal file
@@ -0,0 +1,281 @@
|
|||||||
|
|
||||||
|
/* simtrace2-protocol - USB protocol library code for SIMtrace2
|
||||||
|
*
|
||||||
|
* (C) 2016-2019 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <time.h>
|
||||||
|
#define _GNU_SOURCE
|
||||||
|
#include <getopt.h>
|
||||||
|
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
|
||||||
|
#include <libusb.h>
|
||||||
|
|
||||||
|
//#include <osmocom/simtrace2/libusb_util.h>
|
||||||
|
#include <osmocom/simtrace2/simtrace_prot.h>
|
||||||
|
#include <osmocom/simtrace2/simtrace2_api.h>
|
||||||
|
//#include "apdu_dispatch.h"
|
||||||
|
//#include "simtrace2-discovery.h"
|
||||||
|
|
||||||
|
#include <osmocom/core/utils.h>
|
||||||
|
#include <osmocom/core/socket.h>
|
||||||
|
#include <osmocom/core/msgb.h>
|
||||||
|
#include <osmocom/sim/class_tables.h>
|
||||||
|
#include <osmocom/sim/sim.h>
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
* SIMTRACE core protocol
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
|
/*! \brief allocate a message buffer for simtrace use */
|
||||||
|
static struct msgb *st_msgb_alloc(void)
|
||||||
|
{
|
||||||
|
return msgb_alloc_headroom(1024+32, 32, "SIMtrace");
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
static void apdu_out_cb(uint8_t *buf, unsigned int len, void *user_data)
|
||||||
|
{
|
||||||
|
printf("APDU: %s\n", osmo_hexdump(buf, len));
|
||||||
|
gsmtap_send_sim(buf, len);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*! \brief Transmit a given command to the SIMtrace2 device */
|
||||||
|
int osmo_st2_transp_tx_msg(struct osmo_st2_transport *transp, struct msgb *msg)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
printf("<- %s\n", msgb_hexdump(msg));
|
||||||
|
|
||||||
|
if (transp->udp_fd < 0) {
|
||||||
|
int xfer_len;
|
||||||
|
|
||||||
|
rc = libusb_bulk_transfer(transp->usb_devh, transp->usb_ep.out,
|
||||||
|
msgb_data(msg), msgb_length(msg),
|
||||||
|
&xfer_len, 100000);
|
||||||
|
} else {
|
||||||
|
rc = write(transp->udp_fd, msgb_data(msg), msgb_length(msg));
|
||||||
|
}
|
||||||
|
|
||||||
|
msgb_free(msg);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct simtrace_msg_hdr *st_push_hdr(struct msgb *msg, uint8_t msg_class, uint8_t msg_type,
|
||||||
|
uint8_t slot_nr)
|
||||||
|
{
|
||||||
|
struct simtrace_msg_hdr *sh;
|
||||||
|
|
||||||
|
sh = (struct simtrace_msg_hdr *) msgb_push(msg, sizeof(*sh));
|
||||||
|
memset(sh, 0, sizeof(*sh));
|
||||||
|
sh->msg_class = msg_class;
|
||||||
|
sh->msg_type = msg_type;
|
||||||
|
sh->slot_nr = slot_nr;
|
||||||
|
sh->msg_len = msgb_length(msg);
|
||||||
|
|
||||||
|
return sh;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* transmit a given message to a specified slot. Expects all headers
|
||||||
|
* present before calling the function */
|
||||||
|
int osmo_st2_slot_tx_msg(struct osmo_st2_slot *slot, struct msgb *msg,
|
||||||
|
uint8_t msg_class, uint8_t msg_type)
|
||||||
|
{
|
||||||
|
st_push_hdr(msg, msg_class, msg_type, slot->slot_nr);
|
||||||
|
|
||||||
|
return osmo_st2_transp_tx_msg(slot->transp, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
* Card Emulation protocol
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
|
|
||||||
|
/*! \brief Request the SIMtrace2 to generate a card-insert signal */
|
||||||
|
int osmo_st2_cardem_request_card_insert(struct osmo_st2_cardem_inst *ci, bool inserted)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct cardemu_usb_msg_cardinsert *cins;
|
||||||
|
|
||||||
|
cins = (struct cardemu_usb_msg_cardinsert *) msgb_put(msg, sizeof(*cins));
|
||||||
|
memset(cins, 0, sizeof(*cins));
|
||||||
|
if (inserted)
|
||||||
|
cins->card_insert = 1;
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_CARDINSERT);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Request the SIMtrace2 to transmit a Procedure Byte, then Rx */
|
||||||
|
int osmo_st2_cardem_request_pb_and_rx(struct osmo_st2_cardem_inst *ci, uint8_t pb, uint8_t le)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct cardemu_usb_msg_tx_data *txd;
|
||||||
|
txd = (struct cardemu_usb_msg_tx_data *) msgb_put(msg, sizeof(*txd));
|
||||||
|
|
||||||
|
printf("<= %s(%02x, %d)\n", __func__, pb, le);
|
||||||
|
|
||||||
|
memset(txd, 0, sizeof(*txd));
|
||||||
|
txd->data_len = 1;
|
||||||
|
txd->flags = CEMU_DATA_F_PB_AND_RX;
|
||||||
|
/* one data byte */
|
||||||
|
msgb_put_u8(msg, pb);
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_TX_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Request the SIMtrace2 to transmit a Procedure Byte, then Tx */
|
||||||
|
int osmo_st2_cardem_request_pb_and_tx(struct osmo_st2_cardem_inst *ci, uint8_t pb,
|
||||||
|
const uint8_t *data, uint16_t data_len_in)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct cardemu_usb_msg_tx_data *txd;
|
||||||
|
uint8_t *cur;
|
||||||
|
|
||||||
|
txd = (struct cardemu_usb_msg_tx_data *) msgb_put(msg, sizeof(*txd));
|
||||||
|
|
||||||
|
printf("<= %s(%02x, %s, %d)\n", __func__, pb,
|
||||||
|
osmo_hexdump(data, data_len_in), data_len_in);
|
||||||
|
|
||||||
|
memset(txd, 0, sizeof(*txd));
|
||||||
|
txd->data_len = 1 + data_len_in;
|
||||||
|
txd->flags = CEMU_DATA_F_PB_AND_TX;
|
||||||
|
/* procedure byte */
|
||||||
|
msgb_put_u8(msg, pb);
|
||||||
|
/* data */
|
||||||
|
cur = msgb_put(msg, data_len_in);
|
||||||
|
memcpy(cur, data, data_len_in);
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_TX_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Request the SIMtrace2 to send a Status Word */
|
||||||
|
int osmo_st2_cardem_request_sw_tx(struct osmo_st2_cardem_inst *ci, const uint8_t *sw)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct cardemu_usb_msg_tx_data *txd;
|
||||||
|
uint8_t *cur;
|
||||||
|
|
||||||
|
txd = (struct cardemu_usb_msg_tx_data *) msgb_put(msg, sizeof(*txd));
|
||||||
|
|
||||||
|
printf("<= %s(%02x %02x)\n", __func__, sw[0], sw[1]);
|
||||||
|
|
||||||
|
memset(txd, 0, sizeof(*txd));
|
||||||
|
txd->data_len = 2;
|
||||||
|
txd->flags = CEMU_DATA_F_PB_AND_TX | CEMU_DATA_F_FINAL;
|
||||||
|
cur = msgb_put(msg, 2);
|
||||||
|
cur[0] = sw[0];
|
||||||
|
cur[1] = sw[1];
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_TX_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
|
int osmo_st2_cardem_request_set_atr(struct osmo_st2_cardem_inst *ci, const uint8_t *atr, unsigned int atr_len)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct cardemu_usb_msg_set_atr *satr;
|
||||||
|
uint8_t *cur;
|
||||||
|
|
||||||
|
satr = (struct cardemu_usb_msg_set_atr *) msgb_put(msg, sizeof(*satr));
|
||||||
|
|
||||||
|
printf("<= %s(%s)\n", __func__, osmo_hexdump(atr, atr_len));
|
||||||
|
|
||||||
|
memset(satr, 0, sizeof(*satr));
|
||||||
|
satr->atr_len = atr_len;
|
||||||
|
cur = msgb_put(msg, atr_len);
|
||||||
|
memcpy(cur, atr, atr_len);
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_SET_ATR);
|
||||||
|
}
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
* Modem Control protocol
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
|
static int _modem_reset(struct osmo_st2_slot *slot, uint8_t asserted, uint16_t pulse_ms)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct st_modem_reset *sr ;
|
||||||
|
|
||||||
|
sr = (struct st_modem_reset *) msgb_put(msg, sizeof(*sr));
|
||||||
|
sr->asserted = asserted;
|
||||||
|
sr->pulse_duration_msec = pulse_ms;
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(slot, msg, SIMTRACE_MSGC_MODEM, SIMTRACE_MSGT_DT_MODEM_RESET);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief pulse the RESET line of the modem for \a duration_ms milli-seconds*/
|
||||||
|
int osmo_st2_modem_reset_pulse(struct osmo_st2_slot *slot, uint16_t duration_ms)
|
||||||
|
{
|
||||||
|
return _modem_reset(slot, 2, duration_ms);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief assert the RESET line of the modem */
|
||||||
|
int osmo_st2_modem_reset_active(struct osmo_st2_slot *slot)
|
||||||
|
{
|
||||||
|
return _modem_reset(slot, 1, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief de-assert the RESET line of the modem */
|
||||||
|
int osmo_st2_modem_reset_inactive(struct osmo_st2_slot *slot)
|
||||||
|
{
|
||||||
|
return _modem_reset(slot, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _modem_sim_select(struct osmo_st2_slot *slot, uint8_t remote_sim)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
struct st_modem_sim_select *ss;
|
||||||
|
|
||||||
|
ss = (struct st_modem_sim_select *) msgb_put(msg, sizeof(*ss));
|
||||||
|
ss->remote_sim = remote_sim;
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(slot, msg, SIMTRACE_MSGC_MODEM, SIMTRACE_MSGT_DT_MODEM_SIM_SELECT);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief select local (physical) SIM for given slot */
|
||||||
|
int osmo_st2_modem_sim_select_local(struct osmo_st2_slot *slot)
|
||||||
|
{
|
||||||
|
return _modem_sim_select(slot, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief select remote (emulated/forwarded) SIM for given slot */
|
||||||
|
int osmo_st2_modem_sim_select_remote(struct osmo_st2_slot *slot)
|
||||||
|
{
|
||||||
|
return _modem_sim_select(slot, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Request slot to send us status information about the modem */
|
||||||
|
int osmo_st2_modem_get_status(struct osmo_st2_slot *slot)
|
||||||
|
{
|
||||||
|
struct msgb *msg = st_msgb_alloc();
|
||||||
|
|
||||||
|
return osmo_st2_slot_tx_msg(slot, msg, SIMTRACE_MSGC_MODEM, SIMTRACE_MSGT_BD_MODEM_STATUS);
|
||||||
|
}
|
||||||
10
host/libosmo-simtrace2.pc.in
Normal file
10
host/libosmo-simtrace2.pc.in
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
prefix=@prefix@
|
||||||
|
exec_prefix=@exec_prefix@
|
||||||
|
libdir=@libdir@
|
||||||
|
includedir=@includedir@
|
||||||
|
|
||||||
|
Name: Osmocom SIMtrace2 library
|
||||||
|
Description: Library for SIM Card / Smart Card tracing + emulation
|
||||||
|
Version: @VERSION@
|
||||||
|
Libs: -L${libdir} -losmo-simtrace2
|
||||||
|
Cflags: -I${includedir}/
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
#ifndef _SIMTRACE_H
|
|
||||||
#define _SIMTRACE_H
|
|
||||||
|
|
||||||
#define SIMTRACE_USB_VENDOR 0x1d50
|
|
||||||
#define SIMTRACE_USB_PRODUCT 0x60e3
|
|
||||||
|
|
||||||
#endif
|
|
||||||
@@ -1,94 +0,0 @@
|
|||||||
/* simtrace2-discovery - host PC library to scan for matching USB
|
|
||||||
* devices
|
|
||||||
*
|
|
||||||
* (C) 2016 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
||||||
*/
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
#include <libusb.h>
|
|
||||||
|
|
||||||
/*! \brief obtain the endpoint addresses for a given USB interface */
|
|
||||||
int get_usb_ep_addrs(libusb_device_handle *devh, unsigned int if_num,
|
|
||||||
uint8_t *out, uint8_t *in, uint8_t *irq)
|
|
||||||
{
|
|
||||||
libusb_device *dev = libusb_get_device(devh);
|
|
||||||
struct libusb_config_descriptor *cdesc;
|
|
||||||
const struct libusb_interface_descriptor *idesc;
|
|
||||||
const struct libusb_interface *iface;
|
|
||||||
int rc, l;
|
|
||||||
|
|
||||||
rc = libusb_get_active_config_descriptor(dev, &cdesc);
|
|
||||||
if (rc < 0)
|
|
||||||
return rc;
|
|
||||||
|
|
||||||
iface = &cdesc->interface[if_num];
|
|
||||||
/* FIXME: we assume there's no altsetting */
|
|
||||||
idesc = &iface->altsetting[0];
|
|
||||||
|
|
||||||
for (l = 0; l < idesc->bNumEndpoints; l++) {
|
|
||||||
const struct libusb_endpoint_descriptor *edesc = &idesc->endpoint[l];
|
|
||||||
switch (edesc->bmAttributes & 3) {
|
|
||||||
case LIBUSB_TRANSFER_TYPE_BULK:
|
|
||||||
if (edesc->bEndpointAddress & 0x80) {
|
|
||||||
if (in)
|
|
||||||
*in = edesc->bEndpointAddress;
|
|
||||||
} else {
|
|
||||||
if (out)
|
|
||||||
*out = edesc->bEndpointAddress;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case LIBUSB_TRANSFER_TYPE_INTERRUPT:
|
|
||||||
if (irq)
|
|
||||||
*irq = edesc->bEndpointAddress;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
struct libusb_device_descriptor ddesc;
|
|
||||||
int rc, i, j, k;
|
|
||||||
|
|
||||||
rc = libusb_get_device_descriptor(devh, &ddesc);
|
|
||||||
if (rc < 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (i = 0; i < ddesc.bNumConfigurations; i++) {
|
|
||||||
struct libusb_config_descriptor *cdesc;
|
|
||||||
rc = libusb_get_config_descriptor(devh, i, &cdesc);
|
|
||||||
if (rc < 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
for (j = 0; j < cdesc->bNumInterfaces; j++) {
|
|
||||||
const struct libusb_interface *iface = cdesc->interface[j];
|
|
||||||
for (k = 0; k < iface->num_altsetting; k++) {
|
|
||||||
const struct libusb_interface_descriptor *idesc = iface->altsetting[k];
|
|
||||||
/* make sure this is the interface we're looking for */
|
|
||||||
if (idesc->bInterfaceClass != 0xFF ||
|
|
||||||
idesc->bInterfaceSubClass != if_class ||
|
|
||||||
idsec->bInterfaceProtocol != if_proto)
|
|
||||||
continue;
|
|
||||||
/* FIXME */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
libusb_free_config_descriptor(cdesc);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
/* simtrace2-discovery - host PC library to scan for matching USB
|
|
||||||
* devices
|
|
||||||
*
|
|
||||||
* (C) 2016 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
||||||
*/
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <libusb.h>
|
|
||||||
|
|
||||||
int get_usb_ep_addrs(libusb_device_handle *devh, unsigned int if_num,
|
|
||||||
uint8_t *out, uint8_t *in, uint8_t *irq);
|
|
||||||
@@ -1,772 +0,0 @@
|
|||||||
/* simtrace2-remsim - main program for the host PC to provide a remote SIM
|
|
||||||
* using the SIMtrace 2 firmware in card emulation mode
|
|
||||||
*
|
|
||||||
* (C) 2016 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
||||||
*/
|
|
||||||
#include <errno.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
#include <signal.h>
|
|
||||||
#include <time.h>
|
|
||||||
#define _GNU_SOURCE
|
|
||||||
#include <getopt.h>
|
|
||||||
|
|
||||||
#include <sys/time.h>
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/socket.h>
|
|
||||||
#include <netinet/in.h>
|
|
||||||
#include <arpa/inet.h>
|
|
||||||
|
|
||||||
#include <libusb.h>
|
|
||||||
|
|
||||||
#include "libusb_util.h"
|
|
||||||
#include "simtrace.h"
|
|
||||||
#include "simtrace_prot.h"
|
|
||||||
#include "apdu_dispatch.h"
|
|
||||||
#include "simtrace2-discovery.h"
|
|
||||||
|
|
||||||
#include <osmocom/core/gsmtap.h>
|
|
||||||
#include <osmocom/core/gsmtap_util.h>
|
|
||||||
#include <osmocom/core/utils.h>
|
|
||||||
#include <osmocom/core/socket.h>
|
|
||||||
#include <osmocom/core/msgb.h>
|
|
||||||
#include <osmocom/sim/class_tables.h>
|
|
||||||
#include <osmocom/sim/sim.h>
|
|
||||||
|
|
||||||
/* transport to a SIMtrace device */
|
|
||||||
struct st_transport {
|
|
||||||
/* USB */
|
|
||||||
struct libusb_device_handle *usb_devh;
|
|
||||||
struct {
|
|
||||||
uint8_t in;
|
|
||||||
uint8_t out;
|
|
||||||
uint8_t irq_in;
|
|
||||||
} usb_ep;
|
|
||||||
|
|
||||||
/* UDP */
|
|
||||||
int udp_fd;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* a SIMtrace slot; communicates over a transport */
|
|
||||||
struct st_slot {
|
|
||||||
/* transport through which the slot can be reached */
|
|
||||||
struct st_transport *transp;
|
|
||||||
/* number of the slot within the transport */
|
|
||||||
uint8_t slot_nr;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* One istance of card emulation */
|
|
||||||
struct cardem_inst {
|
|
||||||
/* slot on which this card emulation instance runs */
|
|
||||||
struct st_slot *slot;
|
|
||||||
/* libosmosim SIM card profile */
|
|
||||||
const struct osim_cla_ins_card_profile *card_prof;
|
|
||||||
/* libosmosim SIM card channel */
|
|
||||||
struct osim_chan_hdl *chan;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* global GSMTAP instance */
|
|
||||||
static struct gsmtap_inst *g_gti;
|
|
||||||
|
|
||||||
static int gsmtap_send_sim(const uint8_t *apdu, unsigned int len)
|
|
||||||
{
|
|
||||||
struct gsmtap_hdr *gh;
|
|
||||||
unsigned int gross_len = len + sizeof(*gh);
|
|
||||||
uint8_t *buf = malloc(gross_len);
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
if (!buf)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
memset(buf, 0, sizeof(*gh));
|
|
||||||
gh = (struct gsmtap_hdr *) buf;
|
|
||||||
gh->version = GSMTAP_VERSION;
|
|
||||||
gh->hdr_len = sizeof(*gh)/4;
|
|
||||||
gh->type = GSMTAP_TYPE_SIM;
|
|
||||||
|
|
||||||
memcpy(buf + sizeof(*gh), apdu, len);
|
|
||||||
|
|
||||||
rc = write(gsmtap_inst_fd(g_gti), buf, gross_len);
|
|
||||||
if (rc < 0) {
|
|
||||||
perror("write gsmtap");
|
|
||||||
free(buf);
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
free(buf);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* SIMTRACE pcore protocol
|
|
||||||
***********************************************************************/
|
|
||||||
|
|
||||||
/*! \brief allocate a message buffer for simtrace use */
|
|
||||||
static struct msgb *st_msgb_alloc(void)
|
|
||||||
{
|
|
||||||
return msgb_alloc_headroom(1024+32, 32, "SIMtrace");
|
|
||||||
}
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
static void apdu_out_cb(uint8_t *buf, unsigned int len, void *user_data)
|
|
||||||
{
|
|
||||||
printf("APDU: %s\n", osmo_hexdump(buf, len));
|
|
||||||
gsmtap_send_sim(buf, len);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*! \brief Transmit a given command to the SIMtrace2 device */
|
|
||||||
int st_transp_tx_msg(struct st_transport *transp, struct msgb *msg)
|
|
||||||
{
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
printf("<- %s\n", msgb_hexdump(msg));
|
|
||||||
|
|
||||||
if (transp->udp_fd < 0) {
|
|
||||||
int xfer_len;
|
|
||||||
|
|
||||||
rc = libusb_bulk_transfer(transp->usb_devh, transp->usb_ep.out,
|
|
||||||
msgb_data(msg), msgb_length(msg),
|
|
||||||
&xfer_len, 100000);
|
|
||||||
} else {
|
|
||||||
rc = write(transp->udp_fd, msgb_data(msg), msgb_length(msg));
|
|
||||||
}
|
|
||||||
|
|
||||||
msgb_free(msg);
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct simtrace_msg_hdr *st_push_hdr(struct msgb *msg, uint8_t msg_class, uint8_t msg_type,
|
|
||||||
uint8_t slot_nr)
|
|
||||||
{
|
|
||||||
struct simtrace_msg_hdr *sh;
|
|
||||||
|
|
||||||
sh = (struct simtrace_msg_hdr *) msgb_push(msg, sizeof(*sh));
|
|
||||||
memset(sh, 0, sizeof(*sh));
|
|
||||||
sh->msg_class = msg_class;
|
|
||||||
sh->msg_type = msg_type;
|
|
||||||
sh->slot_nr = slot_nr;
|
|
||||||
sh->msg_len = msgb_length(msg);
|
|
||||||
|
|
||||||
return sh;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* transmit a given message to a specified slot. Expects all headers
|
|
||||||
* present before calling the function */
|
|
||||||
int st_slot_tx_msg(struct st_slot *slot, struct msgb *msg,
|
|
||||||
uint8_t msg_class, uint8_t msg_type)
|
|
||||||
{
|
|
||||||
st_push_hdr(msg, msg_class, msg_type, slot->slot_nr);
|
|
||||||
|
|
||||||
return st_transp_tx_msg(slot->transp, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* Card Emulation protocol
|
|
||||||
***********************************************************************/
|
|
||||||
|
|
||||||
|
|
||||||
/*! \brief Request the SIMtrace2 to generate a card-insert signal */
|
|
||||||
static int cardem_request_card_insert(struct cardem_inst *ci, bool inserted)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct cardemu_usb_msg_cardinsert *cins;
|
|
||||||
|
|
||||||
cins = (struct cardemu_usb_msg_cardinsert *) msgb_put(msg, sizeof(*cins));
|
|
||||||
memset(cins, 0, sizeof(*cins));
|
|
||||||
if (inserted)
|
|
||||||
cins->card_insert = 1;
|
|
||||||
|
|
||||||
return st_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_CARDINSERT);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Request the SIMtrace2 to transmit a Procedure Byte, then Rx */
|
|
||||||
static int cardem_request_pb_and_rx(struct cardem_inst *ci, uint8_t pb, uint8_t le)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct cardemu_usb_msg_tx_data *txd;
|
|
||||||
txd = (struct cardemu_usb_msg_tx_data *) msgb_put(msg, sizeof(*txd));
|
|
||||||
|
|
||||||
printf("<= %s(%02x, %d)\n", __func__, pb, le);
|
|
||||||
|
|
||||||
memset(txd, 0, sizeof(*txd));
|
|
||||||
txd->data_len = 1;
|
|
||||||
txd->flags = CEMU_DATA_F_PB_AND_RX;
|
|
||||||
/* one data byte */
|
|
||||||
msgb_put_u8(msg, pb);
|
|
||||||
|
|
||||||
return st_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_TX_DATA);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Request the SIMtrace2 to transmit a Procedure Byte, then Tx */
|
|
||||||
static int cardem_request_pb_and_tx(struct cardem_inst *ci, uint8_t pb,
|
|
||||||
const uint8_t *data, uint8_t data_len_in)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct cardemu_usb_msg_tx_data *txd;
|
|
||||||
uint8_t *cur;
|
|
||||||
|
|
||||||
txd = (struct cardemu_usb_msg_tx_data *) msgb_put(msg, sizeof(*txd));
|
|
||||||
|
|
||||||
printf("<= %s(%02x, %s, %d)\n", __func__, pb,
|
|
||||||
osmo_hexdump(data, data_len_in), data_len_in);
|
|
||||||
|
|
||||||
memset(txd, 0, sizeof(*txd));
|
|
||||||
txd->data_len = 1 + data_len_in;
|
|
||||||
txd->flags = CEMU_DATA_F_PB_AND_TX;
|
|
||||||
/* procedure byte */
|
|
||||||
msgb_put_u8(msg, pb);
|
|
||||||
/* data */
|
|
||||||
cur = msgb_put(msg, data_len_in);
|
|
||||||
memcpy(cur, data, data_len_in);
|
|
||||||
|
|
||||||
return st_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_TX_DATA);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Request the SIMtrace2 to send a Status Word */
|
|
||||||
static int cardem_request_sw_tx(struct cardem_inst *ci, const uint8_t *sw)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct cardemu_usb_msg_tx_data *txd;
|
|
||||||
uint8_t *cur;
|
|
||||||
|
|
||||||
txd = (struct cardemu_usb_msg_tx_data *) msgb_put(msg, sizeof(*txd));
|
|
||||||
|
|
||||||
printf("<= %s(%02x %02x)\n", __func__, sw[0], sw[1]);
|
|
||||||
|
|
||||||
memset(txd, 0, sizeof(*txd));
|
|
||||||
txd->data_len = 2;
|
|
||||||
txd->flags = CEMU_DATA_F_PB_AND_TX | CEMU_DATA_F_FINAL;
|
|
||||||
cur = msgb_put(msg, 2);
|
|
||||||
cur[0] = sw[0];
|
|
||||||
cur[1] = sw[1];
|
|
||||||
|
|
||||||
return st_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_TX_DATA);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void atr_update_csum(uint8_t *atr, unsigned int atr_len)
|
|
||||||
{
|
|
||||||
uint8_t csum = 0;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 1; i < atr_len - 1; i++)
|
|
||||||
csum = csum ^ atr[i];
|
|
||||||
|
|
||||||
atr[atr_len-1] = csum;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int cardem_request_set_atr(struct cardem_inst *ci, const uint8_t *atr, unsigned int atr_len)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct cardemu_usb_msg_set_atr *satr;
|
|
||||||
uint8_t *cur;
|
|
||||||
|
|
||||||
satr = (struct cardemu_usb_msg_set_atr *) msgb_put(msg, sizeof(*satr));
|
|
||||||
|
|
||||||
printf("<= %s(%s)\n", __func__, osmo_hexdump(atr, atr_len));
|
|
||||||
|
|
||||||
memset(satr, 0, sizeof(*satr));
|
|
||||||
satr->atr_len = atr_len;
|
|
||||||
cur = msgb_put(msg, atr_len);
|
|
||||||
memcpy(cur, atr, atr_len);
|
|
||||||
|
|
||||||
return st_slot_tx_msg(ci->slot, msg, SIMTRACE_MSGC_CARDEM, SIMTRACE_MSGT_DT_CEMU_SET_ATR);
|
|
||||||
}
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* Modem Control protocol
|
|
||||||
***********************************************************************/
|
|
||||||
|
|
||||||
static int _modem_reset(struct st_slot *slot, uint8_t asserted, uint16_t pulse_ms)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct st_modem_reset *sr ;
|
|
||||||
|
|
||||||
sr = (struct st_modem_reset *) msgb_put(msg, sizeof(*sr));
|
|
||||||
sr->asserted = asserted;
|
|
||||||
sr->pulse_duration_msec = pulse_ms;
|
|
||||||
|
|
||||||
return st_slot_tx_msg(slot, msg, SIMTRACE_MSGC_MODEM, SIMTRACE_MSGT_DT_MODEM_RESET);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief pulse the RESET line of the modem for \a duration_ms milli-seconds*/
|
|
||||||
int st_modem_reset_pulse(struct st_slot *slot, uint16_t duration_ms)
|
|
||||||
{
|
|
||||||
return _modem_reset(slot, 2, duration_ms);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief assert the RESET line of the modem */
|
|
||||||
int st_modem_reset_active(struct st_slot *slot)
|
|
||||||
{
|
|
||||||
return _modem_reset(slot, 1, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief de-assert the RESET line of the modem */
|
|
||||||
int st_modem_reset_inactive(struct st_slot *slot)
|
|
||||||
{
|
|
||||||
return _modem_reset(slot, 0, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int _modem_sim_select(struct st_slot *slot, uint8_t remote_sim)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
struct st_modem_sim_select *ss;
|
|
||||||
|
|
||||||
ss = (struct st_modem_sim_select *) msgb_put(msg, sizeof(*ss));
|
|
||||||
ss->remote_sim = remote_sim;
|
|
||||||
|
|
||||||
return st_slot_tx_msg(slot, msg, SIMTRACE_MSGC_MODEM, SIMTRACE_MSGT_DT_MODEM_SIM_SELECT);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief select local (physical) SIM for given slot */
|
|
||||||
int st_modem_sim_select_local(struct st_slot *slot)
|
|
||||||
{
|
|
||||||
return _modem_sim_select(slot, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief select remote (emulated/forwarded) SIM for given slot */
|
|
||||||
int st_modem_sim_select_remote(struct st_slot *slot)
|
|
||||||
{
|
|
||||||
return _modem_sim_select(slot, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Request slot to send us status information about the modem */
|
|
||||||
int st_modem_get_status(struct st_slot *slot)
|
|
||||||
{
|
|
||||||
struct msgb *msg = st_msgb_alloc();
|
|
||||||
|
|
||||||
return st_slot_tx_msg(slot, msg, SIMTRACE_MSGC_MODEM, SIMTRACE_MSGT_BD_MODEM_STATUS);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/***********************************************************************
|
|
||||||
* Incoming Messages
|
|
||||||
***********************************************************************/
|
|
||||||
|
|
||||||
/*! \brief Process a STATUS message from the SIMtrace2 */
|
|
||||||
static int process_do_status(struct cardem_inst *ci, uint8_t *buf, int len)
|
|
||||||
{
|
|
||||||
struct cardemu_usb_msg_status *status;
|
|
||||||
status = (struct cardemu_usb_msg_status *) buf;
|
|
||||||
|
|
||||||
printf("=> STATUS: flags=0x%x, fi=%u, di=%u, wi=%u wtime=%u\n",
|
|
||||||
status->flags, status->fi, status->di, status->wi,
|
|
||||||
status->waiting_time);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Process a PTS indication message from the SIMtrace2 */
|
|
||||||
static int process_do_pts(struct cardem_inst *ci, uint8_t *buf, int len)
|
|
||||||
{
|
|
||||||
struct cardemu_usb_msg_pts_info *pts;
|
|
||||||
pts = (struct cardemu_usb_msg_pts_info *) buf;
|
|
||||||
|
|
||||||
printf("=> PTS req: %s\n", osmo_hexdump(pts->req, sizeof(pts->req)));
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Process a ERROR indication message from the SIMtrace2 */
|
|
||||||
static int process_do_error(struct cardem_inst *ci, uint8_t *buf, int len)
|
|
||||||
{
|
|
||||||
struct cardemu_usb_msg_error *err;
|
|
||||||
err = (struct cardemu_usb_msg_error *) buf;
|
|
||||||
|
|
||||||
printf("=> ERROR: %u/%u/%u: %s\n",
|
|
||||||
err->severity, err->subsystem, err->code,
|
|
||||||
err->msg_len ? (char *)err->msg : "");
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! \brief Process a RX-DATA indication message from the SIMtrace2 */
|
|
||||||
static int process_do_rx_da(struct cardem_inst *ci, uint8_t *buf, int len)
|
|
||||||
{
|
|
||||||
static struct apdu_context ac;
|
|
||||||
struct cardemu_usb_msg_rx_data *data;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
data = (struct cardemu_usb_msg_rx_data *) buf;
|
|
||||||
|
|
||||||
printf("=> DATA: flags=%x, %s: ", data->flags,
|
|
||||||
osmo_hexdump(data->data, data->data_len));
|
|
||||||
|
|
||||||
rc = apdu_segment_in(&ac, data->data, data->data_len,
|
|
||||||
data->flags & CEMU_DATA_F_TPDU_HDR);
|
|
||||||
|
|
||||||
if (rc & APDU_ACT_TX_CAPDU_TO_CARD) {
|
|
||||||
struct msgb *tmsg = msgb_alloc(1024, "TPDU");
|
|
||||||
struct osim_reader_hdl *rh = ci->chan->card->reader;
|
|
||||||
uint8_t *cur;
|
|
||||||
|
|
||||||
/* Copy TPDU header */
|
|
||||||
cur = msgb_put(tmsg, sizeof(ac.hdr));
|
|
||||||
memcpy(cur, &ac.hdr, sizeof(ac.hdr));
|
|
||||||
/* Copy D(c), if any */
|
|
||||||
if (ac.lc.tot) {
|
|
||||||
cur = msgb_put(tmsg, ac.lc.tot);
|
|
||||||
memcpy(cur, ac.dc, ac.lc.tot);
|
|
||||||
}
|
|
||||||
/* send to actual card */
|
|
||||||
tmsg->l3h = tmsg->tail;
|
|
||||||
rc = rh->ops->transceive(rh, tmsg);
|
|
||||||
if (rc < 0) {
|
|
||||||
fprintf(stderr, "error during transceive: %d\n", rc);
|
|
||||||
msgb_free(tmsg);
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
msgb_apdu_sw(tmsg) = msgb_get_u16(tmsg);
|
|
||||||
ac.sw[0] = msgb_apdu_sw(tmsg) >> 8;
|
|
||||||
ac.sw[1] = msgb_apdu_sw(tmsg) & 0xff;
|
|
||||||
printf("SW=0x%04x, len_rx=%d\n", msgb_apdu_sw(tmsg), msgb_l3len(tmsg));
|
|
||||||
if (msgb_l3len(tmsg))
|
|
||||||
cardem_request_pb_and_tx(ci, ac.hdr.ins, tmsg->l3h, msgb_l3len(tmsg));
|
|
||||||
cardem_request_sw_tx(ci, ac.sw);
|
|
||||||
} else if (ac.lc.tot > ac.lc.cur) {
|
|
||||||
cardem_request_pb_and_rx(ci, ac.hdr.ins, ac.lc.tot - ac.lc.cur);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if 0
|
|
||||||
case SIMTRACE_CMD_DO_ERROR
|
|
||||||
rc = process_do_error(ci, buf, len);
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*! \brief Process an incoming message from the SIMtrace2 */
|
|
||||||
static int process_usb_msg(struct cardem_inst *ci, uint8_t *buf, int len)
|
|
||||||
{
|
|
||||||
struct simtrace_msg_hdr *sh = (struct simtrace_msg_hdr *)buf;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
printf("-> %s\n", osmo_hexdump(buf, len));
|
|
||||||
|
|
||||||
buf += sizeof(*sh);
|
|
||||||
|
|
||||||
switch (sh->msg_type) {
|
|
||||||
case SIMTRACE_MSGT_BD_CEMU_STATUS:
|
|
||||||
rc = process_do_status(ci, buf, len);
|
|
||||||
break;
|
|
||||||
case SIMTRACE_MSGT_DO_CEMU_PTS:
|
|
||||||
rc = process_do_pts(ci, buf, len);
|
|
||||||
break;
|
|
||||||
case SIMTRACE_MSGT_DO_CEMU_RX_DATA:
|
|
||||||
rc = process_do_rx_da(ci, buf, len);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
printf("unknown simtrace msg type 0x%02x\n", sh->msg_type);
|
|
||||||
rc = -1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void print_welcome(void)
|
|
||||||
{
|
|
||||||
printf("simtrace2-remsim - Remote SIM card forwarding\n"
|
|
||||||
"(C) 2010-2017 by Harald Welte <laforge@gnumonks.org>\n\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void print_help(void)
|
|
||||||
{
|
|
||||||
printf( "\t-r\t--remote-udp-host HOST\n"
|
|
||||||
"\t-p\t--remote-udp-port PORT\n"
|
|
||||||
"\t-h\t--help\n"
|
|
||||||
"\t-i\t--gsmtap-ip\tA.B.C.D\n"
|
|
||||||
"\t-a\t--skip-atr\n"
|
|
||||||
"\t-k\t--keep-running\n"
|
|
||||||
"\t-V\t--usb-vendor\tVENDOR_ID\n"
|
|
||||||
"\t-P\t--usb-product\tPRODUCT_ID\n"
|
|
||||||
"\t-C\t--usb-config\tCONFIG_ID\n"
|
|
||||||
"\t-I\t--usb-interface\tINTERFACE_ID\n"
|
|
||||||
"\t-S\t--usb-altsetting ALTSETTING_ID\n"
|
|
||||||
"\t-A\t--usb-address\tADDRESS\n"
|
|
||||||
"\n"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
static const struct option opts[] = {
|
|
||||||
{ "remote-udp-host", 1, 0, 'r' },
|
|
||||||
{ "remote-udp-port", 1, 0, 'p' },
|
|
||||||
{ "gsmtap-ip", 1, 0, 'i' },
|
|
||||||
{ "skip-atr", 0, 0, 'a' },
|
|
||||||
{ "help", 0, 0, 'h' },
|
|
||||||
{ "keep-running", 0, 0, 'k' },
|
|
||||||
{ "usb-vendor", 1, 0, 'V' },
|
|
||||||
{ "usb-product", 1, 0, 'P' },
|
|
||||||
{ "usb-config", 1, 0, 'C' },
|
|
||||||
{ "usb-interface", 1, 0, 'I' },
|
|
||||||
{ "usb-altsetting", 1, 0, 'S' },
|
|
||||||
{ "usb-address", 1, 0, 'A' },
|
|
||||||
{ "usb-path", 1, 0, 'H' },
|
|
||||||
{ NULL, 0, 0, 0 }
|
|
||||||
};
|
|
||||||
|
|
||||||
static void run_mainloop(struct cardem_inst *ci)
|
|
||||||
{
|
|
||||||
struct st_transport *transp = ci->slot->transp;
|
|
||||||
unsigned int msg_count, byte_count = 0;
|
|
||||||
uint8_t buf[16*265];
|
|
||||||
int xfer_len;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
printf("Entering main loop\n");
|
|
||||||
|
|
||||||
while (1) {
|
|
||||||
/* read data from SIMtrace2 device (local or via USB) */
|
|
||||||
if (transp->udp_fd < 0) {
|
|
||||||
rc = libusb_bulk_transfer(transp->usb_devh, transp->usb_ep.in,
|
|
||||||
buf, sizeof(buf), &xfer_len, 100);
|
|
||||||
if (rc < 0 && rc != LIBUSB_ERROR_TIMEOUT &&
|
|
||||||
rc != LIBUSB_ERROR_INTERRUPTED &&
|
|
||||||
rc != LIBUSB_ERROR_IO) {
|
|
||||||
fprintf(stderr, "BULK IN transfer error; rc=%d\n", rc);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
rc = read(transp->udp_fd, buf, sizeof(buf));
|
|
||||||
if (rc <= 0) {
|
|
||||||
fprintf(stderr, "shor read from UDP\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
xfer_len = rc;
|
|
||||||
}
|
|
||||||
/* dispatch any incoming data */
|
|
||||||
if (xfer_len > 0) {
|
|
||||||
printf("URB: %s\n", osmo_hexdump(buf, xfer_len));
|
|
||||||
process_usb_msg(ci, buf, xfer_len);
|
|
||||||
msg_count++;
|
|
||||||
byte_count += xfer_len;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct st_transport _transp;
|
|
||||||
|
|
||||||
static struct st_slot _slot = {
|
|
||||||
.transp = &_transp,
|
|
||||||
.slot_nr = 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
struct cardem_inst _ci = {
|
|
||||||
.slot = &_slot,
|
|
||||||
};
|
|
||||||
|
|
||||||
struct cardem_inst *ci = &_ci;
|
|
||||||
|
|
||||||
static void signal_handler(int signal)
|
|
||||||
{
|
|
||||||
switch (signal) {
|
|
||||||
case SIGINT:
|
|
||||||
cardem_request_card_insert(ci, false);
|
|
||||||
exit(0);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int argc, char **argv)
|
|
||||||
{
|
|
||||||
struct st_transport *transp = ci->slot->transp;
|
|
||||||
char *gsmtap_host = "127.0.0.1";
|
|
||||||
int rc;
|
|
||||||
int c, ret = 1;
|
|
||||||
int skip_atr = 0;
|
|
||||||
int keep_running = 0;
|
|
||||||
int remote_udp_port = 52342;
|
|
||||||
int if_num = 0, vendor_id = -1, product_id = -1;
|
|
||||||
int config_id = -1, altsetting = 0, addr = -1;
|
|
||||||
char *remote_udp_host = NULL;
|
|
||||||
char *path = NULL;
|
|
||||||
struct osim_reader_hdl *reader;
|
|
||||||
struct osim_card_hdl *card;
|
|
||||||
|
|
||||||
print_welcome();
|
|
||||||
|
|
||||||
while (1) {
|
|
||||||
int option_index = 0;
|
|
||||||
|
|
||||||
c = getopt_long(argc, argv, "r:p:hi:V:P:C:I:S:A:H:ak", opts, &option_index);
|
|
||||||
if (c == -1)
|
|
||||||
break;
|
|
||||||
switch (c) {
|
|
||||||
case 'r':
|
|
||||||
remote_udp_host = optarg;
|
|
||||||
break;
|
|
||||||
case 'p':
|
|
||||||
remote_udp_port = atoi(optarg);
|
|
||||||
break;
|
|
||||||
case 'h':
|
|
||||||
print_help();
|
|
||||||
exit(0);
|
|
||||||
break;
|
|
||||||
case 'i':
|
|
||||||
gsmtap_host = optarg;
|
|
||||||
break;
|
|
||||||
case 'a':
|
|
||||||
skip_atr = 1;
|
|
||||||
break;
|
|
||||||
case 'k':
|
|
||||||
keep_running = 1;
|
|
||||||
break;
|
|
||||||
case 'V':
|
|
||||||
vendor_id = strtol(optarg, NULL, 16);
|
|
||||||
break;
|
|
||||||
case 'P':
|
|
||||||
product_id = strtol(optarg, NULL, 16);
|
|
||||||
break;
|
|
||||||
case 'C':
|
|
||||||
config_id = atoi(optarg);
|
|
||||||
break;
|
|
||||||
case 'I':
|
|
||||||
if_num = atoi(optarg);
|
|
||||||
break;
|
|
||||||
case 'S':
|
|
||||||
altsetting = atoi(optarg);
|
|
||||||
break;
|
|
||||||
case 'A':
|
|
||||||
addr = atoi(optarg);
|
|
||||||
break;
|
|
||||||
case 'H':
|
|
||||||
path = optarg;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!remote_udp_host && (vendor_id < 0 || product_id < 0)) {
|
|
||||||
fprintf(stderr, "You have to specify the vendor and product ID\n");
|
|
||||||
goto do_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
transp->udp_fd = -1;
|
|
||||||
|
|
||||||
ci->card_prof = &osim_uicc_sim_cic_profile;
|
|
||||||
|
|
||||||
if (!remote_udp_host) {
|
|
||||||
rc = libusb_init(NULL);
|
|
||||||
if (rc < 0) {
|
|
||||||
fprintf(stderr, "libusb initialization failed\n");
|
|
||||||
goto do_exit;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
transp->udp_fd = osmo_sock_init(AF_INET, SOCK_DGRAM, IPPROTO_UDP,
|
|
||||||
remote_udp_host, remote_udp_port+if_num,
|
|
||||||
OSMO_SOCK_F_CONNECT);
|
|
||||||
if (transp->udp_fd < 0) {
|
|
||||||
fprintf(stderr, "error binding UDP port\n");
|
|
||||||
goto do_exit;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
g_gti = gsmtap_source_init(gsmtap_host, GSMTAP_UDP_PORT, 0);
|
|
||||||
if (!g_gti) {
|
|
||||||
perror("unable to open GSMTAP");
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
gsmtap_source_add_sink(g_gti);
|
|
||||||
|
|
||||||
reader = osim_reader_open(OSIM_READER_DRV_PCSC, 0, "", NULL);
|
|
||||||
if (!reader) {
|
|
||||||
perror("unable to open PC/SC reader");
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
card = osim_card_open(reader, OSIM_PROTO_T0);
|
|
||||||
if (!card) {
|
|
||||||
perror("unable to open SIM card");
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
ci->chan = llist_entry(card->channels.next, struct osim_chan_hdl, list);
|
|
||||||
if (!ci->chan) {
|
|
||||||
perror("SIM card has no channel?!?");
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
signal(SIGINT, &signal_handler);
|
|
||||||
|
|
||||||
do {
|
|
||||||
if (transp->udp_fd < 0) {
|
|
||||||
struct usb_interface_match _ifm, *ifm = &_ifm;
|
|
||||||
ifm->vendor = vendor_id;
|
|
||||||
ifm->product = product_id;
|
|
||||||
ifm->configuration = config_id;
|
|
||||||
ifm->interface = if_num;
|
|
||||||
ifm->altsetting = altsetting;
|
|
||||||
ifm->addr = addr;
|
|
||||||
if (path)
|
|
||||||
osmo_strlcpy(ifm->path, path, sizeof(ifm->path));
|
|
||||||
transp->usb_devh = usb_open_claim_interface(NULL, ifm);
|
|
||||||
if (!transp->usb_devh) {
|
|
||||||
fprintf(stderr, "can't open USB device\n");
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
rc = libusb_claim_interface(transp->usb_devh, if_num);
|
|
||||||
if (rc < 0) {
|
|
||||||
fprintf(stderr, "can't claim interface %d; rc=%d\n", if_num, rc);
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
|
|
||||||
rc = get_usb_ep_addrs(transp->usb_devh, if_num, &transp->usb_ep.out,
|
|
||||||
&transp->usb_ep.in, &transp->usb_ep.irq_in);
|
|
||||||
if (rc < 0) {
|
|
||||||
fprintf(stderr, "can't obtain EP addrs; rc=%d\n", rc);
|
|
||||||
goto close_exit;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* simulate card-insert to modem (owhw, not qmod) */
|
|
||||||
cardem_request_card_insert(ci, true);
|
|
||||||
|
|
||||||
/* select remote (forwarded) SIM */
|
|
||||||
st_modem_sim_select_remote(ci->slot);
|
|
||||||
|
|
||||||
/* set the ATR */
|
|
||||||
uint8_t real_atr[] = { 0x3B, 0x9F, 0x96, 0x80, 0x1F, 0xC7, 0x80, 0x31,
|
|
||||||
0xA0, 0x73, 0xBE, 0x21, 0x13, 0x67, 0x43, 0x20,
|
|
||||||
0x07, 0x18, 0x00, 0x00, 0x01, 0xA5 };
|
|
||||||
atr_update_csum(real_atr, sizeof(real_atr));
|
|
||||||
cardem_request_set_atr(ci, real_atr, sizeof(real_atr));
|
|
||||||
|
|
||||||
/* select remote (forwarded) SIM */
|
|
||||||
st_modem_reset_pulse(ci->slot, 300);
|
|
||||||
|
|
||||||
run_mainloop(ci);
|
|
||||||
ret = 0;
|
|
||||||
|
|
||||||
if (transp->udp_fd < 0)
|
|
||||||
libusb_release_interface(transp->usb_devh, 0);
|
|
||||||
close_exit:
|
|
||||||
if (transp->usb_devh)
|
|
||||||
libusb_close(transp->usb_devh);
|
|
||||||
if (keep_running)
|
|
||||||
sleep(1);
|
|
||||||
} while (keep_running);
|
|
||||||
|
|
||||||
if (transp->udp_fd < 0)
|
|
||||||
libusb_exit(NULL);
|
|
||||||
do_exit:
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
../firmware/libcommon/include/simtrace_prot.h
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
../firmware/libcommon/include/simtrace_usb.h
|
|
||||||
16
host/src/Makefile.am
Normal file
16
host/src/Makefile.am
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
AM_CPPFLAGS = $(all_includes) -I$(top_srcdir)/include
|
||||||
|
AM_CFLAGS=-Wall -g $(LIBOSMOCORE_CFLAGS) $(LIBOSMOSIM_CFLAGS) $(LIBUSB_CFLAGS) $(COVERAGE_FLAGS)
|
||||||
|
AM_LDFLAGS=$(COVERAGE_LDFLAGS)
|
||||||
|
|
||||||
|
LDADD= $(top_builddir)/lib/libosmo-simtrace2.la \
|
||||||
|
$(LIBOSMOCORE_LIBS) $(LIBOSMOSIM_LIBS) $(LIBUSB_LIBS)
|
||||||
|
|
||||||
|
bin_PROGRAMS = simtrace2-remsim simtrace2-remsim-usb2udp simtrace2-list simtrace2-sniff
|
||||||
|
|
||||||
|
simtrace2_remsim_SOURCES = simtrace2-remsim.c
|
||||||
|
|
||||||
|
simtrace2_remsim_usb2udp_SOURCES = usb2udp.c
|
||||||
|
|
||||||
|
simtrace2_list_SOURCES = simtrace2_usb.c
|
||||||
|
|
||||||
|
simtrace2_sniff_SOURCES = simtrace2-sniff.c
|
||||||
461
host/src/simtrace2-remsim.c
Normal file
461
host/src/simtrace2-remsim.c
Normal file
@@ -0,0 +1,461 @@
|
|||||||
|
/* simtrace2-remsim - main program for the host PC to provide a remote SIM
|
||||||
|
* using the SIMtrace 2 firmware in card emulation mode
|
||||||
|
*
|
||||||
|
* (C) 2016-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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <time.h>
|
||||||
|
#define _GNU_SOURCE
|
||||||
|
#include <getopt.h>
|
||||||
|
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
|
||||||
|
#include <libusb.h>
|
||||||
|
|
||||||
|
#include <osmocom/simtrace2/libusb_util.h>
|
||||||
|
#include <osmocom/simtrace2/simtrace2_api.h>
|
||||||
|
#include <osmocom/simtrace2/simtrace_prot.h>
|
||||||
|
#include <osmocom/simtrace2/apdu_dispatch.h>
|
||||||
|
#include <osmocom/simtrace2/gsmtap.h>
|
||||||
|
|
||||||
|
#include <osmocom/core/utils.h>
|
||||||
|
#include <osmocom/core/socket.h>
|
||||||
|
#include <osmocom/core/msgb.h>
|
||||||
|
#include <osmocom/sim/class_tables.h>
|
||||||
|
#include <osmocom/sim/sim.h>
|
||||||
|
|
||||||
|
static void atr_update_csum(uint8_t *atr, unsigned int atr_len)
|
||||||
|
{
|
||||||
|
uint8_t csum = 0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 1; i < atr_len - 1; i++)
|
||||||
|
csum = csum ^ atr[i];
|
||||||
|
|
||||||
|
atr[atr_len-1] = csum;
|
||||||
|
}
|
||||||
|
|
||||||
|
/***********************************************************************
|
||||||
|
* Incoming Messages
|
||||||
|
***********************************************************************/
|
||||||
|
|
||||||
|
/*! \brief Process a STATUS message from the SIMtrace2 */
|
||||||
|
static int process_do_status(struct osmo_st2_cardem_inst *ci, uint8_t *buf, int len)
|
||||||
|
{
|
||||||
|
struct cardemu_usb_msg_status *status;
|
||||||
|
status = (struct cardemu_usb_msg_status *) buf;
|
||||||
|
|
||||||
|
printf("=> STATUS: flags=0x%x, fi=%u, di=%u, wi=%u wtime=%u\n",
|
||||||
|
status->flags, status->fi, status->di, status->wi,
|
||||||
|
status->waiting_time);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Process a PTS indication message from the SIMtrace2 */
|
||||||
|
static int process_do_pts(struct osmo_st2_cardem_inst *ci, uint8_t *buf, int len)
|
||||||
|
{
|
||||||
|
struct cardemu_usb_msg_pts_info *pts;
|
||||||
|
pts = (struct cardemu_usb_msg_pts_info *) buf;
|
||||||
|
|
||||||
|
printf("=> PTS req: %s\n", osmo_hexdump(pts->req, sizeof(pts->req)));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Process a RX-DATA indication message from the SIMtrace2 */
|
||||||
|
static int process_do_rx_da(struct osmo_st2_cardem_inst *ci, uint8_t *buf, int len)
|
||||||
|
{
|
||||||
|
static struct osmo_apdu_context ac;
|
||||||
|
struct cardemu_usb_msg_rx_data *data;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
data = (struct cardemu_usb_msg_rx_data *) buf;
|
||||||
|
|
||||||
|
printf("=> DATA: flags=%x, %s: ", data->flags,
|
||||||
|
osmo_hexdump(data->data, data->data_len));
|
||||||
|
|
||||||
|
rc = osmo_apdu_segment_in(&ac, data->data, data->data_len,
|
||||||
|
data->flags & CEMU_DATA_F_TPDU_HDR);
|
||||||
|
|
||||||
|
if (rc & APDU_ACT_TX_CAPDU_TO_CARD) {
|
||||||
|
struct msgb *tmsg = msgb_alloc(1024, "TPDU");
|
||||||
|
struct osim_reader_hdl *rh = ci->chan->card->reader;
|
||||||
|
uint8_t *cur;
|
||||||
|
|
||||||
|
/* Copy TPDU header */
|
||||||
|
cur = msgb_put(tmsg, sizeof(ac.hdr));
|
||||||
|
memcpy(cur, &ac.hdr, sizeof(ac.hdr));
|
||||||
|
/* Copy D(c), if any */
|
||||||
|
if (ac.lc.tot) {
|
||||||
|
cur = msgb_put(tmsg, ac.lc.tot);
|
||||||
|
memcpy(cur, ac.dc, ac.lc.tot);
|
||||||
|
}
|
||||||
|
/* send to actual card */
|
||||||
|
tmsg->l3h = tmsg->tail;
|
||||||
|
rc = rh->ops->transceive(rh, tmsg);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "error during transceive: %d\n", rc);
|
||||||
|
msgb_free(tmsg);
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
msgb_apdu_sw(tmsg) = msgb_get_u16(tmsg);
|
||||||
|
ac.sw[0] = msgb_apdu_sw(tmsg) >> 8;
|
||||||
|
ac.sw[1] = msgb_apdu_sw(tmsg) & 0xff;
|
||||||
|
printf("SW=0x%04x, len_rx=%d\n", msgb_apdu_sw(tmsg), msgb_l3len(tmsg));
|
||||||
|
if (msgb_l3len(tmsg))
|
||||||
|
osmo_st2_cardem_request_pb_and_tx(ci, ac.hdr.ins, tmsg->l3h, msgb_l3len(tmsg));
|
||||||
|
osmo_st2_cardem_request_sw_tx(ci, ac.sw);
|
||||||
|
} else if (ac.lc.tot > ac.lc.cur) {
|
||||||
|
osmo_st2_cardem_request_pb_and_rx(ci, ac.hdr.ins, ac.lc.tot - ac.lc.cur);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Process an incoming message from the SIMtrace2 */
|
||||||
|
static int process_usb_msg(struct osmo_st2_cardem_inst *ci, uint8_t *buf, int len)
|
||||||
|
{
|
||||||
|
struct simtrace_msg_hdr *sh = (struct simtrace_msg_hdr *)buf;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
printf("-> %s\n", osmo_hexdump(buf, len));
|
||||||
|
|
||||||
|
buf += sizeof(*sh);
|
||||||
|
|
||||||
|
switch (sh->msg_type) {
|
||||||
|
case SIMTRACE_MSGT_BD_CEMU_STATUS:
|
||||||
|
rc = process_do_status(ci, buf, len);
|
||||||
|
break;
|
||||||
|
case SIMTRACE_MSGT_DO_CEMU_PTS:
|
||||||
|
rc = process_do_pts(ci, buf, len);
|
||||||
|
break;
|
||||||
|
case SIMTRACE_MSGT_DO_CEMU_RX_DATA:
|
||||||
|
rc = process_do_rx_da(ci, buf, len);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
printf("unknown simtrace msg type 0x%02x\n", sh->msg_type);
|
||||||
|
rc = -1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void print_welcome(void)
|
||||||
|
{
|
||||||
|
printf("simtrace2-remsim - Remote SIM card forwarding\n"
|
||||||
|
"(C) 2010-2017, Harald Welte <laforge@gnumonks.org>\n"
|
||||||
|
"(C) 2018, sysmocom -s.f.m.c. GmbH, Author: Kevin Redon <kredon@sysmocom.de>\n\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void print_help(void)
|
||||||
|
{
|
||||||
|
printf( "\t-r\t--remote-udp-host HOST\n"
|
||||||
|
"\t-p\t--remote-udp-port PORT\n"
|
||||||
|
"\t-h\t--help\n"
|
||||||
|
"\t-i\t--gsmtap-ip\tA.B.C.D\n"
|
||||||
|
"\t-a\t--skip-atr\n"
|
||||||
|
"\t-k\t--keep-running\n"
|
||||||
|
"\t-V\t--usb-vendor\tVENDOR_ID\n"
|
||||||
|
"\t-P\t--usb-product\tPRODUCT_ID\n"
|
||||||
|
"\t-C\t--usb-config\tCONFIG_ID\n"
|
||||||
|
"\t-I\t--usb-interface\tINTERFACE_ID\n"
|
||||||
|
"\t-S\t--usb-altsetting ALTSETTING_ID\n"
|
||||||
|
"\t-A\t--usb-address\tADDRESS\n"
|
||||||
|
"\t-H\t--usb-path\tPATH\n"
|
||||||
|
"\n"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct option opts[] = {
|
||||||
|
{ "remote-udp-host", 1, 0, 'r' },
|
||||||
|
{ "remote-udp-port", 1, 0, 'p' },
|
||||||
|
{ "gsmtap-ip", 1, 0, 'i' },
|
||||||
|
{ "skip-atr", 0, 0, 'a' },
|
||||||
|
{ "help", 0, 0, 'h' },
|
||||||
|
{ "keep-running", 0, 0, 'k' },
|
||||||
|
{ "usb-vendor", 1, 0, 'V' },
|
||||||
|
{ "usb-product", 1, 0, 'P' },
|
||||||
|
{ "usb-config", 1, 0, 'C' },
|
||||||
|
{ "usb-interface", 1, 0, 'I' },
|
||||||
|
{ "usb-altsetting", 1, 0, 'S' },
|
||||||
|
{ "usb-address", 1, 0, 'A' },
|
||||||
|
{ "usb-path", 1, 0, 'H' },
|
||||||
|
{ NULL, 0, 0, 0 }
|
||||||
|
};
|
||||||
|
|
||||||
|
static void run_mainloop(struct osmo_st2_cardem_inst *ci)
|
||||||
|
{
|
||||||
|
struct osmo_st2_transport *transp = ci->slot->transp;
|
||||||
|
unsigned int msg_count, byte_count = 0;
|
||||||
|
uint8_t buf[16*265];
|
||||||
|
int xfer_len;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
printf("Entering main loop\n");
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
/* read data from SIMtrace2 device (local or via USB) */
|
||||||
|
if (transp->udp_fd < 0) {
|
||||||
|
rc = libusb_bulk_transfer(transp->usb_devh, transp->usb_ep.in,
|
||||||
|
buf, sizeof(buf), &xfer_len, 100);
|
||||||
|
if (rc < 0 && rc != LIBUSB_ERROR_TIMEOUT &&
|
||||||
|
rc != LIBUSB_ERROR_INTERRUPTED &&
|
||||||
|
rc != LIBUSB_ERROR_IO) {
|
||||||
|
fprintf(stderr, "BULK IN transfer error; rc=%d\n", rc);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
rc = read(transp->udp_fd, buf, sizeof(buf));
|
||||||
|
if (rc <= 0) {
|
||||||
|
fprintf(stderr, "shor read from UDP\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
xfer_len = rc;
|
||||||
|
}
|
||||||
|
/* dispatch any incoming data */
|
||||||
|
if (xfer_len > 0) {
|
||||||
|
printf("URB: %s\n", osmo_hexdump(buf, xfer_len));
|
||||||
|
process_usb_msg(ci, buf, xfer_len);
|
||||||
|
msg_count++;
|
||||||
|
byte_count += xfer_len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct osmo_st2_transport _transp;
|
||||||
|
|
||||||
|
static struct osmo_st2_slot _slot = {
|
||||||
|
.transp = &_transp,
|
||||||
|
.slot_nr = 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct osmo_st2_cardem_inst _ci = {
|
||||||
|
.slot = &_slot,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct osmo_st2_cardem_inst *ci = &_ci;
|
||||||
|
|
||||||
|
static void signal_handler(int signal)
|
||||||
|
{
|
||||||
|
switch (signal) {
|
||||||
|
case SIGINT:
|
||||||
|
osmo_st2_cardem_request_card_insert(ci, false);
|
||||||
|
exit(0);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
struct osmo_st2_transport *transp = ci->slot->transp;
|
||||||
|
char *gsmtap_host = "127.0.0.1";
|
||||||
|
int rc;
|
||||||
|
int c, ret = 1;
|
||||||
|
int skip_atr = 0;
|
||||||
|
int keep_running = 0;
|
||||||
|
int remote_udp_port = 52342;
|
||||||
|
int if_num = 0, vendor_id = -1, product_id = -1;
|
||||||
|
int config_id = -1, altsetting = 0, addr = -1;
|
||||||
|
char *remote_udp_host = NULL;
|
||||||
|
char *path = NULL;
|
||||||
|
struct osim_reader_hdl *reader;
|
||||||
|
struct osim_card_hdl *card;
|
||||||
|
|
||||||
|
print_welcome();
|
||||||
|
|
||||||
|
while (1) {
|
||||||
|
int option_index = 0;
|
||||||
|
|
||||||
|
c = getopt_long(argc, argv, "r:p:hi:V:P:C:I:S:A:H:ak", opts, &option_index);
|
||||||
|
if (c == -1)
|
||||||
|
break;
|
||||||
|
switch (c) {
|
||||||
|
case 'r':
|
||||||
|
remote_udp_host = optarg;
|
||||||
|
break;
|
||||||
|
case 'p':
|
||||||
|
remote_udp_port = atoi(optarg);
|
||||||
|
break;
|
||||||
|
case 'h':
|
||||||
|
print_help();
|
||||||
|
exit(0);
|
||||||
|
break;
|
||||||
|
case 'i':
|
||||||
|
gsmtap_host = optarg;
|
||||||
|
break;
|
||||||
|
case 'a':
|
||||||
|
skip_atr = 1;
|
||||||
|
break;
|
||||||
|
case 'k':
|
||||||
|
keep_running = 1;
|
||||||
|
break;
|
||||||
|
case 'V':
|
||||||
|
vendor_id = strtol(optarg, NULL, 16);
|
||||||
|
break;
|
||||||
|
case 'P':
|
||||||
|
product_id = strtol(optarg, NULL, 16);
|
||||||
|
break;
|
||||||
|
case 'C':
|
||||||
|
config_id = atoi(optarg);
|
||||||
|
break;
|
||||||
|
case 'I':
|
||||||
|
if_num = atoi(optarg);
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
altsetting = atoi(optarg);
|
||||||
|
break;
|
||||||
|
case 'A':
|
||||||
|
addr = atoi(optarg);
|
||||||
|
break;
|
||||||
|
case 'H':
|
||||||
|
path = optarg;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!remote_udp_host && (vendor_id < 0 || product_id < 0)) {
|
||||||
|
fprintf(stderr, "You have to specify the vendor and product ID\n");
|
||||||
|
goto do_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
transp->udp_fd = -1;
|
||||||
|
|
||||||
|
ci->card_prof = &osim_uicc_sim_cic_profile;
|
||||||
|
|
||||||
|
if (!remote_udp_host) {
|
||||||
|
rc = libusb_init(NULL);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "libusb initialization failed\n");
|
||||||
|
goto do_exit;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
transp->udp_fd = osmo_sock_init(AF_INET, SOCK_DGRAM, IPPROTO_UDP,
|
||||||
|
remote_udp_host, remote_udp_port+if_num,
|
||||||
|
OSMO_SOCK_F_CONNECT);
|
||||||
|
if (transp->udp_fd < 0) {
|
||||||
|
fprintf(stderr, "error binding UDP port\n");
|
||||||
|
goto do_exit;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = osmo_st2_gsmtap_init(gsmtap_host);
|
||||||
|
if (rc < 0) {
|
||||||
|
perror("unable to open GSMTAP");
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
reader = osim_reader_open(OSIM_READER_DRV_PCSC, 0, "", NULL);
|
||||||
|
if (!reader) {
|
||||||
|
perror("unable to open PC/SC reader");
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
card = osim_card_open(reader, OSIM_PROTO_T0);
|
||||||
|
if (!card) {
|
||||||
|
perror("unable to open SIM card");
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
ci->chan = llist_entry(card->channels.next, struct osim_chan_hdl, list);
|
||||||
|
if (!ci->chan) {
|
||||||
|
perror("SIM card has no channel?!?");
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
signal(SIGINT, &signal_handler);
|
||||||
|
|
||||||
|
do {
|
||||||
|
if (transp->udp_fd < 0) {
|
||||||
|
struct usb_interface_match _ifm, *ifm = &_ifm;
|
||||||
|
ifm->vendor = vendor_id;
|
||||||
|
ifm->product = product_id;
|
||||||
|
ifm->configuration = config_id;
|
||||||
|
ifm->interface = if_num;
|
||||||
|
ifm->altsetting = altsetting;
|
||||||
|
ifm->addr = addr;
|
||||||
|
if (path)
|
||||||
|
osmo_strlcpy(ifm->path, path, sizeof(ifm->path));
|
||||||
|
transp->usb_devh = usb_open_claim_interface(NULL, ifm);
|
||||||
|
if (!transp->usb_devh) {
|
||||||
|
fprintf(stderr, "can't open USB device\n");
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = libusb_claim_interface(transp->usb_devh, if_num);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "can't claim interface %d; rc=%d\n", if_num, rc);
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
rc = get_usb_ep_addrs(transp->usb_devh, if_num, &transp->usb_ep.out,
|
||||||
|
&transp->usb_ep.in, &transp->usb_ep.irq_in);
|
||||||
|
if (rc < 0) {
|
||||||
|
fprintf(stderr, "can't obtain EP addrs; rc=%d\n", rc);
|
||||||
|
goto close_exit;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* simulate card-insert to modem (owhw, not qmod) */
|
||||||
|
osmo_st2_cardem_request_card_insert(ci, true);
|
||||||
|
|
||||||
|
/* select remote (forwarded) SIM */
|
||||||
|
osmo_st2_modem_sim_select_remote(ci->slot);
|
||||||
|
|
||||||
|
if (!skip_atr) {
|
||||||
|
/* set the ATR */
|
||||||
|
uint8_t real_atr[] = { 0x3B, 0x9F, 0x96, 0x80, 0x1F, 0xC7, 0x80, 0x31,
|
||||||
|
0xA0, 0x73, 0xBE, 0x21, 0x13, 0x67, 0x43, 0x20,
|
||||||
|
0x07, 0x18, 0x00, 0x00, 0x01, 0xA5 };
|
||||||
|
atr_update_csum(real_atr, sizeof(real_atr));
|
||||||
|
osmo_st2_cardem_request_set_atr(ci, real_atr, sizeof(real_atr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* select remote (forwarded) SIM */
|
||||||
|
osmo_st2_modem_reset_pulse(ci->slot, 300);
|
||||||
|
|
||||||
|
run_mainloop(ci);
|
||||||
|
ret = 0;
|
||||||
|
|
||||||
|
if (transp->udp_fd < 0)
|
||||||
|
libusb_release_interface(transp->usb_devh, 0);
|
||||||
|
close_exit:
|
||||||
|
if (transp->usb_devh)
|
||||||
|
libusb_close(transp->usb_devh);
|
||||||
|
if (keep_running)
|
||||||
|
sleep(1);
|
||||||
|
} while (keep_running);
|
||||||
|
|
||||||
|
if (transp->udp_fd < 0)
|
||||||
|
libusb_exit(NULL);
|
||||||
|
do_exit:
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
@@ -37,14 +37,12 @@
|
|||||||
|
|
||||||
#include <libusb.h>
|
#include <libusb.h>
|
||||||
|
|
||||||
#include "libusb_util.h"
|
#include <osmocom/simtrace2/libusb_util.h>
|
||||||
#include "simtrace.h"
|
#include <osmocom/simtrace2/simtrace_usb.h>
|
||||||
#include "simtrace_usb.h"
|
#include <osmocom/simtrace2/simtrace_prot.h>
|
||||||
#include "simtrace_prot.h"
|
|
||||||
#include "simtrace2-discovery.h"
|
#include <osmocom/simtrace2/gsmtap.h>
|
||||||
|
|
||||||
#include <osmocom/core/gsmtap.h>
|
|
||||||
#include <osmocom/core/gsmtap_util.h>
|
|
||||||
#include <osmocom/core/utils.h>
|
#include <osmocom/core/utils.h>
|
||||||
#include <osmocom/core/socket.h>
|
#include <osmocom/core/socket.h>
|
||||||
#include <osmocom/core/msgb.h>
|
#include <osmocom/core/msgb.h>
|
||||||
@@ -62,39 +60,6 @@ struct st_transport {
|
|||||||
} usb_ep;
|
} usb_ep;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* global GSMTAP instance */
|
|
||||||
static struct gsmtap_inst *g_gti;
|
|
||||||
|
|
||||||
static int gsmtap_send_sim(uint8_t sub_type, const uint8_t *data, unsigned int len)
|
|
||||||
{
|
|
||||||
struct gsmtap_hdr *gh;
|
|
||||||
unsigned int gross_len = len + sizeof(*gh);
|
|
||||||
uint8_t *buf = malloc(gross_len);
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
if (!buf)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
memset(buf, 0, sizeof(*gh));
|
|
||||||
gh = (struct gsmtap_hdr *) buf;
|
|
||||||
gh->version = GSMTAP_VERSION;
|
|
||||||
gh->hdr_len = sizeof(*gh)/4;
|
|
||||||
gh->type = GSMTAP_TYPE_SIM;
|
|
||||||
gh->sub_type = sub_type;
|
|
||||||
|
|
||||||
memcpy(buf + sizeof(*gh), data, len);
|
|
||||||
|
|
||||||
rc = write(gsmtap_inst_fd(g_gti), buf, gross_len);
|
|
||||||
if (rc < 0) {
|
|
||||||
perror("write gsmtap");
|
|
||||||
free(buf);
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
free(buf);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const struct value_string change_flags[] = {
|
const struct value_string change_flags[] = {
|
||||||
{
|
{
|
||||||
.value = SNIFF_CHANGE_FLAG_CARD_INSERT,
|
.value = SNIFF_CHANGE_FLAG_CARD_INSERT,
|
||||||
@@ -239,11 +204,11 @@ static int process_data(enum simtrace_msg_type_sniff type, const uint8_t *buf, i
|
|||||||
/* Send message as GSNTAP */
|
/* Send message as GSNTAP */
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case SIMTRACE_MSGT_SNIFF_ATR:
|
case SIMTRACE_MSGT_SNIFF_ATR:
|
||||||
gsmtap_send_sim(GSMTAP_SIM_ATR, data->data, data->length);
|
osmo_st2_gsmtap_send_apdu(GSMTAP_SIM_ATR, data->data, data->length);
|
||||||
break;
|
break;
|
||||||
case SIMTRACE_MSGT_SNIFF_TPDU:
|
case SIMTRACE_MSGT_SNIFF_TPDU:
|
||||||
/* TPDU is now considered as APDU since SIMtrace sends complete TPDU */
|
/* TPDU is now considered as APDU since SIMtrace sends complete TPDU */
|
||||||
gsmtap_send_sim(GSMTAP_SIM_APDU, data->data, data->length);
|
osmo_st2_gsmtap_send_apdu(GSMTAP_SIM_APDU, data->data, data->length);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
@@ -529,12 +494,11 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
printf("(%s)\n", strbuf);
|
printf("(%s)\n", strbuf);
|
||||||
|
|
||||||
g_gti = gsmtap_source_init(gsmtap_host, GSMTAP_UDP_PORT, 0);
|
rc = osmo_st2_gsmtap_init(gsmtap_host);
|
||||||
if (!g_gti) {
|
if (rc < 0) {
|
||||||
perror("unable to open GSMTAP");
|
perror("unable to open GSMTAP");
|
||||||
goto close_exit;
|
goto close_exit;
|
||||||
}
|
}
|
||||||
gsmtap_source_add_sink(g_gti);
|
|
||||||
|
|
||||||
signal(SIGINT, &signal_handler);
|
signal(SIGINT, &signal_handler);
|
||||||
|
|
||||||
@@ -23,8 +23,8 @@
|
|||||||
|
|
||||||
#include <osmocom/core/utils.h>
|
#include <osmocom/core/utils.h>
|
||||||
|
|
||||||
#include "libusb_util.h"
|
#include <osmocom/simtrace2/libusb_util.h>
|
||||||
#include "simtrace_usb.h"
|
#include <osmocom/simtrace2/simtrace_usb.h>
|
||||||
|
|
||||||
static const struct dev_id compatible_dev_ids[] = {
|
static const struct dev_id compatible_dev_ids[] = {
|
||||||
{ USB_VENDOR_OPENMOKO, USB_PRODUCT_OWHW_SAM3 },
|
{ USB_VENDOR_OPENMOKO, USB_PRODUCT_OWHW_SAM3 },
|
||||||
@@ -85,4 +85,5 @@ int main(int argc, char **argv)
|
|||||||
{
|
{
|
||||||
libusb_init(NULL);
|
libusb_init(NULL);
|
||||||
find_devices();
|
find_devices();
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -35,10 +35,9 @@
|
|||||||
|
|
||||||
#include <libusb.h>
|
#include <libusb.h>
|
||||||
|
|
||||||
#include "simtrace.h"
|
#include <osmocom/simtrace2/simtrace_usb.h>
|
||||||
#include "simtrace_prot.h"
|
#include <osmocom/simtrace2/simtrace_prot.h>
|
||||||
#include "apdu_dispatch.h"
|
#include <osmocom/simtrace2/libusb_util.h>
|
||||||
#include "simtrace2-discovery.h"
|
|
||||||
|
|
||||||
#include <osmocom/core/utils.h>
|
#include <osmocom/core/utils.h>
|
||||||
#include <osmocom/core/socket.h>
|
#include <osmocom/core/socket.h>
|
||||||
@@ -243,7 +242,7 @@ int main(int argc, char **argv)
|
|||||||
|
|
||||||
libusb_set_pollfd_notifiers(NULL, &libusb_fd_added_cb, &libusb_fd_removed_cb, NULL);
|
libusb_set_pollfd_notifiers(NULL, &libusb_fd_added_cb, &libusb_fd_removed_cb, NULL);
|
||||||
|
|
||||||
g_devh = libusb_open_device_with_vid_pid(NULL, SIMTRACE_USB_VENDOR, SIMTRACE_USB_PRODUCT);
|
g_devh = libusb_open_device_with_vid_pid(NULL, USB_VENDOR_OPENMOKO, USB_PRODUCT_OWHW_SAM3);
|
||||||
if (!g_devh) {
|
if (!g_devh) {
|
||||||
fprintf(stderr, "can't open USB device\n");
|
fprintf(stderr, "can't open USB device\n");
|
||||||
goto close_exit;
|
goto close_exit;
|
||||||
Reference in New Issue
Block a user