diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 00000000..341ccc8f --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,162 @@ +name: 'build direwolf' + +on: + # permit to manually trigger the CI + workflow_dispatch: + inputs: + cmake_flags: + description: 'Custom CMAKE flags' + required: false + push: + paths-ignore: + - '.github/**' + pull_request: + paths-ignore: + - '.github/**' + +jobs: + build: + name: ${{ matrix.config.name }} + runs-on: ${{ matrix.config.os }} + strategy: + fail-fast: false + matrix: + config: + - { + name: 'Windows Latest MinGW 64bit', + os: windows-latest, + cc: 'x86_64-w64-mingw32-gcc', + cxx: 'x86_64-w64-mingw32-g++', + ar: 'x86_64-w64-mingw32-ar', + windres: 'x86_64-w64-mingw32-windres', + arch: 'x86_64', + build_type: 'Release', + cmake_extra_flags: '-G "MinGW Makefiles"' + } + - { + name: 'Windows 2019 MinGW 32bit', + os: windows-2019, + cc: 'i686-w64-mingw32-gcc', + cxx: 'i686-w64-mingw32-g++', + ar: 'i686-w64-mingw32-ar', + windres: 'i686-w64-mingw32-windres', + arch: 'i686', + build_type: 'Release', + cmake_extra_flags: '-G "MinGW Makefiles"' + } + - { + name: 'macOS latest', + os: macos-latest, + cc: 'clang', + cxx: 'clang++', + arch: 'x86_64', + build_type: 'Release', + cmake_extra_flags: '' + } + - { + name: 'Ubuntu latest Debug', + os: ubuntu-latest, + cc: 'gcc', + cxx: 'g++', + arch: 'x86_64', + build_type: 'Debug', + cmake_extra_flags: '' + } + - { + name: 'Ubuntu 22.04', + os: ubuntu-22.04, + cc: 'gcc', + cxx: 'g++', + arch: 'x86_64', + build_type: 'Release', + cmake_extra_flags: '' + } + - { + name: 'Ubuntu 20.04', + os: ubuntu-20.04, + cc: 'gcc', + cxx: 'g++', + arch: 'x86_64', + build_type: 'Release', + cmake_extra_flags: '' + } + + steps: + - name: checkout + uses: actions/checkout@v2 + with: + fetch-depth: 8 + - name: dependency + shell: bash + run: | + # this is not perfect but enought for now + if [ "$RUNNER_OS" == "Linux" ]; then + sudo apt-get update + sudo apt-get install libasound2-dev libudev-dev libhamlib-dev gpsd + elif [ "$RUNNER_OS" == "macOS" ]; then + # just to simplify I use homebrew but + # we can use macports (latest direwolf is already available as port) + brew install portaudio hamlib gpsd + elif [ "$RUNNER_OS" == "Windows" ]; then + # add the folder to PATH + echo "C:\msys64\mingw32\bin" >> $GITHUB_PATH + fi + - name: create build environment + run: | + cmake -E make_directory ${{github.workspace}}/build + - name: configure + shell: bash + working-directory: ${{github.workspace}}/build + run: | + if [ "$RUNNER_OS" == "Windows" ]; then + export CC=${{ matrix.config.cc }} + export CXX=${{ matrix.config.cxx }} + export AR=${{ matrix.config.ar }} + export WINDRES=${{ matrix.config.windres }} + fi + cmake $GITHUB_WORKSPACE \ + -DCMAKE_BUILD_TYPE=${{ matrix.config.build_type }} \ + -DCMAKE_C_COMPILER=${{ matrix.config.cc }} \ + -DCMAKE_CXX_COMPILER=${{ matrix.config.cxx }} \ + -DCMAKE_CXX_FLAGS="-Werror" -DUNITTEST=1 \ + ${{ matrix.config.cmake_extra_flags }} \ + ${{ github.event.inputs.cmake_flags }} + - name: build + shell: bash + working-directory: ${{github.workspace}}/build + run: | + if [ "$RUNNER_OS" == "Windows" ]; then + export CC=${{ matrix.config.cc }} + export CXX=${{ matrix.config.cxx }} + export AR=${{ matrix.config.ar }} + export WINDRES=${{ matrix.config.windres }} + fi + cmake --build . --config ${{ matrix.config.build_type }} \ + ${{ github.event.inputs.cmake_flags }} + - name: test + continue-on-error: true + shell: bash + working-directory: ${{github.workspace}}/build + run: | + ctest -C ${{ matrix.config.build_type }} \ + --parallel 2 --output-on-failure \ + ${{ github.event.inputs.cmake_flags }} + - name: package + shell: bash + working-directory: ${{github.workspace}}/build + run: | + if [ "$RUNNER_OS" == "Windows" ] || [ "$RUNNER_OS" == "macOS" ]; then + make package + fi + - name: archive binary + uses: actions/upload-artifact@v4 + with: + name: direwolf_${{ matrix.config.os }}_${{ matrix.config.arch }}_${{ github.sha }} + path: | + ${{github.workspace}}/build/direwolf-*.zip + ${{github.workspace}}/build/direwolf.conf + ${{github.workspace}}/build/src/* + ${{github.workspace}}/build/CMakeCache.txt + !${{github.workspace}}/build/src/cmake_install.cmake + !${{github.workspace}}/build/src/CMakeFiles + !${{github.workspace}}/build/src/Makefile diff --git a/.github/workflows/codeql-analysis-python.yml b/.github/workflows/codeql-analysis-python.yml new file mode 100644 index 00000000..a47a8f8d --- /dev/null +++ b/.github/workflows/codeql-analysis-python.yml @@ -0,0 +1,64 @@ +# For most projects, this workflow file will not need changing; you simply need +# to commit it to your repository. +# +# You may wish to alter this file to override the set of languages analyzed, +# or to provide custom queries or build logic. +# +# ******** NOTE ******** +# We have attempted to detect the languages in your repository. Please check +# the `language` matrix defined below to confirm you have the correct set of +# supported CodeQL languages. +# +name: "CodeQL - Python" + +on: + push: + branches: [ dev ] + pull_request: + # The branches below must be a subset of the branches above + branches: [ dev ] + schedule: + - cron: '25 8 * * 4' + +jobs: + analyze: + name: Analyze + runs-on: ubuntu-latest + permissions: + actions: read + contents: read + security-events: write + + strategy: + fail-fast: false + matrix: + language: [ 'python' ] + # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] + # Learn more about CodeQL language support at https://git.io/codeql-language-support + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + + # Initializes the CodeQL tools for scanning. + - name: Initialize CodeQL + uses: github/codeql-action/init@v3 + with: + languages: ${{ matrix.language }} + setup-python-dependencies: true + # If you wish to specify custom queries, you can do so here or in a config file. + # By default, queries listed here will override any specified in a config file. + # Prefix the list here with "+" to use these queries and those in the config file. + # queries: ./path/to/local/query, your-org/your-repo/queries@main + + # Autobuild attempts to build any compiled languages (C/C++, C#, or Java). + # If this step fails, then you should remove it and run the build manually (see below) + - name: Autobuild + uses: github/codeql-action/autobuild@v3 + + # ℹī¸ Command-line programs to run using the OS shell. + # 📚 https://git.io/JvXDl + + + - name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v3 diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml new file mode 100644 index 00000000..a86300f3 --- /dev/null +++ b/.github/workflows/codeql-analysis.yml @@ -0,0 +1,74 @@ +# For most projects, this workflow file will not need changing; you simply need +# to commit it to your repository. +# +# You may wish to alter this file to override the set of languages analyzed, +# or to provide custom queries or build logic. +# +# ******** NOTE ******** +# We have attempted to detect the languages in your repository. Please check +# the `language` matrix defined below to confirm you have the correct set of +# supported CodeQL languages. +# +name: "CodeQL - CPP" + +on: + push: + branches: [ dev ] + pull_request: + # The branches below must be a subset of the branches above + branches: [ dev ] + schedule: + - cron: '25 8 * * 4' + +jobs: + analyze: + name: Analyze + runs-on: ubuntu-latest + permissions: + actions: read + contents: read + security-events: write + + strategy: + fail-fast: false + matrix: + language: [ 'cpp' ] + # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] + # Learn more about CodeQL language support at https://git.io/codeql-language-support + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + + # Initializes the CodeQL tools for scanning. + - name: Initialize CodeQL + uses: github/codeql-action/init@v3 + with: + languages: ${{ matrix.language }} + setup-python-dependencies: true + # If you wish to specify custom queries, you can do so here or in a config file. + # By default, queries listed here will override any specified in a config file. + # Prefix the list here with "+" to use these queries and those in the config file. + # queries: ./path/to/local/query, your-org/your-repo/queries@main + + # Autobuild attempts to build any compiled languages (C/C++, C#, or Java). + # If this step fails, then you should remove it and run the build manually (see below) + - name: Autobuild + uses: github/codeql-action/autobuild@v3 + + # ℹī¸ Command-line programs to run using the OS shell. + # 📚 https://git.io/JvXDl + + # ✏ī¸ If the Autobuild fails above, remove it and uncomment the following three lines + # and modify them (or add more) to build your code if your project + # uses a compiled language + + - run: | + mkdir build + cd build + cmake -DUNITTEST=1 .. + make + make test + + - name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v3 diff --git a/.gitignore b/.gitignore index 9f3c3763..659c845b 100644 --- a/.gitignore +++ b/.gitignore @@ -105,3 +105,9 @@ $RECYCLE.BIN/ # Windows shortcuts *.lnk +/use_this_sdk +*.dSYM + +# cmake +build/ +tmp/ \ No newline at end of file diff --git a/CHANGES.md b/CHANGES.md index 0102c2eb..4b78ca14 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -2,6 +2,186 @@ # Revision History # +## Version 1.7 -- October 2023 ## + + +### New Documentation: ### + +Additional documentation location to slow down growth of main repository. [https://github.com/wb2osz/direwolf-doc](https://github.com/wb2osz/direwolf-doc) . These are more oriented toward achieving a goal and understanding, as opposed to the User Guide which describes the functionality. + +- ***APRS Digipeaters*** + +- ***Internal Packet Routing*** + +- ***Radio Interface Guide*** + +- ***Successful IGate Operation*** + +- ***Understanding APRS Packets*** + + +### New Features: ### + + + +- New ICHANNEL configuration option to map a KISS client application channel to APRS-IS. Packets from APRS-IS will be presented to client applications as the specified channel. Packets sent, by client applications, to that channel will go to APRS-IS rather than a radio channel. Details in ***Internal-Packet-Routing.pdf***. + +- New variable speed option for gen_packets. For example, "-v 5,0.1" would generate packets from 5% too slow to 5% too fast with increments of 0.1. Some implementations might have imprecise timing. Use this to test how well TNCs tolerate sloppy timing. + +- Improved Layer 2 Protocol [(IL2P)](https://en.wikipedia.org/wiki/FX.25_Forward_Error_Correction). Compatible with Nino TNC for 1200 and 9600 bps. Use "-I 1" on command line to enable transmit for first channel. For more general case, add to config file (simplified version, see User Guide for more details): + + > After: "CHANNEL 1" (or other channel) + > + > Add: "IL2PTX 1" + +- Limited support for CM108/CM119 GPIO PTT on Windows. + +- Dire Wolf now advertises itself using DNS Service Discovery. This allows suitable APRS / Packet Radio applications to find a network KISS TNC without knowing the IP address or TCP port. Thanks to Hessu for providing this. Currently available only for Linux and Mac OSX. [Read all about it here.](https://github.com/hessu/aprs-specs/blob/master/TCP-KISS-DNS-SD.md) + +- The transmit calibration tone (-x) command line option now accepts a radio channel number and/or a single letter mode: a = alternate tones, m = mark tone, s = space tone, p = PTT only no sound. + +- The BEACON configuration now recognizes the SOURCE= option. This replaces the AX.25 source address rather than using the MYCALL value for the channel. This is useful for sending more than 5 analog telemetry channels. Use two, or more, source addresses with up to 5 analog channels each. + +- For more flexibility, the FX.25 transmit property can now be set individually by channel, rather than having a global setting for all channels. The -X on the command line applies only to channel 0. For other channels you need to add a new line to the configuration file. You can specify a specific number of parity bytes (16, 32, 64) or 1 to choose automatically based on packet size. + + > After: "CHANNEL 1" (or other channel) + > + > Add: "FX25TX 1" (or 16 or 32 or 64) + + + +### Bugs Fixed: ### + +- The t/m packet filter incorrectly included bulletins. It now allows only "messages" to specific stations. Use of t/m is discouraged. i/180 is the preferred filter for messages to users recently heard locally. + +- Packet filtering now skips over any third party header before classifying packet types. + +- Fixed build for Alpine Linux. + +### Notes: ### + +The Windows binary distribution now uses gcc (MinGW) version 11.3.0. +The Windows version is built for both 32 and 64 bit operating systems. +Use the 64 bit version if possible; it runs considerably faster. + +## Version 1.6 -- October 2020 ## + +### New Build Procedure: ### + + +- Rather than trying to keep a bunch of different platform specific Makefiles in sync, "cmake" is now used for greater portability and easier maintenance. This was contributed by Davide Gerhard. + +- README.md has a quick summary of the process. More details in the ***User Guide***. + + +### New Features: ### + + +- "-X" option enables FX.25 transmission. FX.25 reception is always enabled so you don't need to do anything special. "What is FX.25?" you might ask. It is forward error correction (FEC) added in a way that is completely compatible with an ordinary AX.25 frame. See new document ***AX25\_plus\_FEC\_equals\_FX25.pdf*** for details. + +- Receive AIS location data from ships. Enable by using "-B AIS" command line option or "MODEM AIS" in the configuration file. AIS NMEA sentences are encapsulated in APRS user-defined data with a "{DA" prefix. This uses 9600 bps so you need to use wide band audio, not what comes out of the speaker. There is also a "-A" option to generate APRS Object Reports. + +- Receive Emergency Alert System (EAS) Specific Area Message Encoding (SAME). Enable by using "-B EAS" command line option or "MODEM EAS" in the configuration file. EAS SAME messages are encapsulated in APRS user-defined data with a "{DE" prefix. This uses low speed AFSK so speaker output is fine. + +- "-t" option now accepts more values to accommodate inconsistent handling of text color control codes by different terminal emulators. The default, 1, should work with most modern terminal types. If the colors are not right, try "-t 9" to see the result of the different choices and pick the best one. If none of them look right, file a bug report and specify: operating system version (e.g. Raspbian Buster), terminal emulator type and version (e.g. LXTerminal 0.3.2). Include a screen capture. + + +- "-g" option to force G3RUH mode for lower speeds where a different modem type may be the default. + +- 2400 bps compatibility with MFJ-2400. See ***2400-4800-PSK-for-APRS-Packet-Radio.pdf*** for details + +- "atest -h" will display the frame in hexadecimal for closer inspection. + +- Add support for Multi-GNSS NMEA sentences. + + + +### Bugs Fixed: ### + +- Proper counting of frames in transmit queue for AGW protocol 'Y' command. + + + +### New Documentation: ### + +- ***AX.25 + FEC = FX.25*** + +- ***AIS Reception*** + +- ***AX.25 Throughput: Why is 9600 bps Packet Radio only twice as fast as 1200?*** + +- [***Ham Radio of Things (HRoT) - IoT over Ham Radio***](https://github.com/wb2osz/hrot) + +- [***EAS SAME to APRS Message Converter***](https://github.com/wb2osz/eas2aprs) + +- [***Dire Wolf PowerPoint Slide Show***](https://github.com/wb2osz/direwolf-presentation) + +### Notes: ### + +The Windows binary distribution now uses gcc (MinGW) version 7.4.0. +The Windows version is built for both 32 and 64 bit operating systems. +Use the 64 bit version if possible; it runs considerably faster. + + + +## Version 1.5 -- September 2018 ## + + +### New Features: ### + +- PTT using GPIO pin of CM108/CM119 (e.g. DMK URI, RB-USB RIM), Linux only. + +- More efficient error recovery for AX.25 connected mode. Better generation and processing of REJ and SREJ to reduce unnecessary duplicate "**I**" frames. + +- New configuration option, "**V20**", for listing stations known to not understand AX.25 v2.2. This will speed up connection by going right to SABM and not trying SABME first and failing. + +- New "**NOXID**" configuration file option to avoid sending XID command to listed station(s). If other end is a partial v2.2 implementation, which recognizes SABME, but not XID, we would waste a lot of time resending XID many times before giving up. This is less drastic than the "**V20**" option which doesn't even attempt to use v2.2 with listed station(s). + +- New application "**kissutil**" for troubleshooting a KISS TNC or interfacing to an application via files. + +- KISS "Set Hardware" command to report transmit queue length. + +- TCP KISS can now handle multiple concurrent applications. + +- Linux can use serial port for KISS in addition to a pseudo terminal. + +- decode_aprs utility can now accept KISS frames and AX.25 frames as series of two digit hexadecimal numbers. + +- Full Duplex operation. (Put "FULLDUP ON" in channel section of configuration file.) + +- Time slots for beaconing. + +- Allow single log file with fixed name rather than starting a new one each day. + + + +### Bugs Fixed: ### + +- Possible crash when CDIGIPEAT did not include the optional alias. + +- PACLEN configuration item no longer restricts length of received frames. + +- Strange failures when trying to use multiple KISS client applications over TCP. Only Linux was affected. + +- Under certain conditions, outgoing connected mode data would get stuck in a queue and not be transmitted. This could happen if client application sends a burst of data larger than the "window" size (MAXFRAME or EMAXFRAME option). + + +- Little typographical / spelling errors in messages. + + +### Documentation: ### + + +- New document ***Bluetooth-KISS-TNC.pdf*** explaining how to use KISS over Bluetooth. + +- Updates describing cheap SDR frequency inaccuracy and how to compensate for it. + +### Notes: ### + +Windows binary distribution now uses gcc (MinGW) version 6.3.0. + +---------- + ## Version 1.4 -- April 2017 ## diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 00000000..182a9b4d --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,414 @@ +cmake_minimum_required(VERSION 3.5.0) + +project(direwolf) + +# configure version +set(direwolf_VERSION_MAJOR "1") +set(direwolf_VERSION_MINOR "7") +set(direwolf_VERSION_PATCH "0") +set(direwolf_VERSION_SUFFIX "Development") + +# options +# See Issue 297. +option(FORCE_SSE "Compile with SSE instruction only" ON) +option(FORCE_SSSE3 "Compile with SSSE3 instruction only" OFF) +option(FORCE_SSE41 "Compile with SSE4.1 instruction only" OFF) +option(OPTIONAL_TEST "Compile optional test (might be broken)" OFF) +# UNITTEST option must be after CMAKE_BUILT_TYPE + +# where cmake find custom modules +list(APPEND CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake/modules) + +# fix c standard used on the project +set(CMAKE_C_STANDARD 99) + +# Set additional project information +set(COMPANY "wb2osz") +add_definitions("-DCOMPANY=\"${COMPANY}\"") +set(APPLICATION_NAME "Dire Wolf") +add_definitions("-DAPPLICATION_NAME=\"${APPLICATION_NAME}\"") +set(APPLICATION_MAINTAINER="John Langner, WB2OSZ") +set(COPYRIGHT "Copyright (c) 2019 John Langner, WB2OSZ. All rights reserved.") +add_definitions("-DCOPYRIGHT=\"${COPYRIGHT}\"") +set(IDENTIFIER "com.${COMPANY}.${APPLICATION_NAME}") +add_definitions("-DIDENTIFIER=\"${IDENTIFIER}\"") +# raspberry as only lxterminal not xterm +if(NOT (WIN32 OR CYGWIN)) + find_program(BINARY_TERMINAL_BIN lxterminal) + if(BINARY_TERMINAL_BIN) + set(APPLICATION_DESKTOP_EXEC "${BINARY_TERMINAL_BIN} -e ${CMAKE_PROJECT_NAME}") + else() + set(APPLICATION_DESKTOP_EXEC "xterm -e ${CMAKE_PROJECT_NAME}") + endif() +endif() + +find_package(Git) +if(GIT_FOUND AND EXISTS "${CMAKE_SOURCE_DIR}/.git/") + # we can also use `git describe --tags` + execute_process(COMMAND "${GIT_EXECUTABLE}" rev-parse --short HEAD + WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}" + RESULT_VARIABLE res + OUTPUT_VARIABLE out + ERROR_QUIET + OUTPUT_STRIP_TRAILING_WHITESPACE) + if(NOT res) + string(REGEX REPLACE "^v([0-9]+)\.([0-9]+)\.([0-9]+)-" "" git_commit ${out}) + set(direwolf_VERSION_SUFFIX "-${git_commit}") + set(direwolf_VERSION_COMMIT "${git_commit}") + endif() +endif() + +# set variables +set(direwolf_VERSION "${direwolf_VERSION_MAJOR}.${direwolf_VERSION_MINOR}.${direwolf_VERSION_PATCH}${direwolf_VERSION_SUFFIX}") +message(STATUS "${APPLICATION_NAME} Version: ${direwolf_VERSION}") +add_definitions("-DIREWOLF_VERSION=\"${direwolf_VERSION}\"") +add_definitions("-DMAJOR_VERSION=${direwolf_VERSION_MAJOR}") +add_definitions("-DMINOR_VERSION=${direwolf_VERSION_MINOR}") +if(direwolf_VERSION_COMMIT) + add_definitions("-DEXTRA_VERSION=${direwolf_VERSION_COMMIT}") +endif() + +set(CUSTOM_SRC_DIR "${CMAKE_SOURCE_DIR}/src") +set(CUSTOM_EXTERNAL_DIR "${CMAKE_SOURCE_DIR}/external") +set(CUSTOM_MISC_DIR "${CUSTOM_EXTERNAL_DIR}/misc") +set(CUSTOM_REGEX_DIR "${CUSTOM_EXTERNAL_DIR}/regex") +set(CUSTOM_HIDAPI_DIR "${CUSTOM_EXTERNAL_DIR}/hidapi") +set(CUSTOM_GEOTRANZ_DIR "${CUSTOM_EXTERNAL_DIR}/geotranz") +set(CUSTOM_DATA_DIR "${CMAKE_SOURCE_DIR}/data") +set(CUSTOM_SCRIPTS_DIR "${CMAKE_SOURCE_DIR}/scripts") +set(CUSTOM_TELEMETRY_DIR "${CUSTOM_SCRIPTS_DIR}/telemetry-toolkit") +set(CUSTOM_CONF_DIR "${CMAKE_SOURCE_DIR}/conf") +set(CUSTOM_DOC_DIR "${CMAKE_SOURCE_DIR}/doc") +set(CUSTOM_MAN_DIR "${CMAKE_SOURCE_DIR}/man") +set(CUSTOM_TEST_DIR "${CMAKE_SOURCE_DIR}/test") +set(CUSTOM_TEST_SCRIPTS_DIR "${CUSTOM_TEST_DIR}/scripts") +set(CUSTOM_SHELL_SHABANG "#!/bin/sh -e") + +# cpack variables +set(CPACK_GENERATOR "ZIP") +set(CPACK_STRIP_FILES true) +set(CPACK_PACKAGE_NAME "${CMAKE_PROJECT_NAME}") +# This has architecture of the build machine, not the target platform. +# e.g. Comes out as x86_64 when building for i686 target platform. +#set(CPACK_PACKAGE_FILE_NAME "${CMAKE_PROJECT_NAME}-${direwolf_VERSION}_${CMAKE_SYSTEM_PROCESSOR}") +# We don't know the target yet so this is set after FindCPUflags. +set(CPACK_PACKAGE_CONTACT "https://github.com/wb2osz/direwolf") +SET(CPACK_PACKAGE_DESCRIPTION_SUMMARY "Dire Wolf is an AX.25 soundcard TNC, digipeater, APRS IGate, GPS tracker, and APRStt gateway") +set(CPACK_PACKAGE_DESCRIPTION_FILE "${CMAKE_SOURCE_DIR}/README.md") +set(CPACK_RESOURCE_FILE_README "${CMAKE_SOURCE_DIR}/README.md") +set(CPACK_RESOURCE_FILE_LICENSE "${CMAKE_SOURCE_DIR}/LICENSE") +set(CPACK_SOURCE_IGNORE_FILES "${PROJECT_BINARY_DIR};/.git/;.gitignore;menu.yml;.travis.yml;.appveyor.yml;default.nix;.envrc;TODOs.org;/.scripts/") +SET(CPACK_PACKAGE_VERSION "${direwolf_VERSION}") +SET(CPACK_PACKAGE_VERSION_MAJOR "${direwolf_VERSION_MAJOR}") +SET(CPACK_PACKAGE_VERSION_MINOR "${direwolf_VERSION_MINOR}") +SET(CPACK_PACKAGE_VERSION_PATCH "${direwolf_VERSION_PATCH}") +SET(CPACK_DEBIAN_PACKAGE_DEPENDS "libasound2,libgps23") + +# if we don't set build_type +if(NOT DEFINED CMAKE_BUILD_TYPE OR "${CMAKE_BUILD_TYPE}" STREQUAL "") + set(CMAKE_BUILD_TYPE Release CACHE STRING "Build type" FORCE) +endif() +message(STATUS "Build type set to: ${CMAKE_BUILD_TYPE}") +message("CMake system: ${CMAKE_SYSTEM_NAME}") + +# Unittest should be on for dev builds and off for releases. +if(CMAKE_BUILD_TYPE MATCHES "Release") + option(UNITTEST "Build unittest binaries." OFF) +else() + option(UNITTEST "Build unittest binaries." ON) +endif() + +# set compiler +include(FindCompiler) + +# find cpu flags (and set compiler) +include(FindCPUflags) + +if(${ARCHITECTURE} MATCHES "x86") + set(CPACK_PACKAGE_FILE_NAME "${CMAKE_PROJECT_NAME}-${direwolf_VERSION}_i686") +else() + set(CPACK_PACKAGE_FILE_NAME "${CMAKE_PROJECT_NAME}-${direwolf_VERSION}_${ARCHITECTURE}") +endif() + +# auto include current directory +set(CMAKE_INCLUDE_CURRENT_DIR ON) + +# set OS dependent variables +if(${CMAKE_SYSTEM_NAME} STREQUAL "Linux") + set(LINUX TRUE) + + configure_file("${CMAKE_SOURCE_DIR}/cmake/cpack/${CMAKE_PROJECT_NAME}.desktop.in" + "${CMAKE_BINARY_DIR}/${CMAKE_PROJECT_NAME}.desktop" @ONLY) + +elseif(${CMAKE_SYSTEM_NAME} STREQUAL "FreeBSD") + set(FREEBSD TRUE) + configure_file("${CMAKE_SOURCE_DIR}/cmake/cpack/${CMAKE_PROJECT_NAME}.desktop.in" + "${CMAKE_BINARY_DIR}/${CMAKE_PROJECT_NAME}.desktop" @ONLY) + +elseif(${CMAKE_SYSTEM_NAME} STREQUAL "OpenBSD") + set(OPENBSD TRUE) + set(HAVE_SNDIO TRUE) + +elseif(APPLE) + if("${CMAKE_OSX_DEPLOYMENT_TARGET}" STREQUAL "") + message(STATUS "Build for macOS target: local version") + else() + message(STATUS "Build for macOS target: ${CMAKE_OSX_DEPLOYMENT_TARGET}") + endif() + + # prepend path to find_*() + set(CMAKE_FIND_ROOT_PATH "/opt/local") + + set(CMAKE_MACOSX_RPATH ON) + message(STATUS "RPATH support: ${CMAKE_MACOSX_RPATH}") + + # just blindly enable dns-sd + set(USE_MACOS_DNSSD ON) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_MACOS_DNSSD") + +elseif (WIN32) + if(C_MSVC) + if (NOT VS2015 AND NOT VS2017 AND NOT VS2019) + message(FATAL_ERROR "You must use Microsoft Visual Studio 2015, 2017 or 2019 as compiler") + else() + # compile with full multicore + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /MP") + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /MP") + set(CUSTOM_SHELL_BIN "") + endif() + endif() +endif() + +if (C_CLANG OR C_GCC) + # _BSD_SOURCE is deprecated we need to use _DEFAULT_SOURCE. + # + # That works find for more modern compilers but we have issues with: + # Centos 7, gcc 4.8.5, glibc 2.17 + # Centos 6, gcc 4.4.7, glibc 2.12 + # + # CentOS 6 & 7: Without -D_BSD_SOURCE, we get Warning: Implicit declaration of + # functions alloca, cfmakeraw, scandir, setlinebuf, strcasecmp, strncasecmp, and strsep. + # When a function (like strsep) returns a pointer, the compiler instead assumes a 32 bit + # int and sign extends it out to be a 64 bit pointer. Use the pointer and Kaboom! + # + # CentOS 6: We have additional problem. Without -D_POSIX_C_SOURCE=199309L, we get + # implicit declaration of function clock_gettime and the linker can't find it. + # + # It turns out that -D_GNU_SOURCE can be used instead of both of those. For more information, + # see https://www.gnu.org/software/libc/manual/html_node/Feature-Test-Macros.html + # + # Why was this not an issue before? If gcc is used without the -std=c99 option, + # it is perfectly happy with clock_gettime, strsep, etc. but with the c99 option, it no longer + # recognizes a bunch of commonly used functions. Using _GNU_SOURCE, rather than _DEFAULT_SOURCE + # solves the problem for CentOS 6 & 7. This also makes -D_XOPEN_SOURCE= unnecessary. + # I hope it doesn't break with newer versions of glibc. + # + # I also took out -Wextra because it spews out so much noise a serious problem was not noticed. + # It might go back in someday when I have more patience to clean up all the warnings. + # + + # TODO: + # Try error checking -fsanitize=bounds-strict -fsanitize=leak + # Requires libubsan and liblsan, respectively. + + ###set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -Wextra -Wvla -ffast-math -ftree-vectorize -D_XOPEN_SOURCE=600 -D_DEFAULT_SOURCE ${EXTRA_FLAGS}") + if(FREEBSD) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -Wextra -Wvla -ffast-math -ftree-vectorize -D_DEFAULT_SOURCE ${EXTRA_FLAGS}") + else() + #set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -Wvla -ffast-math -ftree-vectorize -D_GNU_SOURCE -fsanitize=bounds-strict ${EXTRA_FLAGS}") + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -Wvla -ffast-math -ftree-vectorize -D_GNU_SOURCE ${EXTRA_FLAGS}") + endif() + # + # + # -lm is needed for functions in math.h + if (LINUX) + # We have another problem with CentOS 6. clock_gettime() is in librt so we need -lrt. + # The clock_* functions were moved into gnu libc for version 2.17. + # https://sourceware.org/ml/libc-announce/2012/msg00001.html + # If using gnu libc 2.17, or later, the -lrt is no longer needed but doesn't hurt. + # I'm making this conditional on LINUX because it is not needed for BSD and MacOSX. + link_libraries("-lrt -lm") + else() + link_libraries("-lm") + endif() +elseif (C_MSVC) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -W3 -MP ${EXTRA_FLAGS}") +endif() + +if (C_CLANG) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -ferror-limit=1") +elseif (C_GCC) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fmax-errors=1") +endif() + +# set installation directories +if (WIN32 OR CYGWIN) + set(INSTALL_BIN_DIR ".") + set(INSTALL_DOC_DIR "doc") + set(INSTALL_CONF_DIR ".") + set(INSTALL_SCRIPTS_DIR "scripts") + set(INSTALL_MAN_DIR "man") + set(INSTALL_DATA_DIR "data") +else() + set(INSTALL_BIN_DIR "bin") + set(INSTALL_DOC_DIR "share/doc/${CMAKE_PROJECT_NAME}") + set(INSTALL_CONF_DIR "${INSTALL_DOC_DIR}/conf") + set(INSTALL_SCRIPTS_DIR "${INSTALL_DOC_DIR}/scripts") + if(FREEBSD) + set(INSTALL_MAN_DIR "man/man1") + else() + set(INSTALL_MAN_DIR "share/man/man1") + endif() + set(INSTALL_DATA_DIR "share/${PROJECT_NAME}") +endif(WIN32 OR CYGWIN) + +# requirements + +include(CheckSymbolExists) + +# Some platforms provide their own strlcpy & strlcat. (BSD, MacOSX) +# Others don't so we provide our own. (Windows, most, but not all Linux) +# Here we detect whether these are provided by the OS and set a symbol +# so that: +# (1) libgps does not supply its own version. +# (2) we know whether we need to supply our own copy. +# +# This was all working fine until these were added to the gnu c library 2.38. +# References: +# - https://www.gnu.org/software/libc/sources.html +# - https://sourceware.org/git/?p=glibc.git;a=blob_plain;f=NEWS;hb=HEAD +# +# This test is not detecting them for glibc 2.38 resulting in a conflict. +# Why? Are they declared in a different file or in some strange way? +# +# This is how they are declared in include/string.h: +# +# extern __typeof (strlcpy) __strlcpy; +# libc_hidden_proto (__strlcpy) +# extern __typeof (strlcat) __strlcat; +# libc_hidden_proto (__strlcat) +# +# Apparently cmake does not recognize this style. +# Keep this here for BSD type systems where it behaves as expected. +# We will need to add a hack in direwolf.h to define these if glibc version >= 2.38. + +check_symbol_exists(strlcpy string.h HAVE_STRLCPY) +if(HAVE_STRLCPY) + add_compile_options(-DHAVE_STRLCPY) +endif() +check_symbol_exists(strlcat string.h HAVE_STRLCAT) +if(HAVE_STRLCAT) + add_compile_options(-DHAVE_STRLCAT) +endif() + +set(THREADS_PREFER_PTHREAD_FLAG ON) +find_package(Threads REQUIRED) + +find_package(GPSD) +if(GPSD_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DENABLE_GPSD") +else() + set(GPSD_INCLUDE_DIRS "") + set(GPSD_LIBRARIES "") +endif() + +find_package(hamlib) +if(HAMLIB_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_HAMLIB") +else() + set(HAMLIB_INCLUDE_DIRS "") + set(HAMLIB_LIBRARIES "") +endif() + +if(LINUX) + find_package(ALSA REQUIRED) + if(ALSA_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_ALSA") + endif() + + find_package(udev) + if(UDEV_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_CM108") + endif() + + find_package(Avahi) + if(AVAHI_CLIENT_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_AVAHI_CLIENT") + endif() + +elseif (HAVE_SNDIO) + find_package(sndio REQUIRED) + if(SNDIO_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_SNDIO") + endif() + +elseif (NOT WIN32 AND NOT CYGWIN) + find_package(Portaudio REQUIRED) + if(PORTAUDIO_FOUND) + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_PORTAUDIO") + endif() + +else() + set(ALSA_INCLUDE_DIRS "") + set(ALSA_LIBRARIES "") + set(UDEV_INCLUDE_DIRS "") + set(UDEV_LIBRARIES "") + # Version 1.7 supports CM108/CM119 GPIO PTT for Windows. + set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -DUSE_CM108") + set(PORTAUDIO_INCLUDE_DIRS "") + set(PORTAUDIO_LIBRARIES "") + set(SNDIO_INCLUDE_DIRS "") + set(SNDIO_LIBRARIES "") +endif() + +# manage and fetch new data +add_subdirectory(data) + +# external libraries +add_subdirectory(${CUSTOM_GEOTRANZ_DIR}) +add_subdirectory(${CUSTOM_REGEX_DIR}) +add_subdirectory(${CUSTOM_HIDAPI_DIR}) +add_subdirectory(${CUSTOM_MISC_DIR}) + +# direwolf source code and utilities +add_subdirectory(src) + +# ctest +if(UNITTEST) + message(STATUS "Build unit test binaries") + include(CTest) + enable_testing() + add_subdirectory(test) +endif(UNITTEST) + +# manage scripts +add_subdirectory(scripts) + +# manage config +add_subdirectory(conf) + +# install basic docs +install(FILES ${CMAKE_SOURCE_DIR}/CHANGES.md DESTINATION ${INSTALL_DOC_DIR}) +install(FILES ${CMAKE_SOURCE_DIR}/LICENSE DESTINATION ${INSTALL_DOC_DIR}) +install(FILES ${CMAKE_SOURCE_DIR}/external/LICENSE DESTINATION ${INSTALL_DOC_DIR}/external) +add_subdirectory(doc) +add_subdirectory(man) + +# install desktop link +if (LINUX OR FREEBSD) + install(FILES ${CMAKE_BINARY_DIR}/${CMAKE_PROJECT_NAME}.desktop DESTINATION share/applications) + install(FILES ${CMAKE_SOURCE_DIR}/cmake/cpack/${CMAKE_PROJECT_NAME}_icon.png DESTINATION share/pixmaps) +endif() + +############ uninstall target ################ +configure_file( + "${CMAKE_CURRENT_SOURCE_DIR}/cmake/include/uninstall.cmake.in" + "${CMAKE_CURRENT_BINARY_DIR}/uninstall.cmake" + IMMEDIATE @ONLY) + +add_custom_target(uninstall + COMMAND ${CMAKE_COMMAND} -P + ${CMAKE_CURRENT_BINARY_DIR}/uninstall.cmake) + +############ packaging ################ +add_subdirectory(cmake/cpack) diff --git a/LICENSE-dire-wolf.txt b/LICENSE similarity index 100% rename from LICENSE-dire-wolf.txt rename to LICENSE diff --git a/Makefile b/Makefile index 0ae53948..7b0dcf76 100644 --- a/Makefile +++ b/Makefile @@ -1,16 +1,36 @@ -# Select proper Makefile for operating system. -# The Windows version is built with the help of Cygwin. -# In my case, I see CYGWIN_NT-6.1-WOW so we don't check for -# equal to some value. Your mileage my vary. - -win := $(shell uname | grep CYGWIN) -dar := $(shell uname | grep Darwin) - -ifneq ($(win),) - include Makefile.win -else ifeq ($(dar),Darwin) - include Makefile.macosx -else - include Makefile.linux -endif +all: + @echo "The build procedure has changed in version 1.6." + @echo "In general, it now looks like this:" + @echo " " + @echo "Download the source code:" + @echo " " + @echo " cd ~" + @echo " git clone https://www.github.com/wb2osz/direwolf" + @echo " cd direwolf" + @echo " " + @echo "Optional - Do this to get the latest development version" + @echo "rather than the latest stable release." + @echo " " + @echo " git checkout dev" + @echo " " + @echo "Build it. There are two new steps not used for earlier releases." + @echo " " + @echo " mkdir build && cd build" + @echo " cmake .." + @echo " make -j4" + @echo " " + @echo "Install:" + @echo " " + @echo " sudo make install" + @echo " make install-conf" + @echo " " + @echo "You will probably need to install additional applications and" + @echo "libraries depending on your operating system." + @echo "More details are in the README.md file." + @echo " " + @echo "Questions?" + @echo " " + @echo " - Extensive documentation can be found in the 'doc' directory." + @echo " - Join the discussion forum here: https://groups.io/g/direwolf" + @echo " " diff --git a/Makefile.linux b/Makefile.linux deleted file mode 100644 index 55777d04..00000000 --- a/Makefile.linux +++ /dev/null @@ -1,909 +0,0 @@ -# -# Makefile for Linux version of Dire Wolf. -# - -APPS := direwolf decode_aprs text2tt tt2text ll2utm utm2ll aclients atest log2gpx gen_packets ttcalc - -all : $(APPS) direwolf.desktop direwolf.conf - @echo " " - @echo "Next step - install with:" - @echo " " - @echo " sudo make install" - @echo " " - -CC := gcc - -# Just for fun, let's see how clang compares to gcc. First install like this: -# sudo apt-get update -# apt-cache search clang -# sudo apt-get install clang-3.5 -# -# CC := clang-3.5 - -# _XOPEN_SOURCE=600 and _DEFAULT_SOURCE=1 are needed for glibc >= 2.24. -# Explanation here: https://github.com/wb2osz/direwolf/issues/62 - -# There are a few source files where it had been necessary to define __USE_XOPEN2KXSI, -# __USE_XOPEN, or _POSIX_C_SOURCE. Doesn't seem to be needed after adding this. - -# The first assignment to CFLAGS and LDFLAGS uses +=, rather than :=, so -# we will inherit options already set in build environment. -# Explanation - https://github.com/wb2osz/direwolf/pull/38 - -CFLAGS += -O3 -pthread -Igeotranz -D_XOPEN_SOURCE=600 -D_DEFAULT_SOURCE=1 -Wall - -# That was fine for a recent Ubuntu and Raspbian Jessie. -# However, Raspbian wheezy was then missing declaration for strsep and definition of fd_set. - -CFLAGS += -D_BSD_SOURCE - -LDFLAGS += -lm -lpthread -lrt - - - -# -# The DSP filters spend a lot of time spinning around in little -# loops multiplying and adding arrays of numbers. The Intel "SSE" -# instructions, introduced in 1999 with the Pentium III series, -# can speed this up considerably. -# -# SSE2 instructions, added in 2000, don't seem to offer any advantage. -# -# -# Let's take a look at the effect of the compile options. -# -# -# Times are elapsed time to process Track 2 of the TNC test CD. -# -# i.e. "./atest 02_Track_2.wav" -# Default demodulator type is new "E" added for version 1.2. -# - -# -# ---------- x86 (32 bit) ---------- -# - -# -# gcc 4.6.3 running on Ubuntu 12.04.05. -# Intel(R) Celeron(R) CPU 2.53GHz. Appears to have only 32 bit instructions. -# Probably from around 2004 or 2005. -# -# When gcc is generating code for a 32 bit x86 target, it assumes the ancient -# i386 processor. This is good for portability but bad for performance. -# -# The code can run considerably faster by taking advantage of the SSE instructions -# available in the Pentium 3 or later. -# -# seconds options comments -# ------ ------- -------- -# 524 -# 183 -O2 -# 182 -O3 -# 183 -O3 -ffast-math (should be same as -Ofast) -# 184 -Ofast -# 189 -O3 -ffast-math -march=pentium -# 122 -O3 -ffast-math -msse -# 122 -O3 -ffast-math -march=pentium -msse -# 121 -O3 -ffast-math -march=pentium3 (this implies -msse) -# 120 -O3 -ffast-math -march=native -# -# Note that "-march=native" is essentially the same as "-march=pentium3." -# - -# If the compiler is generating code for the i386 target, we can -# get much better results by telling it we have at least a Pentium 3. - -arch := $(shell echo | gcc -E -dM - | grep __i386__) -ifneq ($(arch),) -CFLAGS += -march=pentium3 -endif - - -# -# ---------- x86_64 ---------- -# - -# -# gcc 4.8.2 running on Ubuntu 14.04.1. -# Intel Core 2 Duo from around 2007 or 2008. -# -# 64 bit target implies that we have SSE and probably even better vector instructions. -# -# seconds options comments -# ------ ------- -------- -# 245 -# 75 -01 -# 72 -02 -# 71 -03 -# 73 -O3 -march=native -# 42 -O3 -ffast-math -# 42 -Ofast (note below) -# 40 -O3 -ffast-math -march=native -# -# -# Note that "-Ofast" is a newer option roughly equivalent to "-O3 -ffast-math". -# I use the longer form because it is compatible with older compilers. -# -# Why don't I don't have "-march=native?" -# Older compilers don't recognize "native" as one of the valid options. -# One article said it was added with gcc 4.2 but I haven't verified that. -# - -# ---------- How does clang compare? - Ubuntu x86_64 ---------- -# -# I keep hearing a lot about "clang." Let's see how it compares with gcc. -# Simply use different compiler and keep all options the same. -# -# test case: atest 02_Track_2.wav -# -# gcc 4.8.4: 988 packets decoded in 40.503 seconds. 38.3 x realtime -# 988 packets decoded in 40.403 seconds. 38.4 x realtime -# -# clang 3.8.0-2: 988 packets decoded in 77.454 seconds. 20.0 x realtime -# 988 packets decoded in 77.232 seconds. 20.1 x realtime -# -# I'm not impressed. Almost twice as long. Maybe we need to try some other compiler options. -# -march=native did not help. -# Makefile.macosx, which uses clang, has these: -# -fvectorize -fslp-vectorize -fslp-vectorize-aggressive -# Those did not help. -# - - -useffast := $(shell gcc --help -v 2>/dev/null | grep ffast-math) -ifneq ($(useffast),) -CFLAGS += -ffast-math -endif - - -# -# ---------- ARM - Raspberry Pi 1 models ---------- -# -# Raspberry Pi (before RPi model 2), ARM11 (ARMv6 + VFP2) -# gcc (Debian 4.6.3-14+rpi1) 4.6.3 -# -# -# seconds options comments -# ------ ------- --------- -# 892 -O3 -# 887 -O3 -ffast-math -# x -O3 -ffast-math -march=native (error: bad value for -march switch) -# 887 -O3 -ffast-math -mfpu=vfp -# 890 -O3 -ffast-math -march=armv6zk -mcpu=arm1176jzf-s -mfloat-abi=hard -mfpu=vfp -# -# -# The compiler, supplied with Raspbian, is configured with these options which are -# good for the pre version 2 models. -# --with-arch=armv6 --with-fpu=vfp --with-float=hard -# -# Run "gcc --help -v 2" and look near the end. -# -# - -# -# ---------- ARM - Raspberry Pi 2 ---------- -# -# Besides the higher clock speed, the Raspberry Pi 2 has the NEON instruction set -# which which should make things considerably faster. -# -# -# seconds options comments -# ------ ------- --------- -# 426 -O3 -ffast-math (already more than twice as fast) -# 429 -O3 -mfpu=neon -# 419 -O3 -mfpu=neon -funsafe-math-optimizations -# 412 -O3 -ffast-math -mfpu=neon -# 413 -O3 -ffast-math -mfpu=neon-vfpv4 -# 430 -O3 -ffast-math -mfpu=neon-vfpv4 -march=armv7-a -# 412 -O3 -ffast-math -mfpu=neon-vfpv4 -mtune=arm7 -# 410 -O3 -ffast-math -mfpu=neon-vfpv4 -funsafe-math-optimizations - -# -# gcc -march=armv7-a -mfpu=neon-vfpv4 -# -# I expected the -mfpu=neon option to have a much larger impact. -# Adding -march=armv7-a makes it slower! - -# -# If you compile with the RPi 2 specific options above and try to run it on the RPi -# model B (pre version 2), it will die with "illegal instruction." -# -# Dire Wolf is known to work on the BeagleBone, CubieBoard2, etc. -# The best compiler options will depend on the specific type of processor -# and the compiler target defaults. -# - -neon := $(shell cat /proc/cpuinfo | grep neon) -ifneq ($(neon),) -CFLAGS += -mfpu=neon -endif - - -# -# You would expect "-march=native" to produce the fastest code. -# Why don't I use it here? -# -# 1. In my benchmarks, above, it has a negligible impact if any at all. -# 2. Some older versions of gcc don't recognize "native" as a valid choice. -# 3. Results are less portable. Not a consideration if you are -# building only for your own use but very important for anyone -# redistributing a "binary" version. -# -# If you are planning to distribute the binary version to other -# people (in some ham radio software collection, RPM, or DEB package), -# avoid fine tuning it for your particular computer. It could -# cause compatibility issues for those with older computers. -# - -# ---------- How does clang compare? - ARM - Raspberry Pi 2 ---------- -# -# I keep hearing a lot about "clang." Let's see how it compares with gcc. -# Simply use different compiler and keep all options the same. -# -# test case: atest 02_Track_2.wav -# -# gcc 4.9.2-10: 988 packets decoded in 353.025 seconds. 4.4 x realtime -# 988 packets decoded in 352.752 seconds. 4.4 x realtime -# -# clang 3.5.0-10: 988 packets decoded in 825.879 seconds. 1.9 x realtime -# 988 packets decoded in 831.469 seconds. 1.9 x realtime -# -# There might be a different set of options which produce faster code -# but the initial test doesn't look good. About 2.3 times as slow. - -# If you want to use OSS (for FreeBSD, OpenBSD) instead of -# ALSA (for Linux), comment out (or remove) the two lines below. - -# TODO: Can we automate this somehow? - -CFLAGS += -DUSE_ALSA -LDFLAGS += -lasound - - -# Enable GPS if header file is present. -# Finding libgps.so* is more difficult because it -# is in different places on different operating systems. - -enable_gpsd := $(wildcard /usr/include/gps.h) -ifneq ($(enable_gpsd),) -CFLAGS += -DENABLE_GPSD -LDFLAGS += -lgps -endif - - -# Uncomment following lines to enable hamlib support. -# TODO: automate this too. See if hamlib has been installed. - -#CFLAGS += -DUSE_HAMLIB -#LDFLAGS += -lhamlib - - -# Name of current directory. -# Used to generate zip file name for distribution. - -z := $(notdir ${CURDIR}) - - - -# -------------------------------- Main application ----------------------------------------- - - - -direwolf : direwolf.o config.o recv.o demod.o dsp.o demod_afsk.o demod_psk.o demod_9600.o hdlc_rec.o \ - hdlc_rec2.o multi_modem.o rdq.o rrbb.o dlq.o \ - fcs_calc.o ax25_pad.o ax25_pad2.o xid.o \ - decode_aprs.o symbols.o server.o kiss.o kissnet.o kiss_frame.o hdlc_send.o fcs_calc.o \ - gen_tone.o audio.o audio_stats.o digipeater.o cdigipeater.o pfilter.o dedupe.o tq.o xmit.o morse.o \ - ptt.o beacon.o encode_aprs.o latlong.o encode_aprs.o latlong.o textcolor.o \ - dtmf.o aprs_tt.o tt_user.o tt_text.o igate.o waypoint.o serial_port.o log.o telemetry.o \ - dwgps.o dwgpsnmea.o dwgpsd.o dtime_now.o mheard.o ax25_link.o \ - misc.a geotranz.a - $(CC) -o $@ $^ $(LDFLAGS) -ifneq ($(enable_gpsd),) - @echo " " - @echo "This includes support for gpsd." -else - @echo " " - @echo "This does NOT include support for gpsd." -endif - -# Optimization for slow processors. - -demod.o : fsk_fast_filter.h - -demod_afsk.o : fsk_fast_filter.h - - -fsk_fast_filter.h : gen_fff - ./gen_fff > fsk_fast_filter.h - -gen_fff : demod_afsk.c dsp.c textcolor.c - echo " " > tune.h - $(CC) $(CFLAGS) -DGEN_FFF -o $@ $^ $(LDFLAGS) - - -# -# The destination field is often used to identify the manufacturer/model. -# These are not hardcoded into Dire Wolf. Instead they are read from -# a file called tocalls.txt at application start up time. -# -# The original permanent symbols are built in but the "new" symbols, -# using overlays, are often updated. These are also read from files. -# -# You can obtain an updated copy by typing "make tocalls-symbols". -# This is not part of the normal build process. You have to do this explicitly. -# -# The locations below appear to be the most recent. -# The copy at http://www.aprs.org/tocalls.txt is out of date. -# - -.PHONY: tocalls-symbols -tocalls-symbols : - cp tocalls.txt tocalls.txt~ - wget http://www.aprs.org/aprs11/tocalls.txt -O tocalls.txt - -diff -Z tocalls.txt~ tocalls.txt - cp symbols-new.txt symbols-new.txt~ - wget http://www.aprs.org/symbols/symbols-new.txt -O symbols-new.txt - -diff -Z symbols-new.txt~ symbols-new.txt - cp symbolsX.txt symbolsX.txt~ - wget http://www.aprs.org/symbols/symbolsX.txt -O symbolsX.txt - -diff -Z symbolsX.txt~ symbolsX.txt - - -# ---------------------------------------- Other utilities included ------------------------------ - - -# Separate application to decode raw data. - -decode_aprs : decode_aprs.c dwgpsnmea.o dwgps.o dwgpsd.o serial_port.o symbols.o ax25_pad.o textcolor.o fcs_calc.o latlong.o log.o telemetry.o tt_text.o misc.a - $(CC) $(CFLAGS) -DDECAMAIN -o $@ $^ $(LDFLAGS) - - - -# Convert between text and touch tone representation. - -text2tt : tt_text.c misc.a - $(CC) $(CFLAGS) -DENC_MAIN -o $@ $^ $(LDFLAGS) - -tt2text : tt_text.c misc.a - $(CC) $(CFLAGS) -DDEC_MAIN -o $@ $^ $(LDFLAGS) - - -# Convert between Latitude/Longitude and UTM coordinates. - -ll2utm : ll2utm.c geotranz.a textcolor.o misc.a - $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) - -utm2ll : utm2ll.c geotranz.a textcolor.o misc.a - $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) - - -# Convert from log file to GPX. - -log2gpx : log2gpx.c textcolor.o misc.a - $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) - - -# Test application to generate sound. - -gen_packets : gen_packets.c ax25_pad.c hdlc_send.c fcs_calc.c gen_tone.c morse.c dtmf.c textcolor.c dsp.c misc.a - $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) - -# Unit test for AFSK demodulator - -atest : atest.c demod.o demod_afsk.o demod_psk.o demod_9600.o \ - dsp.o hdlc_rec.o hdlc_rec2.o multi_modem.o rrbb.o \ - fcs_calc.o ax25_pad.o decode_aprs.o dwgpsnmea.o \ - dwgps.o dwgpsd.o serial_port.o telemetry.o dtime_now.o latlong.o symbols.o tt_text.o textcolor.o \ - misc.a - $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) - - -# Multiple AGWPE network or serial port clients to test TNCs side by side. - -aclients : aclients.c ax25_pad.c fcs_calc.c textcolor.o misc.a - $(CC) $(CFLAGS) -g -o $@ $^ - - -# Touch Tone to Speech sample application. - -ttcalc : ttcalc.o ax25_pad.o fcs_calc.o textcolor.o misc.a - $(CC) $(CFLAGS) -g -o $@ $^ - - -# ----------------------------------------- Libraries -------------------------------------------- - -# UTM, USNG, MGRS conversions. - -geotranz.a : error_string.o mgrs.o polarst.o tranmerc.o ups.o usng.o utm.o - ar -cr $@ $^ - -error_string.o : geotranz/error_string.c - $(CC) $(CFLAGS) -c -o $@ $^ - -mgrs.o : geotranz/mgrs.c - $(CC) $(CFLAGS) -c -o $@ $^ - -polarst.o : geotranz/polarst.c - $(CC) $(CFLAGS) -c -o $@ $^ - -tranmerc.o : geotranz/tranmerc.c - $(CC) $(CFLAGS) -c -o $@ $^ - -ups.o : geotranz/ups.c - $(CC) $(CFLAGS) -c -o $@ $^ - -usng.o : geotranz/usng.c - $(CC) $(CFLAGS) -c -o $@ $^ - -utm.o : geotranz/utm.c - $(CC) $(CFLAGS) -c -o $@ $^ - - -# Provide our own copy of strlcpy and strlcat because they are not included with Linux. -# We don't need the others in that same directory. - -misc.a : strlcpy.o strlcat.o - ar -cr $@ $^ - -strlcpy.o : misc/strlcpy.c - $(CC) $(CFLAGS) -I. -c -o $@ $^ - -strlcat.o : misc/strlcat.c - $(CC) $(CFLAGS) -I. -c -o $@ $^ - - - -# ------------------------------------- Installation ---------------------------------- - - - -# Generate apprpriate sample configuration file for this platform. -# Originally, there was one sample for all platforms. It got too cluttered -# and confusing saying, this is for windows, and this is for Linux, and this ... -# Trying to maintain 3 different versions in parallel is error prone. -# We now have a single generic version which can be used to generate -# the various platform specific versions. - -# generic.conf should be checked into source control. -# direwolf.conf should NOT. It is generated when compiling on the target platform. - -direwolf.conf : generic.conf - egrep '^C|^L' generic.conf | cut -c2-999 > direwolf.conf - - -# Where should we install it? - -# My understanding, of the convention, is that something you compile -# from source, that is not a standard part of the operating system, -# should go in /usr/local/bin. - -# However, if you are preparing a "binary" DEB or RPM package, the -# installation location should be /usr/bin. - -# This is a step in the right direction but not sufficient to use /usr instead. -# Eventually I'd like to have targets here to build the .DEB and .RPM packages. - -INSTALLDIR := /usr/local - -# Command to "install" to system directories. Use "ginstall" for Mac. - -INSTALL=install - -# direwolf.desktop was previously handcrafted for the Raspberry Pi. -# It was hardcoded with lxterminal, /home/pi, and so on. -# In version 1.2, try to customize this to match other situations better. - -# TODO: Test this better. - - -direwolf.desktop : - @echo "Generating customized direwolf.desktop ..." - @echo '[Desktop Entry]' > $@ - @echo 'Type=Application' >> $@ -ifneq ($(wildcard /usr/bin/lxterminal),) - @echo "Exec=lxterminal -t \"Dire Wolf\" -e \"$(INSTALLDIR)/bin/direwolf\"" >> $@ -else ifneq ($(wildcard /usr/bin/lxterm),) - @echo "Exec=lxterm -hold -title \"Dire Wolf\" -bg white -e \"$(INSTALLDIR)/bin/direwolf\"" >> $@ -else - @echo "Exec=xterm -hold -title \"Dire Wolf\" -bg white -e \"$(INSTALLDIR)/bin/direwolf\"" >> $@ -endif - @echo 'Name=Dire Wolf' >> $@ - @echo 'Comment=APRS Soundcard TNC' >> $@ - @echo 'Icon=/usr/share/direwolf/dw-icon.png' >> $@ - @echo "Path=$(HOME)" >> $@ - @echo '#Terminal=true' >> $@ - @echo 'Categories=HamRadio' >> $@ - @echo 'Keywords=Ham Radio;APRS;Soundcard TNC;KISS;AGWPE;AX.25' >> $@ - - -# Installation into /usr/local/... -# Needs to be run as root or with sudo. - - -.PHONY: install -install : $(APPS) direwolf.conf tocalls.txt symbols-new.txt symbolsX.txt dw-icon.png direwolf.desktop -# -# Applications, not installed with package manager, normally go in /usr/local/bin. -# /usr/bin is used instead when installing from .DEB or .RPM package. -# - $(INSTALL) direwolf $(INSTALLDIR)/bin - $(INSTALL) decode_aprs $(INSTALLDIR)/bin - $(INSTALL) text2tt $(INSTALLDIR)/bin - $(INSTALL) tt2text $(INSTALLDIR)/bin - $(INSTALL) ll2utm $(INSTALLDIR)/bin - $(INSTALL) utm2ll $(INSTALLDIR)/bin - $(INSTALL) aclients $(INSTALLDIR)/bin - $(INSTALL) log2gpx $(INSTALLDIR)/bin - $(INSTALL) gen_packets $(INSTALLDIR)/bin - $(INSTALL) atest $(INSTALLDIR)/bin - $(INSTALL) ttcalc $(INSTALLDIR)/bin - $(INSTALL) dwespeak.sh $(INSTALLDIR)/bin -# -# Telemetry Toolkit executables. Other .conf and .txt files will go into doc directory. -# - $(INSTALL) telemetry-toolkit/telem-balloon.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-bits.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-data.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-data91.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-eqns.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-parm.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-seq.sh $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-unit.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-volts.py $(INSTALLDIR)/bin -# -# Misc. data such as "tocall" to system mapping. -# - $(INSTALL) -D --mode=644 tocalls.txt /usr/share/direwolf/tocalls.txt - $(INSTALL) -D --mode=644 symbols-new.txt /usr/share/direwolf/symbols-new.txt - $(INSTALL) -D --mode=644 symbolsX.txt /usr/share/direwolf/symbolsX.txt - $(INSTALL) -D --mode=644 dw-icon.png /usr/share/direwolf/dw-icon.png - $(INSTALL) -D --mode=644 direwolf.desktop /usr/share/applications/direwolf.desktop -# -# Documentation. Various plain text files and PDF. -# - $(INSTALL) -D --mode=644 CHANGES.md $(INSTALLDIR)/share/doc/direwolf/CHANGES.md - $(INSTALL) -D --mode=644 LICENSE-dire-wolf.txt $(INSTALLDIR)/share/doc/direwolf/LICENSE-dire-wolf.txt - $(INSTALL) -D --mode=644 LICENSE-other.txt $(INSTALLDIR)/share/doc/direwolf/LICENSE-other.txt -# -# ./README.md is an overview for the project main page. -# doc/README.md contains an overview of the PDF file contents and is more useful here. -# - $(INSTALL) -D --mode=644 doc/README.md $(INSTALLDIR)/share/doc/direwolf/README.md - $(INSTALL) -D --mode=644 doc/User-Guide.pdf $(INSTALLDIR)/share/doc/direwolf/User-Guide.pdf - $(INSTALL) -D --mode=644 doc/Raspberry-Pi-APRS.pdf $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-APRS.pdf - $(INSTALL) -D --mode=644 doc/Raspberry-Pi-APRS-Tracker.pdf $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-APRS-Tracker.pdf - $(INSTALL) -D --mode=644 doc/Raspberry-Pi-SDR-IGate.pdf $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-SDR-IGate.pdf - $(INSTALL) -D --mode=644 doc/APRStt-Implementation-Notes.pdf $(INSTALLDIR)/share/doc/direwolf/APRStt-Implementation-Notes.pdf - $(INSTALL) -D --mode=644 doc/APRStt-interface-for-SARTrack.pdf $(INSTALLDIR)/share/doc/direwolf/APRStt-interface-for-SARTrack.pdf - $(INSTALL) -D --mode=644 doc/APRS-Telemetry-Toolkit.pdf $(INSTALLDIR)/share/doc/direwolf/APRS-Telemetry-Toolkit.pdf - $(INSTALL) -D --mode=644 doc/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf $(INSTALLDIR)/share/doc/direwolf/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf - $(INSTALL) -D --mode=644 doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf $(INSTALLDIR)/share/doc/direwolf/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf -# -# Various sample config and other files go into examples under the doc directory. -# When building from source, these can be put in home directory with "make install-conf". -# When installed from .DEB or .RPM package, the user will need to copy these to -# the home directory or other desired location. -# - $(INSTALL) -D --mode=644 direwolf.conf $(INSTALLDIR)/share/doc/direwolf/examples/direwolf.conf - $(INSTALL) -D --mode=755 dw-start.sh $(INSTALLDIR)/share/doc/direwolf/examples/dw-start.sh - $(INSTALL) -D --mode=644 sdr.conf $(INSTALLDIR)/share/doc/direwolf/examples/sdr.conf - $(INSTALL) -D --mode=644 telemetry-toolkit/telem-m0xer-3.txt $(INSTALLDIR)/share/doc/direwolf/examples/telem-m0xer-3.txt - $(INSTALL) -D --mode=644 telemetry-toolkit/telem-balloon.conf $(INSTALLDIR)/share/doc/direwolf/examples/telem-balloon.conf - $(INSTALL) -D --mode=644 telemetry-toolkit/telem-volts.conf $(INSTALLDIR)/share/doc/direwolf/examples/telem-volts.conf -# -# "man" pages -# - $(INSTALL) -D --mode=644 man1/aclients.1 $(INSTALLDIR)/man/man1/aclients.1 - $(INSTALL) -D --mode=644 man1/atest.1 $(INSTALLDIR)/man/man1/atest.1 - $(INSTALL) -D --mode=644 man1/decode_aprs.1 $(INSTALLDIR)/man/man1/decode_aprs.1 - $(INSTALL) -D --mode=644 man1/direwolf.1 $(INSTALLDIR)/man/man1/direwolf.1 - $(INSTALL) -D --mode=644 man1/gen_packets.1 $(INSTALLDIR)/man/man1/gen_packets.1 - $(INSTALL) -D --mode=644 man1/ll2utm.1 $(INSTALLDIR)/man/man1/ll2utm.1 - $(INSTALL) -D --mode=644 man1/log2gpx.1 $(INSTALLDIR)/man/man1/log2gpx.1 - $(INSTALL) -D --mode=644 man1/text2tt.1 $(INSTALLDIR)/man/man1/text2tt.1 - $(INSTALL) -D --mode=644 man1/tt2text.1 $(INSTALLDIR)/man/man1/tt2text.1 - $(INSTALL) -D --mode=644 man1/utm2ll.1 $(INSTALLDIR)/man/man1/utm2ll.1 -# - @echo " " - @echo "If this is your first install, not an upgrade, type this to put a copy" - @echo "of the sample configuration file (direwolf.conf) in your home directory:" - @echo " " - @echo " make install-conf" - @echo " " - - -# Put sample configuration files in home directory. -# These would be done as ordinary user. - -# The Raspberry Pi has ~/Desktop but Ubuntu does not. - -# TODO: Handle Linux variations correctly. - -# Version 1.4 - Add "-n" option to avoid clobbering existing, probably customized, config files. - - -.PHONY: install-conf -install-conf : direwolf.conf - cp -n direwolf.conf ~ - cp -n sdr.conf ~ - cp -n telemetry-toolkit/telem-m0xer-3.txt ~ - cp -n telemetry-toolkit/telem-*.conf ~ -ifneq ($(wildcard $(HOME)/Desktop),) - @echo " " - @echo "This will add a desktop icon on some systems:" - @echo " " - @echo " make install-rpi" - @echo " " -endif - - -# dw-start.sh is greatly improved in version 1.4. -# It should probably be part of install-conf because it is not just for the RPi. - -.PHONY: install-rpi -install-rpi : dw-start.sh - chmod +x dw-start.sh - cp -n dw-start.sh ~ - ln -f -s /usr/share/applications/direwolf.desktop ~/Desktop/direwolf.desktop - - - -# ---------------------------------- Automated Smoke Test -------------------------------- - - - -# Combine some unit tests into a single regression sanity check. - - -check : dtest ttest tttexttest pftest tlmtest lltest enctest kisstest pad2test xidtest dtmftest check-modem1200 check-modem300 check-modem9600 check-modem19200 check-modem2400 check-modem4800 - -# Can we encode and decode at popular data rates? - -check-modem1200 : gen_packets atest - ./gen_packets -n 100 -o /tmp/test12.wav - ./atest -F0 -PE -L63 -G71 /tmp/test12.wav - ./atest -F1 -PE -L70 -G75 /tmp/test12.wav - rm /tmp/test12.wav - -check-modem300 : gen_packets atest - ./gen_packets -B300 -n 100 -o /tmp/test3.wav - ./atest -B300 -F0 -L68 -G69 /tmp/test3.wav - ./atest -B300 -F1 -L73 -G75 /tmp/test3.wav - rm /tmp/test3.wav - -check-modem9600 : gen_packets atest - ./gen_packets -B9600 -n 100 -o /tmp/test96.wav - ./atest -B9600 -F0 -L50 -G54 /tmp/test96.wav - ./atest -B9600 -F1 -L55 -G59 /tmp/test96.wav - rm /tmp/test96.wav - -check-modem19200 : gen_packets atest - ./gen_packets -r 96000 -B19200 -n 100 -o /tmp/test19.wav - ./atest -B19200 -F0 -L55 -G59 /tmp/test19.wav - ./atest -B19200 -F1 -L60 -G64 /tmp/test19.wav - rm /tmp/test19.wav - -check-modem2400 : gen_packets atest - ./gen_packets -B2400 -n 100 -o /tmp/test24.wav - ./atest -B2400 -F0 -L70 -G78 /tmp/test24.wav - ./atest -B2400 -F1 -L80 -G87 /tmp/test24.wav - rm /tmp/test24.wav - -check-modem4800 : gen_packets atest - ./gen_packets -B2400 -n 100 -o /tmp/test48.wav - ./atest -B2400 -F0 -L70 -G79 /tmp/test48.wav - ./atest -B2400 -F1 -L80 -G90 /tmp/test48.wav - rm /tmp/test48.wav - - -# Unit test for inner digipeater algorithm - -.PHONY : dtest -dtest : digipeater.c dedupe.c pfilter.c \ - ax25_pad.o fcs_calc.o tq.o textcolor.o \ - decode_aprs.o dwgpsnmea.o dwgps.o dwgpsd.o serial_port.o latlong.o telemetry.o symbols.o tt_text.o misc.a - $(CC) $(CFLAGS) -DDIGITEST -o $@ $^ $(LDFLAGS) - ./dtest - rm dtest - - -# Unit test for APRStt tone sequence parsing. - -.PHONY : ttest -ttest : aprs_tt.c tt_text.c latlong.o textcolor.o misc.a geotranz.a misc.a - $(CC) $(CFLAGS) -DTT_MAIN -o $@ $^ $(LDFLAGS) - ./ttest - rm ttest - - -# Unit test for APRStt tone sequence / text conversions. - -.PHONY: tttexttest -tttexttest : tt_text.c textcolor.o misc.a - $(CC) $(CFLAGS) -DTTT_TEST -o $@ $^ $(LDFLAGS) - ./tttexttest - rm tttexttest - - -# Unit test for Packet Filtering. - -.PHONY: pftest -pftest : pfilter.c ax25_pad.o textcolor.o fcs_calc.o decode_aprs.o dwgpsnmea.o dwgps.o dwgpsd.o serial_port.o latlong.o symbols.o telemetry.o tt_text.o misc.a - $(CC) $(CFLAGS) -DPFTEST -o $@ $^ $(LDFLAGS) - ./pftest - rm pftest - -# Unit test for telemetry decoding. - -.PHONY: tlmtest -tlmtest : telemetry.c ax25_pad.o fcs_calc.o textcolor.o misc.a - $(CC) $(CFLAGS) -DTEST -o $@ $^ $(LDFLAGS) - ./tlmtest - rm tlmtest - -# Unit test for location coordinate conversion. - -.PHONY: lltest -lltest : latlong.c textcolor.o misc.a - $(CC) $(CFLAGS) -DLLTEST -o $@ $^ $(LDFLAGS) - ./lltest - rm lltest - -# Unit test for encoding position & object report. - -.PHONY: enctest -enctest : encode_aprs.c latlong.c textcolor.c misc.a - $(CC) $(CFLAGS) -DEN_MAIN -o $@ $^ $(LDFLAGS) - ./enctest - rm enctest - - -# Unit test for KISS encapsulation. - -.PHONY: kisstest -kisstest : kiss_frame.c - $(CC) $(CFLAGS) -DKISSTEST -o $@ $^ $(LDFLAGS) - ./kisstest - rm kisstest - -# Unit test for constructing frames besides UI. - -.PHONY: pad2test -pad2test : ax25_pad2.c ax25_pad.c fcs_calc.o textcolor.o misc.a - $(CC) $(CFLAGS) -DPAD2TEST -o $@ $^ $(LDFLAGS) - ./pad2test - rm pad2test - - -# Unit Test for XID frame encode/decode. - -.PHONY: xidtest -xidtest : xid.c textcolor.o misc.a - $(CC) $(CFLAGS) -DXIDTEST -o $@ $^ $(LDFLAGS) - ./xidtest - rm xidtest - - -# Unit Test for DTMF encode/decode. - -.PHONY: dtmftest -dtmftest : dtmf.c textcolor.o - $(CC) $(CFLAGS) -DDTMF_TEST -o $@ $^ $(LDFLAGS) - ./dtmftest - rm dtmftest - - - -# ----------------------------- Manual tests and experiments --------------------------- - -# These are not included in a normal build. Might be broken. - -# Unit test for IGate - -itest : igate.c textcolor.c ax25_pad.c fcs_calc.c textcolor.o misc.a - $(CC) $(CFLAGS) -DITEST -o $@ $^ - ./itest - -# Unit test for UDP reception with AFSK demodulator. -# Temporary during development. Might not be useful anymore. - -udptest : udp_test.c demod.o dsp.o demod_afsk.o demod_psk.o demod_9600.o hdlc_rec.o hdlc_rec2.o multi_modem.o rrbb.o \ - fcs_calc.o ax25_pad.o decode_aprs.o symbols.o textcolor.o misc.a - $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) - ./udptest - -# For demodulator tweaking experiments. -# Dependencies of demod*.c, rather than .o, are intentional. - -demod.o : tune.h - -demod_afsk.o : tune.h - -demod_9600.o : tune.h - -demod_psk.o : tune.h - -tune.h : - echo " " > tune.h - - -testagc : atest.c demod.c dsp.c demod_afsk.c demod_psk.c demod_9600.c hdlc_rec.o hdlc_rec2.o multi_modem.o rrbb.o \ - fcs_calc.o ax25_pad.o decode_aprs.o telemetry.o dtime_now.o latlong.o symbols.o tune.h textcolor.o misc.a - $(CC) $(CFLAGS) -o atest $^ $(LDFLAGS) - ./atest 02_Track_2.wav | grep "packets decoded in" > atest.out - - -testagc96 : atest.c fsk_fast_filter.h tune.h demod.c demod_afsk.c demod_psk.c demod_9600.c \ - dsp.o hdlc_rec.o hdlc_rec2.o multi_modem.o \ - rrbb.o fcs_calc.o ax25_pad.o decode_aprs.o \ - dwgpsnmea.o dwgps.o dwgpsd.o serial_port.o latlong.o \ - symbols.o tt_text.o textcolor.o telemetry.o dtime_now.o \ - misc.a - rm -f atest96 - $(CC) $(CFLAGS) -o atest96 $^ $(LDFLAGS) - ./atest96 -B 9600 ../walkabout9600c.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 noisy96.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 19990303_0225_9600_8bis_22kHz.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 19990303_0225_9600_16bit_22kHz.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 -P + z8-22k.wav| grep "packets decoded in" >atest.out - #./atest96 -B 9600 test9600.wav | grep "packets decoded in" >atest.out - echo " " > tune.h - - - - -# ------------------------------- Source distribution --------------------------------- - -# probably obsolete and can be removed after move to github. - - - -.PHONY: dist-src -dist-src : README.md CHANGES.md - doc/User-Guide.pdf doc/Raspberry-Pi-APRS.pdf \ - doc/Raspberry-Pi-APRS-Tracker.pdf doc/APRStt-Implementation-Notes.pdf \ - dw-start.sh dwespeak.bat dwespeak.sh \ - tocalls.txt symbols-new.txt symbolsX.txt direwolf.spec - rm -f fsk_fast_filter.h - echo " " > tune.h - rm -f ../$z-src.zip - (cd .. ; zip $z-src.zip \ - $z/README.md \ - $z/CHANGES.md \ - $z/LICENSE* \ - $z/doc/User-Guide.pdf \ - $z/doc/Raspberry-Pi-APRS.pdf \ - $z/doc/Raspberry-Pi-APRS-Tracker.pdf \ - $z/doc/APRStt-Implementation-Notes.pdf \ - $z/doc/APRS-Telemetry-Toolkit.pdf \ - $z/Makefile* \ - $z/*.c $z/*.h \ - $z/regex/* $z/misc/* $z/geotranz/* \ - $z/man1/* \ - $z/generic.conf \ - $z/tocalls.txt $z/symbols-new.txt $z/symbolsX.txt \ - $z/dw-icon.png $z/dw-icon.rc $z/dw-icon.ico \ - $z/dw-start.sh $z/direwolf.spec \ - $z/dwespeak.bat $z/dwespeak.sh \ - $z/telemetry-toolkit/* ) - - -# ----------------------------------------------------------------------------------------- - - -.PHONY: clean -clean : - rm -f $(APPS) gen_fff tune.h fsk_fast_filter.h *.o *.a direwolf.desktop - - -depend : $(wildcard *.c) - makedepend -f $(lastword $(MAKEFILE_LIST)) -- $(CFLAGS) -- $^ - - -# -# The following is updated by "make depend" -# -# DO NOT DELETE - - diff --git a/Makefile.macosx b/Makefile.macosx deleted file mode 100644 index 262ed453..00000000 --- a/Makefile.macosx +++ /dev/null @@ -1,641 +0,0 @@ -# -# Makefile for Macintosh 10.6+ version of Dire Wolf. -# - -# TODO: This is a modified version of Makefile.linux and it -# has fallen a little behind. For example, it is missing the check target. -# It would be more maintainable if we could use a single file for both. -# The differences are not that great. -# Maybe the most of the differences could go in to platform specific include -# files rather than cluttering it up with too many if blocks. - -# Changes: -# -# 16 Dec 2015 -# 1. Added condition check for gps/gpsd code. Commented out due to 32/64 bit -# library issues. Macports gpsd build problem. -# 2. SDK version checks are now performed by a bash script 'search_sdks.sh'. -# This should resolve the varied locations Apple stored the SDKs on the different -# Xcode/OS versions. Executing 'make' on the first pass asks the operator -# which SDK he/she wishes to use. Executing 'make clean' resets the SDK -# selection and operator intervention is once again required. Selected SDK -# information resides in a file named './use_this_sdk' in the current working -# directory. -# 3. Removed fsk_fast_filter.h from atest receipe, clang compiler was having -# a hissy fit. Not check with GCC. - -APPS := direwolf decode_aprs text2tt tt2text ll2utm utm2ll aclients atest log2gpx gen_packets ttcalc - -all : $(APPS) direwolf.desktop direwolf.conf @echo " " - @echo "Next step install with: " - @echo " " - @echo " sudo make install" - @echo " " - @echo " " - -SYS_LIBS := -SYS_MIN := -#SDK := $(shell find /Developer -maxdepth 1 -type d -name "SDKs") -#$(info $$SDK = ${SDK}) -#ifeq (${SDK},/Developer/SDKs) -# SDK := $(shell find /Developer/SDKs -maxdepth 1 -type d -name "MacOSX10.8.sdk") -# ifeq (${SDK},/Developer/SDKs/MacOSX10.8.sdk) -# SYS_LIBS := -isystem /Developer/SDKs/MacOSX10.8.sdk -# SYS_MIN := -mmacosx-version-min=10.8 -# else -# SDK := $(shell find /Developer/SDKs -maxdepth 1 -type d -name "MacOSX10.9.sdk") -# ifeq (${SDK},/Developer/SDKs/MacOSX10.9.sdk) -# SYS_LIBS := -isystem /Developer/SDKs/MacOSX10.9.sdk -# SYS_MIN := -mmacosx-version-min=10.9 -# else -# SDK := $(shell find /Developer/SDKs -maxdepth 1 -type d -name "MacOSX10.10.sdk") -# ifeq (${SDK},/Developer/SDKs/MacOSX10.10.sdk) -# SYS_LIBS := -isystem /Developer/SDKs/MacOSX10.10.sdk -# SYS_MIN := -mmacosx-version-min=10.10 -# endif -# endif -# endif -#endif - -SYS_LIBS := $(shell ./search_sdks.sh) -EXTRA_CFLAGS := -DARWIN_CC := $(shell which clang) -ifeq (${DARWIN_CC},) -DARWIN_CC := $(shell which gcc) -EXTRA_CFLAGS := -else -EXTRA_CFLAGS := -fvectorize -fslp-vectorize -fslp-vectorize-aggressive -pthread -endif - -# Change as required in support of the available libraries - -#CC := $(DARWIN_CC) -m64 $(SYS_LIBS) $(SYS_MIN) -CC := $(DARWIN_CC) -m32 $(SYS_LIBS) $(SYS_MIN) - -# _XOPEN_SOURCE=600 and _DEFAULT_SOURCE=1 are needed for glibc >= 2.24. -# Explanation here: https://github.com/wb2osz/direwolf/issues/62 - -CFLAGS := -Os -pthread -Igeotranz -D_XOPEN_SOURCE=600 -D_DEFAULT_SOURCE=1 $(EXTRA_CFLAGS) - -# That was fine for a recent Ubuntu and Raspbian Jessie. -# However, Raspbian wheezy was then missing declaration for strsep and definition of fd_set. - -CFLAGS += -D_BSD_SOURCE - - -# $(info $$CC is [${CC}]) - -# -# The DSP filters spend a lot of time spinning around in little -# loops multiplying and adding arrays of numbers. The Intel "SSE" -# instructions, introduced in 1999 with the Pentium III series, -# can speed this up considerably. -# -# SSE2 instructions, added in 2000, don't seem to offer any advantage. -# -# -# Let's take a look at the effect of the compile options. -# -# -# Times are elapsed time to process Track 2 of the TNC test CD. -# -# i.e. "./atest 02_Track_2.wav" -# Default demodulator type is new "E" added for version 1.2. -# - -# -# ---------- x86 (32 bit) ---------- -# - -# -# gcc 4.6.3 running on Ubuntu 12.04.05. -# Intel(R) Celeron(R) CPU 2.53GHz. Appears to have only 32 bit instructions. -# Probably from around 2004 or 2005. -# -# When gcc is generating code for a 32 bit x86 target, it assumes the ancient -# i386 processor. This is good for portability but bad for performance. -# -# The code can run considerably faster by taking advantage of the SSE instructions -# available in the Pentium 3 or later. -# -# seconds options comments -# ------ ------- -------- -# 524 -# 183 -O2 -# 182 -O3 -# 183 -O3 -ffast-math (should be same as -Ofast) -# 184 -Ofast -# 189 -O3 -ffast-math -march=pentium -# 122 -O3 -ffast-math -msse -# 122 -O3 -ffast-math -march=pentium -msse -# 121 -O3 -ffast-math -march=pentium3 (this implies -msse) -# 120 -O3 -ffast-math -march=native -# -# Note that "-march=native" is essentially the same as "-march=pentium3." -# - -# If the compiler is generating code for the i386 target, we can -# get much better results by telling it we have at least a Pentium 3. - -CFLAGS += -march=core2 -msse4.1 -std=gnu99 -#CFLAGS += -march=pentium3 -sse - -# -# gcc 4.8.2 running on Ubuntu 14.04.1. -# Intel Core 2 Duo from around 2007 or 2008. -# -# 64 bit target implies that we have SSE and probably even better vector instructions. -# -# seconds options comments -# ------ ------- -------- -# 245 -# 75 -01 -# 72 -02 -# 71 -03 -# 73 -O3 -march=native -# 42 -O3 -ffast-math -# 42 -Ofast (note below) -# 40 -O3 -ffast-math -march=native -# -# -# Note that "-Ofast" is a newer option roughly equivalent to "-O3 -ffast-math". -# I use the longer form because it is compatible with older compilers. -# -# Why don't I don't have "-march=native?" -# Older compilers don't recognize "native" as one of the valid options. -# One article said it was added with gcc 4.2 but I haven't verified that. -# - -# Add -ffastmath in only if compiler version recognizes it. - -useffast := $(shell gcc --help -v 2>/dev/null | grep ffast-math) -ifneq ($(useffast),) -CFLAGS += -ffast-math -endif - -# -# You would expect "-march=native" to produce the fastest code. -# Why don't I use it here? -# -# 1. In my benchmarks, above, it has a negligible impact if any at all. -# 2. Some older versions of gcc don't recognize "native" as a valid choice. -# 3. Results are less portable. Not a consideration if you are -# building only for your own use but very important for anyone -# redistributing a "binary" version. -# -# If you are planning to distribute the binary version to other -# people (in some ham radio software collection, RPM, or DEB package), -# avoid # fine tuning it for your particular computer. It could -# cause compatibility issues for those with older computers. -# - -#CFLAGS += -D_FORTIFY_SOURCE - -# Use PortAudio Library - -# Force static linking of portaudio if the static library is available. -PA_LIB_STATIC := $(shell find /opt/local/lib -maxdepth 1 -type f -name "libportaudio.a") -#$(info $$PA_LIB_STATIC is [${PA_LIB_STATIC}]) -ifeq (${PA_LIB_STATIC},) -LDLIBS += -L/opt/local/lib -lportaudio -else -LDLIBS += /opt/local/lib/libportaudio.a -endif - -# Include libraries portaudio requires. -LDLIBS += -framework CoreAudio -framework AudioUnit -framework AudioToolbox -LDLIBS += -framework Foundation -framework CoreServices - -CFLAGS += -DUSE_PORTAUDIO -I/opt/local/include - -# Uncomment following lines to enable GPS interface & tracker function. -# Not available for MacOSX (as far as I know). -# Although MacPorts has gpsd, wonder if it's the same thing. Add the check -# just in case it works. -# Well never mind, issue with Macports with 64bit libs ;-( leave the check in -# until (if ever) Macports fixes the issue. - -#GPS_HEADER := $(shell find /opt/local/include -maxdepth 1 -type f -name "gps.h") -#ifeq (${GPS_HEADER},) -#GPS_OBJS := -#else -#CFLAGS += -DENABLE_GPSD -#LDLIBS += -L/opt/local/lib -lgps -lgpsd -#GPS_OBJS := dwgps.o dwgpsnmea.o dwgpsd.o -#endif - -# Name of current directory. -# Used to generate zip file name for distribution. - -z := $(notdir ${CURDIR}) - - -# Main application. - -direwolf : direwolf.o aprs_tt.o audio_portaudio.o audio_stats.o ax25_link.o ax25_pad.o ax25_pad2.o beacon.o \ - config.o decode_aprs.o dedupe.o demod_9600.o demod_afsk.o demod_psk.o \ - demod.o digipeater.o cdigipeater.o dlq.o dsp.o dtime_now.o dtmf.o dwgps.o \ - encode_aprs.o encode_aprs.o fcs_calc.o fcs_calc.o gen_tone.o \ - geotranz.a hdlc_rec.o hdlc_rec2.o hdlc_send.o igate.o kiss_frame.o \ - kiss.o kissnet.o latlong.o latlong.o log.o morse.o multi_modem.o \ - waypoint.o serial_port.o pfilter.o ptt.o rdq.o recv.o rrbb.o server.o \ - symbols.o telemetry.o textcolor.o tq.o tt_text.o tt_user.o xid.o xmit.o \ - dwgps.o dwgpsnmea.o mheard.o - $(CC) $(CFLAGS) -o $@ $^ -lpthread $(LDLIBS) -lm - - -# Optimization for slow processors. - -demod.o : fsk_fast_filter.h - -demod_afsk.o : fsk_fast_filter.h - - -fsk_fast_filter.h : gen_fff - ./gen_fff > fsk_fast_filter.h - -gen_fff : demod_afsk.c dsp.c textcolor.c - echo " " > tune.h - $(CC) $(CFLAGS) -DGEN_FFF -o $@ $^ $(LDFLAGS) - - - -# UTM, USNG, MGRS conversions. - -geotranz.a : error_string.o mgrs.o polarst.o tranmerc.o ups.o usng.o utm.o - ar -cr $@ $^ - -error_string.o : geotranz/error_string.c - $(CC) $(CFLAGS) -c -o $@ $^ - -mgrs.o : geotranz/mgrs.c - $(CC) $(CFLAGS) -c -o $@ $^ - -polarst.o : geotranz/polarst.c - $(CC) $(CFLAGS) -c -o $@ $^ - -tranmerc.o : geotranz/tranmerc.c - $(CC) $(CFLAGS) -c -o $@ $^ - -ups.o : geotranz/ups.c - $(CC) $(CFLAGS) -c -o $@ $^ - -usng.o : geotranz/usng.c - $(CC) $(CFLAGS) -c -o $@ $^ - -utm.o : geotranz/utm.c - $(CC) $(CFLAGS) -c -o $@ $^ - - - -# Generate apprpriate sample configuration file for this platform. - -direwolf.conf : generic.conf - egrep '^C|^M' generic.conf | cut -c2-999 > direwolf.conf - - -# Where should we install it? - -# My understanding, of the convention, is that something you compile -# from source, that is not a standard part of the operating system, -# should go in /usr/local/bin. - -# This is a step in the right direction but not sufficient to use /usr instead. - -INSTALLDIR := /usr/local - -# TODO: Test this better. - -# Optional installation into /usr/local/... -# Needs to be run as root or with sudo. -# TODO: Review file locations. - -# Command to "install" to system directories. "install" for Linux. "ginstall" for Mac. - -INSTALL=ginstall - -.PHONY: install -install : $(APPS) direwolf.conf tocalls.txt symbols-new.txt symbolsX.txt dw-icon.png direwolf.desktop -# -# Applications, not installed with package manager, normally go in /usr/local/bin. -# /usr/bin is used instead when installing from .DEB or .RPM package. -# - $(INSTALL) direwolf $(INSTALLDIR)/bin - $(INSTALL) decode_aprs $(INSTALLDIR)/bin - $(INSTALL) text2tt $(INSTALLDIR)/bin - $(INSTALL) tt2text $(INSTALLDIR)/bin - $(INSTALL) ll2utm $(INSTALLDIR)/bin - $(INSTALL) utm2ll $(INSTALLDIR)/bin - $(INSTALL) aclients $(INSTALLDIR)/bin - $(INSTALL) log2gpx $(INSTALLDIR)/bin - $(INSTALL) gen_packets $(INSTALLDIR)/bin - $(INSTALL) atest $(INSTALLDIR)/bin - $(INSTALL) ttcalc $(INSTALLDIR)/bin - $(INSTALL) dwespeak.sh $(INSTALLDIR)/bin -# -# Telemetry Toolkit executables. Other .conf and .txt files will go into doc directory. -# - $(INSTALL) telemetry-toolkit/telem-balloon.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-bits.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-data.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-data91.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-eqns.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-parm.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-unit.pl $(INSTALLDIR)/bin - $(INSTALL) telemetry-toolkit/telem-volts.py $(INSTALLDIR)/bin -# -# Misc. data such as "tocall" to system mapping. -# - $(INSTALL) -D --mode=644 tocalls.txt /usr/share/direwolf/tocalls.txt - $(INSTALL) -D --mode=644 symbols-new.txt /usr/share/direwolf/symbols-new.txt - $(INSTALL) -D --mode=644 symbolsX.txt /usr/share/direwolf/symbolsX.txt - $(INSTALL) -D --mode=644 dw-icon.png /usr/share/direwolf/dw-icon.png - $(INSTALL) -D --mode=644 direwolf.desktop /usr/share/applications/direwolf.desktop -# -# Documentation. Various plain text files and PDF. -# - $(INSTALL) -D --mode=644 README.md $(INSTALLDIR)/share/doc/direwolf/README.md - $(INSTALL) -D --mode=644 CHANGES.md $(INSTALLDIR)/share/doc/direwolf/CHANGES.md - $(INSTALL) -D --mode=644 LICENSE-dire-wolf.txt $(INSTALLDIR)/share/doc/direwolf/LICENSE-dire-wolf.txt - $(INSTALL) -D --mode=644 LICENSE-other.txt $(INSTALLDIR)/share/doc/direwolf/LICENSE-other.txt -# - $(INSTALL) -D --mode=644 doc/User-Guide.pdf $(INSTALLDIR)/share/doc/direwolf/User-Guide.pdf - $(INSTALL) -D --mode=644 doc/Raspberry-Pi-APRS.pdf $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-APRS.pdf - $(INSTALL) -D --mode=644 doc/Raspberry-Pi-APRS-Tracker.pdf $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-APRS-Tracker.pdf - $(INSTALL) -D --mode=644 doc/Raspberry-Pi-SDR-IGate.pdf $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-SDR-IGate.pdf - $(INSTALL) -D --mode=644 doc/APRStt-Implementation-Notes.pdf $(INSTALLDIR)/share/doc/direwolf/APRStt-Implementation-Notes.pdf - $(INSTALL) -D --mode=644 doc/APRStt-interface-for-SARTrack.pdf $(INSTALLDIR)/share/doc/direwolf/APRStt-interface-for-SARTrack.pdf - $(INSTALL) -D --mode=644 doc/APRS-Telemetry-Toolkit.pdf $(INSTALLDIR)/share/doc/direwolf/APRS-Telemetry-Toolkit.pdf - $(INSTALL) -D --mode=644 doc/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf $(INSTALLDIR)/share/doc/direwolf/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf - $(INSTALL) -D --mode=644 doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf $(INSTALLDIR)/share/doc/direwolf/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf -# -# Sample config files also go into the doc directory. -# When building from source, these can be put in home directory with "make install-conf". -# When installed from .DEB or .RPM package, the user will need to copy these to -# the home directory or other desired location. -# Someone suggested that these could go into an "examples" subdirectory under doc. -# - $(INSTALL) -D --mode=644 direwolf.conf $(INSTALLDIR)/share/doc/direwolf/direwolf.conf - $(INSTALL) -D --mode=644 telemetry-toolkit/telem-m0xer-3.txt $(INSTALLDIR)/share/doc/direwolf/telem-m0xer-3.txt - $(INSTALL) -D --mode=644 telemetry-toolkit/telem-balloon.conf $(INSTALLDIR)/share/doc/direwolf/telem-balloon.conf - $(INSTALL) -D --mode=644 telemetry-toolkit/telem-volts.conf $(INSTALLDIR)/share/doc/direwolf/telem-volts.conf -# -# "man" pages -# - $(INSTALL) -D --mode=644 man1/aclients.1 $(INSTALLDIR)/man/man1/aclients.1 - $(INSTALL) -D --mode=644 man1/atest.1 $(INSTALLDIR)/man/man1/atest.1 - $(INSTALL) -D --mode=644 man1/decode_aprs.1 $(INSTALLDIR)/man/man1/decode_aprs.1 - $(INSTALL) -D --mode=644 man1/direwolf.1 $(INSTALLDIR)/man/man1/direwolf.1 - $(INSTALL) -D --mode=644 man1/gen_packets.1 $(INSTALLDIR)/man/man1/gen_packets.1 - $(INSTALL) -D --mode=644 man1/ll2utm.1 $(INSTALLDIR)/man/man1/ll2utm.1 - $(INSTALL) -D --mode=644 man1/log2gpx.1 $(INSTALLDIR)/man/man1/log2gpx.1 - $(INSTALL) -D --mode=644 man1/text2tt.1 $(INSTALLDIR)/man/man1/text2tt.1 - $(INSTALL) -D --mode=644 man1/tt2text.1 $(INSTALLDIR)/man/man1/tt2text.1 - $(INSTALL) -D --mode=644 man1/utm2ll.1 $(INSTALLDIR)/man/man1/utm2ll.1 -# - @echo " " - @echo "If this is your first install, not an upgrade, type this to put a copy" - @echo "of the sample configuration file (direwolf.conf) in your home directory:" - @echo " " - @echo " make install-conf" - @echo " " - - -# TODO: Should we put the sample direwolf.conf file somewhere like -# /usr/share/doc/direwolf/examples and add that to the -# end of the search path list? -# That would make it easy to see user customizations compared to the -# latest sample. - -# These would be done as ordinary user. - - -.PHONY: install-conf -install-conf : direwolf.conf - cp direwolf.conf ~ - cp telemetry-toolkit/telem-m0xer-3.txt ~ - cp telemetry-toolkit/telem-*.conf ~ - - -# Separate application to decode raw data. - -decode_aprs : decode_aprs.c dwgpsnmea.o dwgps.o dwgpsd.o serial_port.o symbols.o ax25_pad.o textcolor.o fcs_calc.o latlong.o log.o telemetry.o tt_text.o - $(CC) $(CFLAGS) -DDECAMAIN -o $@ $^ -lm - -# Convert between text and touch tone representation. - -text2tt : tt_text.c - $(CC) $(CFLAGS) -DENC_MAIN -o $@ $^ - -tt2text : tt_text.c - $(CC) $(CFLAGS) -DDEC_MAIN -o $@ $^ - - -# Convert between Latitude/Longitude and UTM coordinates. - -ll2utm : ll2utm.c geotranz.a - $(CC) $(CFLAGS) -o $@ $^ -lm - -utm2ll : utm2ll.c geotranz.a - $(CC) $(CFLAGS) -o $@ $^ -lm - - -# Convert from log file to GPX. - -log2gpx : log2gpx.c - $(CC) $(CFLAGS) -o $@ $^ -lm - - -# Test application to generate sound. - -gen_packets : gen_packets.c ax25_pad.c hdlc_send.c fcs_calc.c gen_tone.c morse.c dtmf.c textcolor.c dsp.c - $(CC) $(CFLAGS) -o $@ $^ $(LDLIBS) -lm - -demod.o : tune.h - -demod_afsk.o : tune.h - -demod_9600.o : tune.h - -demod_psk.o : tune.h - -tune.h : - echo " " > tune.h - - -testagc : atest.c demod.c dsp.c demod_afsk.c demod_9600.c hdlc_rec.c hdlc_rec2.o multi_modem.o rrbb.o \ - fcs_calc.c ax25_pad.c decode_aprs.c telemetry.c dtime_now.o latlong.c symbols.c tune.h textcolor.c - $(CC) $(CFLAGS) -o atest $^ -lm - ./atest 02_Track_2.wav | grep "packets decoded in" > atest.out - - -# Unit test for demodulators - -atest : atest.c demod.c dsp.c demod_afsk.c demod_psk.c demod_9600.c hdlc_rec.c hdlc_rec2.o multi_modem.o rrbb.o \ - fcs_calc.c ax25_pad.c decode_aprs.c dwgpsnmea.o dwgps.o serial_port.o telemetry.c dtest_now.o latlong.c symbols.c textcolor.c tt_text.c - $(CC) $(CFLAGS) -o $@ $^ -lm -#atest : atest.c fsk_fast_filter.h demod.c dsp.c demod_afsk.c demod_psk.c demod_9600.c hdlc_rec.c hdlc_rec2.o multi_modem.o rrbb.o \ -# fcs_calc.c ax25_pad.c decode_aprs.c dwgpsnmea.o dwgps.o serial_port.o telemetry.c latlong.c symbols.c textcolor.c tt_text.c -# $(CC) $(CFLAGS) -o $@ $^ -lm - -# Unit test for inner digipeater algorithm - - -dtest : digipeater.c pfilter.o ax25_pad.o dedupe.o fcs_calc.o tq.o textcolor.o \ - decode_aprs.o dwgpsnmea.o dwgps.o serial_port.o latlong.o telemetry.o symbols.o tt_text.o - $(CC) $(CFLAGS) -DTEST -o $@ $^ - ./dtest - - -# Unit test for APRStt. - -ttest : aprs_tt.c tt_text.c latlong.c geotranz.a - $(CC) $(CFLAGS) -DTT_MAIN -o $@ $^ - - -# Unit test for IGate - - -itest : igate.c textcolor.c ax25_pad.c fcs_calc.c - $(CC) $(CFLAGS) -DITEST -o $@ $^ - ./itest - - -# Unit test for UDP reception with AFSK demodulator - -udptest : udp_test.c demod.c dsp.c demod_afsk.c demod_9600.c hdlc_rec.c hdlc_rec2.c multi_modem.c rrbb.c fcs_calc.c ax25_pad.c decode_aprs.c symbols.c textcolor.c - $(CC) $(CFLAGS) -o $@ $^ -lm - ./udptest - - -# Unit test for telemetry decoding. - - -tlmtest : telemetry.c ax25_pad.c fcs_calc.c textcolor.c - $(CC) $(CFLAGS) -o $@ $^ -lm - ./tlmtest - - -# Multiple AGWPE network or serial port clients to test TNCs side by side. - -aclients : aclients.c ax25_pad.c fcs_calc.c textcolor.c - $(CC) $(CFLAGS) -g -o $@ $^ - - -# Touch Tone to Speech sample application. - -ttcalc : ttcalc.o ax25_pad.o fcs_calc.o textcolor.o - $(CC) $(CFLAGS) -g -o $@ $^ - - -depend : $(wildcard *.c) - makedepend -f $(lastword $(MAKEFILE_LIST)) -- $(CFLAGS) -- $^ - - -.PHONY: clean -clean : - rm -f $(APPS) gen_fff \ - fsk_fast_filter.h *.o *.a use_this_sdk - echo " " > tune.h - - -.PHONY: dist-mac -dist-mac: direwolf decode_aprs text2tt tt2text ll2utm utm2ll aclients log2gpx gen_packets \ - tocalls.txt symbols-new.txt symbolsX.txt dw-icon.png - rm -f ../direwolf_dist_bin.zip - (cd .. ; zip direwolf_dist_bin.zip \ - $(INSTALLDIR)/bin/direwolf \ - $(INSTALLDIR)/bin/decode_aprs \ - $(INSTALLDIR)/bin/text2tt \ - $(INSTALLDIR)/bin/tt2text \ - $(INSTALLDIR)/bin/ll2utm \ - $(INSTALLDIR)/bin/utm2ll \ - $(INSTALLDIR)/bin/aclients \ - $(INSTALLDIR)/bin/log2gpx \ - $(INSTALLDIR)/bin/gen_packets \ - $(INSTALLDIR)/bin/atest \ - $(INSTALLDIR)/bin/ttcalc \ - $(INSTALLDIR)/bin/dwespeak.sh \ - $(INSTALLDIR)/share/direwolf/tocalls.txt \ - $(INSTALLDIR)/share/direwolf/config/direwolf.conf \ - $(INSTALLDIR)/share/direwolf/symbols-new.txt \ - $(INSTALLDIR)/share/direwolf/symbolsX.txt \ - $(INSTALLDIR)/share/direwolf/dw-icon.png \ - $(INSTALLDIR)/share/doc/direwolf/README.md \ - $(INSTALLDIR)/share/doc/direwolf/CHANGES.md \ - $(INSTALLDIR)/share/doc/direwolf/LICENSE-dire-wolf.txt \ - $(INSTALLDIR)/share/doc/direwolf/LICENSE-other.txt \ - $(INSTALLDIR)/share/doc/direwolf/User-Guide.pdf \ - $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-APRS.pdf \ - $(INSTALLDIR)/share/doc/direwolf/Raspberry-Pi-APRS-Tracker.pdf \ - $(INSTALLDIR)/share/doc/direwolf/APRStt-Implementation-Notes.pdf \ - $(INSTALLDIR)/share/doc/direwolf/APRS-Telemetry-Toolkit.pdf \ - $(INSTALLDIR)/man/man1/aclients.1 \ - $(INSTALLDIR)/man/man1/atest.1 \ - $(INSTALLDIR)/man/man1/decode_aprs.1 \ - $(INSTALLDIR)/man/man1/direwolf.1 \ - $(INSTALLDIR)/man/man1/gen_packets.1 \ - $(INSTALLDIR)/man/man1/ll2utm.1 \ - $(INSTALLDIR)/man/man1/log2gpx.1 \ - $(INSTALLDIR)/man/man1/text2tt.1 \ - $(INSTALLDIR)/man/man1/tt2text.1 \ - $(INSTALLDIR)/man/man1/utm2ll.1 \ - ) - -# Package it up for distribution. - -.PHONY: dist-src -dist-src : README.md CHANGES.md \ - doc/User-Guide.pdf doc/Raspberry-Pi-APRS.pdf \ - doc/Raspberry-Pi-APRS-Tracker.pdf doc/APRStt-Implementation-Notes.pdf \ - dw-start.sh dwespeak.bat dwespeak.sh \ - tocalls.txt symbols-new.txt symbolsX.txt direwolf.spec - rm -f fsk_fast_filter.h - echo " " > tune.h - rm -f ../$z-src.zip - (cd .. ; zip $z-src.zip \ - $z/README.md \ - $z/CHANGES.md \ - $z/LICENSE* \ - $z/doc/User-Guide.pdf \ - $z/doc/Raspberry-Pi-APRS.pdf \ - $z/doc/Raspberry-Pi-APRS-Tracker.pdf \ - $z/doc/APRStt-Implementation-Notes.pdf \ - $z/Makefile* \ - $z/*.c $z/*.h \ - $z/regex/* $z/misc/* $z/geotranz/* \ - $z/man1/* \ - $z/generic.conf \ - $z/tocalls.txt $z/symbols-new.txt $z/symbolsX.txt \ - $z/dw-icon.png $z/dw-icon.rc $z/dw-icon.ico \ - $z/dw-start.sh $z/direwolf.spec \ - $z/dwespeak.bat $z/dwespeak.sh \ - $z/telemetry-toolkit/* ) - - -# -# The destination field is often used to identify the manufacturer/model. -# These are not hardcoded into Dire Wolf. Instead they are read from -# a file called tocalls.txt at application start up time. -# -# The original permanent symbols are built in but the "new" symbols, -# using overlays, are often updated. These are also read from files. -# -# You can obtain an updated copy by typing "make tocalls-symbols". -# This is not part of the normal build process. You have to do this explicitly. -# -# The locations below appear to be the most recent. -# The copy at http://www.aprs.org/tocalls.txt is out of date. -# - -.PHONY: tocalls-symbols -tocalls-symbols : - cp tocalls.txt tocalls.txt~ - wget http://www.aprs.org/aprs11/tocalls.txt -O tocalls.txt - -diff -Z tocalls.txt~ tocalls.txt - cp symbols-new.txt symbols-new.txt~ - wget http://www.aprs.org/symbols/symbols-new.txt -O symbols-new.txt - -diff -Z symbols-new.txt~ symbols-new.txt - cp symbolsX.txt symbolsX.txt~ - wget http://www.aprs.org/symbols/symbolsX.txt -O symbolsX.txt - -diff -Z symbolsX.txt~ symbolsX.txt diff --git a/Makefile.win b/Makefile.win deleted file mode 100644 index 3e1cefc1..00000000 --- a/Makefile.win +++ /dev/null @@ -1,650 +0,0 @@ -# -# Makefile for native Windows version of Dire Wolf. -# -# -# This is built in the Cygwin environment but with the -# compiler from http://www.mingw.org/ so there is no -# dependency on extra DLLs. -# -# The MinGW/bin directory must be in the PATH for the -# compiler. e.g. export PATH=/cygdrive/c/MinGW/bin:$PATH -# -# Failure to have the path set correctly will result in the -# obscure message: Makefile.win:... recipe for target ... failed. -# -# Type "which gcc" to make sure you are getting the right one! -# - - -all : direwolf decode_aprs text2tt tt2text ll2utm utm2ll aclients log2gpx gen_packets atest ttcalc tnctest - - -# People say we need -mthreads option for threads to work properly. -# They also say it creates a dependency on mingwm10.dll but I'm not seeing that. -# Maybe that is for pthreads. We are using the Windows threads. - -# -Ofast was added in gcc 4.6 which was the MinGW version back in 2012. - -CC := gcc -CFLAGS := -Ofast -march=pentium3 -msse -Iregex -Iutm -Igeotranz -mthreads -DUSE_REGEX_STATIC -Wall -Wlogical-op -AR := ar - -CFLAGS += -g - -# For version 1.4, we upgrade from gcc 4.6.2 to 4.9.3. - -# gcc 4.8 adds these. Try them just for fun. -# No, it needs libasan which is not on Windows. -#CFLAGS += -fsanitize=address -fno-omit-frame-pointer - -# Helpful for the demodulators. Overkill for non-hot spots. -#CFLAGS += -Wdouble-promotion - -# Don't have the patience for this right now. -#CFLAGS += -Wextra - -# Continue working on these. -CFLAGS += -Wsign-compare -CFLAGS += -Wuninitialized -CFLAGS += -Wold-style-declaration -# CFLAGS += -fdelete-null-pointer-checks -Wnull-dereference ---not recognized -#CFLAGS += -Wold-style-definition -#-Wmissing-prototypes - -# -# Let's see impact of various optimization levels. -# Benchmark results with MinGW gcc version 4.6.2. -# -# seconds options, comments -# ------ ----------------- -# 119.8 -O2 Used for version 0.8 -# 92.1 -O3 -# 88.7 -Ofast (should be same as -O3 -ffastmath) -# 87.5 -Ofast -march=pentium -# 74.1 -Ofast -msse -# 72.2 -Ofast -march=pentium -msse -# 62.0 -Ofast -march=pentium3 (this implies -msse) -# 61.9 -Ofast -march=pentium3 -msse -# -# A minimum of Windows XP is required due to some of the system -# features being used. XP requires a Pentium processor or later. -# The DSP filters can be sped up considerably with the SSE instructions. -# The SSE instructions were introduced in 1999 with the -# Pentium III series. -# SSE2 instructions, added in 2000, don't seem to offer any advantage. -# -# For version 0.9, a Pentium 3 or equivalent is now the minimum required -# for the prebuilt Windows distribution. -# If you insist on using a computer from the previous century, -# you can compile this yourself with different options. -# - - - - -# -------------------------------------- Main application -------------------------------- - -# Not sure why this is here. - -demod.o : fsk_demod_state.h - -demod_9600.o : fsk_demod_state.h - -demod_afsk.o : fsk_demod_state.h - -demod_psk.o : fsk_demod_state.h - - -direwolf : direwolf.o config.o recv.o demod.o dsp.o demod_afsk.o demod_psk.o demod_9600.o hdlc_rec.o \ - hdlc_rec2.o multi_modem.o rdq.o rrbb.o dlq.o \ - fcs_calc.o ax25_pad.o ax25_pad2.o xid.o \ - decode_aprs.o symbols.o server.o kiss.o kissnet.o kiss_frame.o hdlc_send.o fcs_calc.o \ - gen_tone.o morse.o audio_win.o audio_stats.o digipeater.o cdigipeater.o pfilter.o dedupe.o tq.o xmit.o \ - ptt.o beacon.o dwgps.o encode_aprs.o latlong.o textcolor.o \ - dtmf.o aprs_tt.o tt_user.o tt_text.o igate.o waypoint.o serial_port.o log.o telemetry.o \ - dwgps.o dwgpsnmea.o dtime_now.o mheard.o ax25_link.o \ - dw-icon.o regex.a misc.a geotranz.a - $(CC) $(CFLAGS) -o $@ $^ -lwinmm -lws2_32 - -dw-icon.o : dw-icon.rc dw-icon.ico - windres dw-icon.rc -o $@ - - -# Optimization for slow processors. - -demod.o : fsk_fast_filter.h - -demod_afsk.o : fsk_fast_filter.h - - -fsk_fast_filter.h : gen_fff - ./gen_fff > fsk_fast_filter.h - -gen_fff : demod_afsk.c dsp.c textcolor.c - echo " " > tune.h - $(CC) $(CFLAGS) -DGEN_FFF -o $@ $^ - - -# -# The destination field is often used to identify the manufacturer/model. -# These are not hardcoded into Dire Wolf. Instead they are read from -# a file called tocalls.txt at application start up time. -# -# The original permanent symbols are built in but the "new" symbols, -# using overlays, are often updated. These are also read from files. -# -# You can obtain an updated copy by typing "make tocalls-symbols". -# This is not part of the normal build process. You have to do this explicitly. -# -# The locations below appear to be the most recent. -# The copy at http://www.aprs.org/tocalls.txt is out of date. -# - -.PHONY: tocalls-symbols -tocalls-symbols : - cp tocalls.txt tocalls.txt~ - wget http://www.aprs.org/aprs11/tocalls.txt -O tocalls.txt - -diff tocalls.txt~ tocalls.txt - cp symbols-new.txt symbols-new.txt~ - wget http://www.aprs.org/symbols/symbols-new.txt -O symbols-new.txt - -diff symbols-new.txt~ symbols-new.txt - cp symbolsX.txt symbolsX.txt~ - wget http://www.aprs.org/symbols/symbolsX.txt -O symbolsX.txt - -diff symbolsX.txt~ symbolsX.txt - - - -# ---------------------------- Other utilities included with distribution ------------------------- - - -# Separate application to decode raw data. - -decode_aprs : decode_aprs.c dwgpsnmea.o dwgps.o serial_port.o symbols.o ax25_pad.o textcolor.o fcs_calc.o latlong.o log.o telemetry.o tt_text.c regex.a misc.a geotranz.a - $(CC) $(CFLAGS) -DDECAMAIN -o decode_aprs $^ - - -# Convert between text and touch tone representation. - -text2tt : tt_text.c misc.a - $(CC) $(CFLAGS) -DENC_MAIN -o $@ $^ - -tt2text : tt_text.c misc.a - $(CC) $(CFLAGS) -DDEC_MAIN -o $@ $^ - - -# Convert between Latitude/Longitude and UTM coordinates. - -ll2utm : ll2utm.c textcolor.c geotranz.a misc.a - $(CC) $(CFLAGS) -o $@ $^ - -utm2ll : utm2ll.c textcolor.c geotranz.a misc.a - $(CC) $(CFLAGS) -o $@ $^ - - -# Convert from log file to GPX. - -log2gpx : log2gpx.c textcolor.o misc.a - $(CC) $(CFLAGS) -o $@ $^ - - -# Test application to generate sound. - -gen_packets : gen_packets.o ax25_pad.o hdlc_send.o fcs_calc.o gen_tone.o morse.o dtmf.o textcolor.o dsp.o misc.a regex.a - $(CC) $(CFLAGS) -o $@ $^ - - - -# ------------------------------------------- Libraries -------------------------------------------- - - - -# UTM, USNG, MGRS conversions. - -geotranz.a : error_string.o mgrs.o polarst.o tranmerc.o ups.o usng.o utm.o - ar -cr $@ $^ - -error_string.o : geotranz/error_string.c - $(CC) $(CFLAGS) -c -o $@ $^ - -mgrs.o : geotranz/mgrs.c - $(CC) $(CFLAGS) -c -o $@ $^ - -polarst.o : geotranz/polarst.c - $(CC) $(CFLAGS) -c -o $@ $^ - -tranmerc.o : geotranz/tranmerc.c - $(CC) $(CFLAGS) -c -o $@ $^ - -ups.o : geotranz/ups.c - $(CC) $(CFLAGS) -c -o $@ $^ - -usng.o : geotranz/usng.c - $(CC) $(CFLAGS) -c -o $@ $^ - -utm.o : geotranz/utm.c - $(CC) $(CFLAGS) -c -o $@ $^ - - -# -# When building for Linux, we use regular expression -# functions supplied by the gnu C library. -# For the native WIN32 version, we need to use our own copy. -# These were copied from http://gnuwin32.sourceforge.net/packages/regex.htm -# Consider upgrading from https://www.gnu.org/software/libc/sources.html - -regex.a : regex.o - ar -cr $@ $^ - -regex.o : regex/regex.c - $(CC) $(CFLAGS) -Dbool=int -Dtrue=1 -Dfalse=0 -c -o $@ $^ - - - -# There are several string functions found in Linux -# but not on Windows. Need to provide our own copy. - -misc.a : strsep.o strtok_r.o strcasestr.o strlcpy.o strlcat.o - ar -cr $@ $^ - -strsep.o : misc/strsep.c - $(CC) $(CFLAGS) -c -o $@ $^ - -strtok_r.o : misc/strtok_r.c - $(CC) $(CFLAGS) -c -o $@ $^ - -strcasestr.o : misc/strcasestr.c - $(CC) $(CFLAGS) -c -o $@ $^ - -strlcpy.o : misc/strlcpy.c - $(CC) $(CFLAGS) -I. -c -o $@ $^ - -strlcat.o : misc/strlcat.c - $(CC) $(CFLAGS) -I. -c -o $@ $^ - - -# --------------------------------- Automated Smoke Test -------------------------------- - - -# Combine some unit tests into a single regression sanity check. - - -check : dtest ttest tttexttest pftest tlmtest lltest enctest kisstest pad2test xidtest dtmftest check-modem1200 check-modem300 check-modem9600 check-modem19200 check-modem2400 check-modem4800 - -# Can we encode and decode at popular data rates? -# Verify that single bit fixup increases the count. - -check-modem1200 : gen_packets atest - gen_packets -n 100 -o test12.wav - atest -F0 -PE -L64 -G72 test12.wav - atest -F1 -PE -L70 -G75 test12.wav - rm test12.wav - -check-modem300 : gen_packets atest - gen_packets -B300 -n 100 -o test3.wav - atest -B300 -F0 -L68 -G69 test3.wav - atest -B300 -F1 -L71 -G75 test3.wav - rm test3.wav - -#FIXME: test full amplitude. - -check-modem9600 : gen_packets atest - gen_packets -B9600 -a 170 -o test96.wav - sleep 1 - atest -B9600 -F0 -L4 -G4 test96.wav - sleep 1 - rm test96.wav - sleep 1 - gen_packets -B9600 -n 100 -o test96.wav - sleep 1 - atest -B9600 -F0 -L50 -G54 test96.wav - atest -B9600 -F1 -L55 -G59 test96.wav - sleep 1 - rm test96.wav - -check-modem19200 : gen_packets atest - gen_packets -r 96000 -B19200 -a 170 -o test19.wav - sleep 1 - atest -B19200 -F0 -L4 test19.wav - sleep 1 - rm test19.wav - sleep 1 - gen_packets -r 96000 -B19200 -n 100 -o test19.wav - sleep 1 - atest -B19200 -F0 -L55 -G59 test19.wav - atest -B19200 -F1 -L60 -G64 test19.wav - sleep 1 - rm test19.wav - -check-modem2400 : gen_packets atest - gen_packets -B2400 -n 100 -o test24.wav - sleep 1 - atest -B2400 -F0 -L70 -G78 test24.wav - atest -B2400 -F1 -L80 -G87 test24.wav - sleep 1 - rm test24.wav - -check-modem4800 : gen_packets atest - gen_packets -B4800 -n 100 -o test48.wav - sleep 1 - atest -B4800 -F0 -L70 -G74 test48.wav - atest -B4800 -F1 -L79 -G84 test48.wav - sleep 1 - rm test48.wav - - -# Unit test for demodulators - -atest : atest.c fsk_fast_filter.h demod.c demod_afsk.c demod_psk.c demod_9600.c \ - dsp.o hdlc_rec.o hdlc_rec2.o multi_modem.o \ - rrbb.o fcs_calc.o ax25_pad.o decode_aprs.o \ - dwgpsnmea.o dwgps.o serial_port.o latlong.c \ - symbols.c tt_text.c textcolor.c telemetry.c dtime_now.o \ - decode_aprs.o log.o \ - misc.a regex.a - echo " " > tune.h - $(CC) $(CFLAGS) -o $@ $^ - #./atest ..\\direwolf-0.2\\02_Track_2.wav - #atest -B 9600 z9.wav - #atest za100.wav - -atest9 : atest.c demod.c dsp.c demod_afsk.c demod_psk.c demod_9600.c hdlc_rec.c hdlc_rec2.c multi_modem.c \ - rrbb.c fcs_calc.c ax25_pad.c decode_aprs.c latlong.c symbols.c textcolor.c telemetry.c dtime_now.o misc.a regex.a \ - fsk_fast_filter.h - echo " " > tune.h - $(CC) $(CFLAGS) -o $@ $^ - ./atest9 -B 9600 ../walkabout9600.wav | grep "packets decoded in" >atest.out - #./atest9 -B 9600 noise96.wav - - -# Unit test for inner digipeater algorithm - -.PHONY: dtest -dtest : digipeater.c dedupe.c pfilter.c \ - ax25_pad.o fcs_calc.o tq.o textcolor.o \ - decode_aprs.o dwgpsnmea.o dwgps.o serial_port.o latlong.o telemetry.o symbols.o tt_text.o misc.a regex.a - $(CC) $(CFLAGS) -DDIGITEST -o $@ $^ - ./dtest - rm dtest.exe - -# Unit test for APRStt tone seqence parsing. - -.PHONTY: ttest -ttest : aprs_tt.c tt_text.c latlong.o textcolor.o geotranz.a misc.a - $(CC) $(CFLAGS) -Igeotranz -DTT_MAIN -o $@ $^ - ./ttest - rm ttest.exe - -# Unit test for APRStt tone sequence / text conversions. - -.PHONY: tttexttest -tttexttest : tt_text.c textcolor.o misc.a - $(CC) $(CFLAGS) -DTTT_TEST -o $@ $^ - ./tttexttest - rm tttexttest.exe - -# Unit test for Packet Filtering. - -.PHONY: pftest -pftest : pfilter.c ax25_pad.o textcolor.o fcs_calc.o decode_aprs.o dwgpsnmea.o dwgps.o serial_port.o latlong.o symbols.o telemetry.o tt_text.o misc.a regex.a - $(CC) $(CFLAGS) -DPFTEST -o $@ $^ - ./pftest - rm pftest.exe - - - -# Unit test for telemetry decoding. - -.PHONY: tlmtest -tlmtest : telemetry.c ax25_pad.o fcs_calc.o textcolor.o misc.a regex.a - $(CC) $(CFLAGS) -DTEST -o $@ $^ - ./tlmtest - rm tlmtest.exe - - -# Unit test for location coordinate conversion. - -.PHONY: lltest -lltest : latlong.c textcolor.o misc.a - $(CC) $(CFLAGS) -DLLTEST -o $@ $^ - ./lltest - rm lltest.exe - -# Unit test for encoding position & object report. - -.PHONY: enctest -enctest : encode_aprs.c latlong.c textcolor.c misc.a - $(CC) $(CFLAGS) -DEN_MAIN -o $@ $^ - ./enctest - rm enctest.exe - - -# Unit test for KISS encapsulation. - -.PHONY: kisstest -kisstest : kiss_frame.c - $(CC) $(CFLAGS) -DKISSTEST -o $@ $^ - ./kisstest - rm kisstest.exe - - -# Unit test for constructing frames besides UI. - -.PHONY: pad2test -pad2test : ax25_pad2.c ax25_pad.c fcs_calc.o textcolor.o regex.a misc.a - $(CC) $(CFLAGS) -DPAD2TEST -o $@ $^ - ./pad2test - rm pad2test.exe - -# Unit Test for XID frame encode/decode. - -.PHONY: xidtest -xidtest : xid.c textcolor.o misc.a - $(CC) $(CFLAGS) -DXIDTEST -o $@ $^ - ./xidtest - rm xidtest.exe - -# Unit Test for DTMF encode/decode. - -.PHONY: dtmftest -dtmftest : dtmf.c textcolor.o - $(CC) $(CFLAGS) -DDTMF_TEST -o $@ $^ - ./dtmftest - rm dtmftest.exe - - -# ------------------------------ Other manual testing & experimenting ------------------------------- - - -tnctest : tnctest.c textcolor.o dtime_now.o serial_port.o misc.a - $(CC) $(CFLAGS) -o $@ $^ -lwinmm -lws2_32 - - -# For tweaking the demodulator. - -demod.o : tune.h -demod_9600.o : tune.h -demod_afsk.o : tune.h -demod_psk.o : tune.h - -testagc : atest.c demod.c dsp.c demod_afsk.c demod_psk.c demod_9600.o fsk_demod_agc.h \ - hdlc_rec.o hdlc_rec2.o multi_modem.o \ - rrbb.o fcs_calc.o ax25_pad.o decode_aprs.o latlong.o symbols.o textcolor.o telemetry.o \ - dwgpsnmea.o dwgps.o serial_port.o tt_text.o dtime_now.o regex.a misc.a - rm -f atest.exe - $(CC) $(CFLAGS) -o atest $^ - ./atest -P GGG- -F 0 ../02_Track_2.wav | grep "packets decoded in" >atest.out - echo " " > tune.h - - -noisy3.wav : gen_packets - ./gen_packets -B 300 -n 100 -o noisy3.wav - -testagc3 : atest.c demod.c dsp.c demod_afsk.c demod_psk.c demod_9600.c hdlc_rec.c hdlc_rec2.c multi_modem.c \ - rrbb.c fcs_calc.c ax25_pad.c decode_aprs.c latlong.c symbols.c textcolor.c telemetry.c dtime_now.o regex.a misc.a \ - tune.h - rm -f atest3.exe - $(CC) $(CFLAGS) -o atest3 $^ - ./atest3 -B 300 -P D -D 3 noisy3.wav | grep "packets decoded in" >atest.out - echo " " > tune.h - - -noisy96.wav : gen_packets - ./gen_packets -B 9600 -n 100 -o noisy96.wav - -testagc96 : atest.c fsk_fast_filter.h tune.h demod.c demod_afsk.c demod_psk.c demod_9600.c \ - dsp.o hdlc_rec.o hdlc_rec2.o multi_modem.o \ - rrbb.o fcs_calc.o ax25_pad.o decode_aprs.o \ - dwgpsnmea.o dwgps.o serial_port.o latlong.o \ - symbols.o tt_text.o textcolor.o telemetry.o dtime_now.o \ - misc.a regex.a - rm -f atest96.exe - $(CC) $(CFLAGS) -o atest96 $^ - ./atest96 -B 9600 ../walkabout9600c.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 noisy96.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 19990303_0225_9600_8bis_22kHz.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 19990303_0225_9600_16bit_22kHz.wav | grep "packets decoded in" >atest.out - #./atest96 -B 9600 -P + z8-22k.wav| grep "packets decoded in" >atest.out - #./atest96 -B 9600 test9600.wav | grep "packets decoded in" >atest.out - echo " " > tune.h - -testagc24 : atest.c fsk_fast_filter.h tune.h demod.c demod_afsk.c demod_psk.c demod_9600.c \ - dsp.o hdlc_rec.o hdlc_rec2.o multi_modem.o \ - rrbb.o fcs_calc.o ax25_pad.o decode_aprs.o \ - dwgpsnmea.o dwgps.o serial_port.o latlong.o \ - symbols.o tt_text.o textcolor.o telemetry.o dtime_now.o \ - misc.a regex.a - rm -f atest24.exe - sleep 1 - $(CC) $(CFLAGS) -o atest24 $^ - ./atest24 -B 2400 test2400.wav | grep "packets decoded in" >atest.out - echo " " > tune.h - -testagc48 : atest.c fsk_fast_filter.h tune.h demod.c demod_afsk.c demod_psk.c demod_9600.c \ - dsp.o hdlc_rec.o hdlc_rec2.o multi_modem.o \ - rrbb.o fcs_calc.o ax25_pad.o decode_aprs.o \ - dwgpsnmea.o dwgps.o serial_port.o latlong.o \ - symbols.o tt_text.o textcolor.o telemetry.o dtime_now.o \ - misc.a regex.a - rm -f atest48.exe - sleep 1 - $(CC) $(CFLAGS) -o atest48 $^ - ./atest48 -B 4800 test4800.wav | grep "packets decoded in" >atest.out - #./atest48 -B 4800 test4800.wav - echo " " > tune.h - - -# Unit test for IGate - -itest : igate.c textcolor.c ax25_pad.c fcs_calc.c misc.a regex.a - $(CC) $(CFLAGS) -DITEST -o $@ $^ -lwinmm -lws2_32 - - - - - -# Multiple AGWPE network or serial port clients to test TNCs side by side. - -aclients : aclients.c ax25_pad.c fcs_calc.c textcolor.c misc.a regex.a - $(CC) $(CFLAGS) -o $@ $^ -lwinmm -lws2_32 - - -# Touch Tone to Speech sample application. - -ttcalc : ttcalc.o ax25_pad.o fcs_calc.o textcolor.o misc.a regex.a - $(CC) $(CFLAGS) -o $@ $^ -lwinmm -lws2_32 - - -# Send GPS location to KISS TNC each second. - -walk96 : walk96.c dwgps.o dwgpsnmea.o kiss_frame.o \ - latlong.o encode_aprs.o serial_port.o textcolor.o \ - ax25_pad.o fcs_calc.o \ - xmit.o hdlc_send.o gen_tone.o ptt.o tq.o \ - hdlc_rec.o hdlc_rec2.o rrbb.o dsp.o audio_win.o \ - multi_modem.o demod.o demod_afsk.o demod_psk.c demod_9600.o rdq.o \ - server.o morse.o dtmf.o audio_stats.o dtime_now.o dlq.o \ - regex.a misc.a - $(CC) $(CFLAGS) -DWALK96 -o $@ $^ -lwinmm -lws2_32 - - - -#-------------------------------------------------------------- - - -.PHONY: depend -depend : $(wildcard *.c) - makedepend -f $(lastword $(MAKEFILE_LIST)) -- $(CFLAGS) -- $^ - -.PHONY: clean -clean : - rm -f *.o *.a *.exe fsk_fast_filter.h noisy96.wav - echo " " > tune.h - - - -# ------------------------------- Packaging for distribution ---------------------- - -# Name of zip file for distribution. - -z := $(notdir ${CURDIR}) - - -.PHONY: dist-win -dist-win : direwolf.exe decode_aprs.exe text2tt.exe tt2text.exe ll2utm.exe utm2ll.exe \ - aclients.exe log2gpx.exe gen_packets.exe atest.exe ttcalc.exe \ - generic.conf dwespeak.bat \ - README.md CHANGES.md \ - doc/User-Guide.pdf \ - doc/Raspberry-Pi-APRS.pdf \ - doc/APRStt-Implementation-Notes.pdf - rm -f ../$z-win.zip - egrep '^C|^W' generic.conf | cut -c2-999 > direwolf.conf - unix2dos direwolf.conf - zip --junk-paths ../$z-win.zip \ - README.md \ - CHANGES.md \ - doc/User-Guide.pdf \ - doc/Raspberry-Pi-APRS.pdf \ - doc/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf \ - doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf \ - doc/APRS-Telemetry-Toolkit.pdf \ - doc/APRStt-Implementation-Notes.pdf \ - doc/APRStt-interface-for-SARTrack.pdf \ - doc/APRStt-Listening-Example.pdf \ - doc/Raspberry-Pi-APRS.pdf \ - doc/Raspberry-Pi-APRS-Tracker.pdf \ - doc/Raspberry-Pi-SDR-IGate.pdf \ - doc/User-Guide.pdf \ - doc/WA8LMF-TNC-Test-CD-Results.pdf \ - LICENSE* \ - direwolf.conf \ - direwolf.exe \ - decode_aprs.exe \ - tocalls.txt symbols-new.txt symbolsX.txt \ - text2tt.exe tt2text.exe \ - ll2utm.exe utm2ll.exe \ - aclients.exe \ - log2gpx.exe \ - gen_packets.exe \ - atest.exe \ - ttcalc.exe \ - dwespeak.bat \ - telemetry-toolkit/* - - -# Reminders if pdf files are not up to date. - - - - -.PHONY: backup -backup : - mkdir /cygdrive/e/backup-cygwin-home/`date +"%Y-%m-%d"` - cp -r . /cygdrive/e/backup-cygwin-home/`date +"%Y-%m-%d"` - - -# -# The following is updated by "make depend" -# -# DO NOT DELETE - diff --git a/README.md b/README.md index 7d91dd3f..3006a1ef 100644 --- a/README.md +++ b/README.md @@ -3,60 +3,127 @@ ### Decoded Information from Radio Emissions for Windows Or Linux Fans ### -In the early days of Amateur Packet Radio, it was necessary to use an expensive “Terminal Node Controller” (TNC) with specialized hardware. Those days are gone. You can now get better results at lower cost by connecting your radio to the “soundcard” interface of a computer and using software to decode the signals. - -Dire Wolf is a software "soundcard" modem/TNC and [APRS](http://www.aprs.org/) encoder/decoder. It can be used stand-alone to observe APRS traffic, as a digipeater, [APRStt](http://www.aprs.org/aprstt.html) gateway, or Internet Gateway (IGate). It can also be used as a virtual TNC for other applications such as [APRSIS32](http://aprsisce.wikidot.com/), [UI-View32](http://www.ui-view.net/), [Xastir](http://xastir.org/index.php/Main_Page), [APRS-TW](http://aprstw.blandranch.net/), [YAAC](http://www.ka2ddo.org/ka2ddo/YAAC.html), [UISS](http://users.belgacom.net/hamradio/uiss.htm), [Linux AX25](http://www.linux-ax25.org/wiki/Main_Page), [SARTrack](http://www.sartrack.co.nz/index.html), [RMS Express](http://www.winlink.org/RMSExpress), [BPQ32](http://www.cantab.net/users/john.wiseman/Documents/BPQ32.html), [Outpost PM](http://www.outpostpm.org/) and many others. +In the early days of Amateur Packet Radio, it was necessary to use an expensive "Terminal Node Controller" (TNC) with specialized hardware. Those days are gone. You can now get better results at lower cost by connecting your radio to the "soundcard" interface of a computer and using software to decode the signals. + +Why waste $200 and settle for mediocre receive performance from a 1980's technology TNC using an old modem chip? Dire Wolf decodes over 1000 error-free frames from Track 2 of the [WA8LMF TNC Test CD](https://github.com/wb2osz/direwolf/tree/dev/doc/WA8LMF-TNC-Test-CD-Results.pdf), leaving all the hardware TNCs, and first generation "soundcard" modems, behind in the dust. + +![](tnc-test-cd-results.png) + +Dire Wolf includes [FX.25](https://en.wikipedia.org/wiki/FX.25_Forward_Error_Correction) which adds Forward Error Correction (FEC) in a way that is completely compatible with existing systems. If both ends are capable of FX.25, your information will continue to get through under conditions where regular AX.25 is completely useless. This was originally developed for satellites and is now seeing widespread use on HF. + +![](fx25.png) + +Version 1.7 adds [IL2P](https://en.wikipedia.org/wiki/Improved_Layer_2_Protocol), a different method of FEC with less overhead but it is not compatible with AX.25. + + + +### Dire Wolf is a modern software replacement for the old 1980's style TNC built with special hardware. ### + +Without any additional software, it can perform as: + + - APRS GPS Tracker + - Digipeater + - Internet Gateway (IGate) +- [APRStt](http://www.aprs.org/aprstt.html) gateway + + +It can also be used as a virtual TNC for other applications such as [APRSIS32](http://aprsisce.wikidot.com/), [Xastir](http://xastir.org/index.php/Main_Page), [APRS-TW](http://aprstw.blandranch.net/), [YAAC](http://www.ka2ddo.org/ka2ddo/YAAC.html), [PinPoint APRS](http://www.pinpointaprs.com/), [UI-View32](http://www.ui-view.net/),[UISS](http://users.belgacom.net/hamradio/uiss.htm), [Linux AX25](http://www.linux-ax25.org/wiki/Main_Page), [SARTrack](http://www.sartrack.co.nz/index.html), [Winlink Express (formerly known as RMS Express, formerly known as Winlink 2000 or WL2K)](http://www.winlink.org/RMSExpress), [BPQ32](http://www.cantab.net/users/john.wiseman/Documents/BPQ32.html), [Outpost PM](http://www.outpostpm.org/), [Ham Radio of Things](https://github.com/wb2osz/hrot), [Packet Compressed Sensing Imaging (PCSI)](https://maqifrnswa.github.io/PCSI/), and many others. ## Features & Benefits ## -- Lower cost, higher performance alternative to hardware TNC. -Decodes more than 1000 error-free frames from [WA8LMF TNC Test CD](http://wa8lmf.net/TNCtest/). +![](direwolf-block-diagram.png) + +### Dire Wolf includes: ### + + + +- **Beaconing, Tracker, Telemetry Toolkit.** + + Send periodic beacons to provide information to others. For tracking the location is provided by a GPS receiver. + Build your own telemetry applications with the toolkit. + + +- **APRStt Gateway.** + + Very few hams have portable equipment for APRS but nearly everyone has a handheld radio that can send DTMF tones. APRStt allows a user, equipped with only DTMF (commonly known as Touch Tone) generation capability, to enter information into the global APRS data network. Responses can be sent by Morse Code or synthesized speech. + +- **Digipeaters for APRS and traditional Packet Radio.** + + Extend the range of other stations by re-transmitting their signals. Unmatched flexibility for cross band repeating and filtering to limit what is retransmitted. + +- **Internet Gateway (IGate).** + + IGate stations allow communication between disjoint radio networks by allowing some content to flow between them over the Internet. + +- **Ham Radio of Things (HRoT).** + + There have been occasional mentions of merging Ham Radio with the Internet of Things but only ad hoc incompatible narrowly focused applications. Here is a proposal for a standardized more flexible method so different systems can communicate with each other. + + [Ham Radio of Things - IoT over Ham Radio](https://github.com/wb2osz/hrot) + +- **AX.25 v2.2 Link Layer.** + + Traditional connected mode packet radio where the TNC automatically retries transmissions and delivers data in the right order. -- Ideal for building a Raspberry Pi digipeater & IGate. +- **KISS Interface (TCP/IP, serial port, Bluetooth) & AGW network Interface (TCP/IP).** -- Data rates: 300 AFSK, 1200 AFSK, 2400 QPSK, 4800 8PSK, and 9600/19200/38400 bps K9NG/G3RUH. + Dire Wolf can be used as a virtual TNC for applications such as [APRSIS32](http://aprsisce.wikidot.com/), [Xastir](http://xastir.org/index.php/Main_Page), [APRS-TW](http://aprstw.blandranch.net/), [YAAC](http://www.ka2ddo.org/ka2ddo/YAAC.html), [PinPoint APRS](http://www.pinpointaprs.com/), [UI-View32](http://www.ui-view.net/),[UISS](http://users.belgacom.net/hamradio/uiss.htm), [Linux AX25](http://www.linux-ax25.org/wiki/Main_Page), [SARTrack](http://www.sartrack.co.nz/index.html), [Winlink Express (formerly known as RMS Express, formerly known as Winlink 2000 or WL2K)](http://www.winlink.org/RMSExpress), [BPQ32](http://www.cantab.net/users/john.wiseman/Documents/BPQ32.html), [Outpost PM](http://www.outpostpm.org/), [Ham Radio of Things](https://github.com/wb2osz/hrot), [Packet Compressed Sensing Imaging (PCSI)](https://maqifrnswa.github.io/PCSI/), and many others. -- Interface with applications by - - [AGW](http://uz7.ho.ua/includes/agwpeapi.htm) network protocol - - [KISS](http://www.ax25.net/kiss.aspx) serial port - - [KISS](http://www.ax25.net/kiss.aspx) TCP network protocol - -- Decoding of received information for troubleshooting. +### Radio Interfaces: ### -- Conversion from APRS to waypoint sentences in popular formats: $GPWPL, $PGRMW, $PMGNWPL, $PKWDWPL. +- **Uses computer's "soundcard" and digital signal processing.** -- Logging and conversion to GPX file format. + Lower cost and better performance than specialized hardware. -- Beaconing for yourself or other nearby entities. + Compatible interfaces include [DRAWS](http://nwdigitalradio.com/draws/), [UDRC](https://nw-digital-radio.groups.io/g/udrc/wiki/UDRC%E2%84%A2-and-Direwolf-Packet-Modem), [SignaLink USB](http://www.tigertronics.com/slusbmain.htm), [DMK URI](http://www.dmkeng.com/URI_Order_Page.htm), [RB-USB RIM](http://www.repeater-builder.com/products/usb-rim-lite.html), [RA-35](http://www.masterscommunications.com/products/radio-adapter/ra35.html), [DINAH](https://hamprojects.info/dinah/), [SHARI](https://hamprojects.info/shari/), and many others. -- Very flexible Digipeating with routing and filtering between up to 6 ports. -- APRStt gateway - converts touch tone sequences to APRS objects and voice responses. -- APRS Internet Gateway (IGate) with IPv6 support and special SATGate mode. +- **Modems:** -- APRS Telemetry Toolkit. + 300 bps AFSK for HF -- Compatible with software defined radios (SDR) such as [gqrx](http://gqrx.dk/), [rtl_fm](http://sdr.osmocom.org/trac/wiki/rtl-sdr), and SDR#. + 1200 bps AFSK most common for VHF/UHF -- Includes separate raw packet decoder, decode_aprs. + 2400 & 4800 bps PSK -- AX.25 v2.2 connected mode. (New in version 1.4.) + 9600 bps GMSK/G3RUH + + AIS reception + + EAS SAME reception + + + +- **DTMF ("Touch Tone") Decoding and Encoding.** + +- **Speech Synthesizer interface & Morse code generator.** + + Transmit human understandable messages. + +- **Compatible with Software Defined Radios such as gqrx, rtl_fm, and SDR#.** + +- **Concurrent operation with up to 3 soundcards and 6 radios.** + +### Portable & Open Source: ### + +- **Runs on Windows, Linux (PC/laptop, Raspberry Pi, etc.), Mac OSX.** -- Open source so you can see how it works and make your own modifications. -- Runs in 3 different environments: - - Microsoft Windows XP or later - - Linux, regular PC/laptop or single board computer such as Raspberry Pi, BeagleBone Black, cubieboard 2, or C.H.I.P. - - Mac OS X ## Documentation ## + [Stable Version](https://github.com/wb2osz/direwolf/tree/master/doc) -[Latest Development Version](https://github.com/wb2osz/direwolf/tree/dev/doc) +[Latest Development Version ("dev" branch)](https://github.com/wb2osz/direwolf/tree/dev/doc) +[Additional Topics](https://github.com/wb2osz/direwolf-doc) + +[Power Point presentations](https://github.com/wb2osz/direwolf-presentation) -- Why not give a talk at a local club meeting? + +Youtube has many interesting and helpful videos. Searching for [direwolf tnc](https://www.youtube.com/results?search_query=direwolf+tnc) or [direwolf aprs](https://www.youtube.com/results?search_query=direwolf+aprs) will produce the most relevant results. ## Installation ## @@ -64,62 +131,103 @@ Decodes more than 1000 error-free frames from [WA8LMF TNC Test CD](http://wa8lmf Go to the [**releases** page](https://github.com/wb2osz/direwolf/releases). Download a zip file with "win" in its name, unzip it, and run direwolf.exe from a command window. -For more details see the **User Guide** in the [**doc** directory](https://github.com/wb2osz/direwolf/tree/master/doc). +You can also build it yourself from source. For more details see the **User Guide** in the [**doc** directory](https://github.com/wb2osz/direwolf/tree/master/doc). ### Linux - Using git clone (recommended) ### +***Note that this has changed for version 1.6. There are now a couple extra steps.*** + + +First you will need to install some software development packages using different commands depending on your flavor of Linux. +In most cases, the first few will already be there and the package installer will tell you that installation is not necessary. + +On Debian / Ubuntu / Raspbian / Raspberry Pi OS: + + sudo apt-get install git + sudo apt-get install gcc + sudo apt-get install g++ + sudo apt-get install make + sudo apt-get install cmake + sudo apt-get install libasound2-dev + sudo apt-get install libudev-dev + sudo apt-get install libavahi-client-dev + +Or on Red Hat / Fedora / CentOS: + + sudo yum install git + sudo yum install gcc + sudo yum install gcc-c++ + sudo yum install make + sudo yum install alsa-lib-devel + sudo yum install libudev-devel + sudo yum install avahi-devel + +CentOS 6 & 7 currently have cmake 2.8 but we need 3.1 or later. +First you need to enable the EPEL repository. Add a symlink if you don't already have the older version and want to type cmake rather than cmake3. + + sudo yum install epel-release + sudo rpm -e cmake + sudo yum install cmake3 + sudo ln -s /usr/bin/cmake3 /usr/bin/cmake + +Then on any flavor of Linux: + cd ~ git clone https://www.github.com/wb2osz/direwolf cd direwolf - make + git checkout dev + mkdir build && cd build + cmake .. + make -j4 sudo make install make install-conf -This should give you the most recent stable release. If you want the latest (possibly unstable) development version, use "git checkout dev" before the first "make" command. +This gives you the latest development version. Leave out the "git checkout dev" to get the most recent stable release. For more details see the **User Guide** in the [**doc** directory](https://github.com/wb2osz/direwolf/tree/master/doc). Special considerations for the Raspberry Pi are found in **Raspberry-Pi-APRS.pdf** ### Linux - Using apt-get (Debian flavor operating systems) ### -Results will vary depending on your hardware platform and operating system version because it depends on various volunteers who perform the packaging. +Results will vary depending on your hardware platform and operating system version because it depends on various volunteers who perform the packaging. Expect the version to lag significantly behind development. sudo apt-get update - apt-cache showpkg direwolf + apt-cache showpkg direwolf sudo apt-get install direwolf ### Linux - Using yum (Red Hat flavor operating systems) ### -Results will vary depending on your hardware platform and operating system version because it depends on various volunteers who perform the packaging. +Results will vary depending on your hardware platform and operating system version because it depends on various volunteers who perform the packaging. Expect the version to lag significantly behind development. sudo yum check-update - sudo yum list direwolf + sudo yum list direwolf sudo yum install direwolf -### Linux - Download source in tar or zip file ### -Go to the [releases page](https://github.com/wb2osz/direwolf/releases). Chose desired release and download the source as zip or compressed tar file. Unpack the files, with "unzip" or "tar xfz," and then: +### Macintosh OS X ### - cd direwolf-* - make - sudo make install - make install-conf +Read the **User Guide** in the [**doc** directory](https://github.com/wb2osz/direwolf/tree/master/doc). It is more complicated than Linux. + +If you have problems, post them to the [Dire Wolf packet TNC](https://groups.io/g/direwolf) discussion group. + +You can also install a pre-built version from Mac Ports. Keeping this up to date depends on volunteers who perform the packaging. This version could lag behind development. + + sudo port install direwolf -For more details see the **User Guide** in the [**doc** directory](https://github.com/wb2osz/direwolf/tree/master/doc). Special considerations for the Raspberry Pi are found in **Raspberry-Pi-APRS.pdf** ## Join the conversation ## Here are some good places to ask questions and share your experience: -- [Dire Wolf packet TNC](https://groups.yahoo.com/neo/groups/direwolf_packet/info) +- [Dire Wolf Software TNC](https://groups.io/g/direwolf) -- [Raspberry Pi 4 Ham Radio](https://groups.yahoo.com/neo/groups/Raspberry_Pi_4-Ham_RADIO/info) +- [Raspberry Pi 4 Ham Radio](https://groups.io/g/RaspberryPi-4-HamRadio) -- [linuxham](https://groups.yahoo.com/neo/groups/linuxham/info) +- [linuxham](https://groups.io/g/linuxham) - [TAPR aprssig](http://www.tapr.org/pipermail/aprssig/) diff --git a/cmake/cpack/CMakeLists.txt b/cmake/cpack/CMakeLists.txt new file mode 100644 index 00000000..845c377c --- /dev/null +++ b/cmake/cpack/CMakeLists.txt @@ -0,0 +1 @@ +include(CPack) diff --git a/cmake/cpack/direwolf.desktop.in b/cmake/cpack/direwolf.desktop.in new file mode 100644 index 00000000..79c63aa6 --- /dev/null +++ b/cmake/cpack/direwolf.desktop.in @@ -0,0 +1,10 @@ +[Desktop Entry] +Name=@APPLICATION_NAME@ +Comment=APRS Soundcard TNC +Exec=@APPLICATION_DESKTOP_EXEC@ +Icon=@CMAKE_PROJECT_NAME@_icon.png +StartupNotify=true +Terminal=false +Type=Application +Categories=HamRadio +Keywords=Ham Radio;APRS;Soundcard TNC;KISS;AGWPE;AX.25 \ No newline at end of file diff --git a/cmake/cpack/direwolf.rc b/cmake/cpack/direwolf.rc new file mode 100644 index 00000000..99de6d9f --- /dev/null +++ b/cmake/cpack/direwolf.rc @@ -0,0 +1 @@ +MAINICON ICON "direwolf_icon.ico" \ No newline at end of file diff --git a/dw-icon.ico b/cmake/cpack/direwolf_icon.ico similarity index 100% rename from dw-icon.ico rename to cmake/cpack/direwolf_icon.ico diff --git a/dw-icon.png b/cmake/cpack/direwolf_icon.png similarity index 100% rename from dw-icon.png rename to cmake/cpack/direwolf_icon.png diff --git a/cmake/cpu_tests/test_arm_neon.cxx b/cmake/cpu_tests/test_arm_neon.cxx new file mode 100644 index 00000000..cb48159f --- /dev/null +++ b/cmake/cpu_tests/test_arm_neon.cxx @@ -0,0 +1,16 @@ +#include +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + uint32x4_t x={0}; + x=veorq_u32(x,x); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_avx.cxx b/cmake/cpu_tests/test_x86_avx.cxx new file mode 100644 index 00000000..2344fbcb --- /dev/null +++ b/cmake/cpu_tests/test_x86_avx.cxx @@ -0,0 +1,15 @@ +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + __m256d x = _mm256_setzero_pd(); + x=_mm256_addsub_pd(x,x); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_avx2.cxx b/cmake/cpu_tests/test_x86_avx2.cxx new file mode 100644 index 00000000..369186de --- /dev/null +++ b/cmake/cpu_tests/test_x86_avx2.cxx @@ -0,0 +1,15 @@ +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + __m256i x = _mm256_setzero_si256(); + x=_mm256_add_epi64 (x,x); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_avx512.cxx b/cmake/cpu_tests/test_x86_avx512.cxx new file mode 100644 index 00000000..eed07d3f --- /dev/null +++ b/cmake/cpu_tests/test_x86_avx512.cxx @@ -0,0 +1,16 @@ +#include +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + uint64_t x[8] = {0}; + __m512i y = _mm512_loadu_si512((__m512i*)x); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_sse2.cxx b/cmake/cpu_tests/test_x86_sse2.cxx new file mode 100644 index 00000000..98eb27ea --- /dev/null +++ b/cmake/cpu_tests/test_x86_sse2.cxx @@ -0,0 +1,15 @@ +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + __m128i x = _mm_setzero_si128(); + x=_mm_add_epi64(x,x); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_sse3.cxx b/cmake/cpu_tests/test_x86_sse3.cxx new file mode 100644 index 00000000..70a31e3f --- /dev/null +++ b/cmake/cpu_tests/test_x86_sse3.cxx @@ -0,0 +1,16 @@ +#include +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + __m128d x = _mm_setzero_pd(); + x=_mm_addsub_pd(x,x); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_sse41.cxx b/cmake/cpu_tests/test_x86_sse41.cxx new file mode 100644 index 00000000..e08697fb --- /dev/null +++ b/cmake/cpu_tests/test_x86_sse41.cxx @@ -0,0 +1,18 @@ +#include +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + __m128i x = _mm_setzero_si128(); + __m128i a = _mm_setzero_si128(); + __m128i b = _mm_setzero_si128(); + x=_mm_blend_epi16(a,b,4); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_sse42.cxx b/cmake/cpu_tests/test_x86_sse42.cxx new file mode 100644 index 00000000..58032a57 --- /dev/null +++ b/cmake/cpu_tests/test_x86_sse42.cxx @@ -0,0 +1,15 @@ +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + unsigned int x=32; + x=_mm_crc32_u8(x,4); + return 0; +} diff --git a/cmake/cpu_tests/test_x86_ssse3.cxx b/cmake/cpu_tests/test_x86_ssse3.cxx new file mode 100644 index 00000000..01688f4a --- /dev/null +++ b/cmake/cpu_tests/test_x86_ssse3.cxx @@ -0,0 +1,16 @@ +#include +#include +#include +#include + +void signalHandler(int signum) { + exit(signum); // SIGILL = 4 +} + +int main(int argc, char* argv[]) +{ + signal(SIGILL, signalHandler); + __m128i x = _mm_setzero_si128(); + x=_mm_alignr_epi8(x,x,2); + return 0; +} diff --git a/cmake/include/uninstall.cmake.in b/cmake/include/uninstall.cmake.in new file mode 100644 index 00000000..2037e365 --- /dev/null +++ b/cmake/include/uninstall.cmake.in @@ -0,0 +1,21 @@ +if(NOT EXISTS "@CMAKE_CURRENT_BINARY_DIR@/install_manifest.txt") + message(FATAL_ERROR "Cannot find install manifest: @CMAKE_CURRENT_BINARY_DIR@/install_manifest.txt") +endif(NOT EXISTS "@CMAKE_CURRENT_BINARY_DIR@/install_manifest.txt") + +file(READ "@CMAKE_CURRENT_BINARY_DIR@/install_manifest.txt" files) +string(REGEX REPLACE "\n" ";" files "${files}") +foreach(file ${files}) + message(STATUS "Uninstalling $ENV{DESTDIR}${file}") + if(IS_SYMLINK "$ENV{DESTDIR}${file}" OR EXISTS "$ENV{DESTDIR}${file}") + exec_program( + "@CMAKE_COMMAND@" ARGS "-E remove \"$ENV{DESTDIR}${file}\"" + OUTPUT_VARIABLE rm_out + RETURN_VALUE rm_retval + ) + if(NOT "${rm_retval}" STREQUAL 0) + message(FATAL_ERROR "Problem when removing $ENV{DESTDIR}${file}") + endif(NOT "${rm_retval}" STREQUAL 0) + else(IS_SYMLINK "$ENV{DESTDIR}${file}" OR EXISTS "$ENV{DESTDIR}${file}") + message(STATUS "File $ENV{DESTDIR}${file} does not exist.") + endif(IS_SYMLINK "$ENV{DESTDIR}${file}" OR EXISTS "$ENV{DESTDIR}${file}") +endforeach(file) diff --git a/cmake/modules/FindAvahi.cmake b/cmake/modules/FindAvahi.cmake new file mode 100644 index 00000000..9dc27618 --- /dev/null +++ b/cmake/modules/FindAvahi.cmake @@ -0,0 +1,19 @@ + +find_library(AVAHI_COMMON_LIBRARY NAMES avahi-common PATHS ${PC_AVAHI_CLIENT_LIBRARY_DIRS}) +if(AVAHI_COMMON_LIBRARY) + set(AVAHI_COMMON_FOUND TRUE) +endif() + +find_library(AVAHI_CLIENT_LIBRARY NAMES avahi-client PATHS ${PC_AVAHI_CLIENT_LIBRARY_DIRS}) +if(AVAHI_CLIENT_LIBRARY) + set(AVAHI_CLIENT_FOUND TRUE) +endif() + +FIND_PACKAGE_HANDLE_STANDARD_ARGS(Avahi DEFAULT_MSG AVAHI_COMMON_FOUND AVAHI_CLIENT_FOUND) + +if (AVAHI_FOUND) + set(AVAHI_INCLUDE_DIRS ${AVAHI_UI_INCLUDE_DIR}) + set(AVAHI_LIBRARIES ${AVAHI_COMMON_LIBRARY} ${AVAHI_CLIENT_LIBRARY}) +endif() + +mark_as_advanced(AVAHI_INCLUDE_DIRS AVAHI_LIBRARIES) diff --git a/cmake/modules/FindCPUflags.cmake b/cmake/modules/FindCPUflags.cmake new file mode 100644 index 00000000..abb9e184 --- /dev/null +++ b/cmake/modules/FindCPUflags.cmake @@ -0,0 +1,383 @@ +# Clang or AppleClang (see CMP0025) +if(NOT DEFINED C_CLANG AND CMAKE_CXX_COMPILER_ID MATCHES "Clang") + set(C_CLANG 1) +elseif(NOT DEFINED C_GCC AND CMAKE_CXX_COMPILER_ID MATCHES "GNU") + set(C_GCC 1) +elseif(NOT DEFINED C_MSVC AND CMAKE_CXX_COMPILER_ID MATCHES "MSVC") + set(C_MSVC 1) +endif() + +# Detect current compilation architecture and create standard definitions +include(CheckSymbolExists) +function(detect_architecture symbol arch) + if (NOT DEFINED ARCHITECTURE) + set(CMAKE_REQUIRED_QUIET 1) + check_symbol_exists("${symbol}" "" ARCHITECTURE_${arch}) + unset(CMAKE_REQUIRED_QUIET) + + # The output variable needs to be unique across invocations otherwise + # CMake's crazy scope rules will keep it defined + if (ARCHITECTURE_${arch}) + set(ARCHITECTURE "${arch}" PARENT_SCOPE) + set(ARCHITECTURE_${arch} 1 PARENT_SCOPE) + add_definitions(-DARCHITECTURE_${arch}=1) + endif() + endif() +endfunction() + +# direwolf versions thru 1.5 were available pre-built for 32 bit Windows targets. +# Research and experimentation revealed that the SSE instructions made a big +# difference in runtime speed but SSE2 and later were not significantly better +# for this application. I decided to build with only the SSE instructions making +# the Pentium 3 the minimum requirement. SSE2 would require at least a Pentium 4 +# and offered no significant performance advantage. +# These are ancient history - from the previous Century - but old computers, generally +# considered useless for anything else, often end up in the ham shack. +# +# When cmake was first used for direwolf, the default target became 64 bit and the +# SSE2, SSE3, SSE4.1, and SSE4.2 instructions were automatically enabled based on the +# build machine capabilities. This was fine until I tried running the application +# on a computer much older than where it was built. It did not have the SSE4 instructions +# and the application died without a clue for the reason. +# Just how much benefit do these new instructions provide for this application? +# +# These were all run on the same computer, but compiled in different ways. +# Times to run atest with Track 1 of the TNC test CD: +# +# direwolf 1.5 - 32 bit target - gcc 6.3.0 +# +# 60.4 sec. Pentium 3 with SSE +# +# direwolf 1.6 - 32 bit target - gcc 7.4.0 +# +# 81.0 sec. with no SIMD instructions enabled. +# 54.4 sec. with SSE +# 52.0 sec. with SSE2 +# 52.4 sec. with SSE2, SSE3 +# 52.3 sec. with SSE2, SSE3, SSE4.1, SSE4.2 +# 49.9 sec. Fedora standard: -m32 -march=i686 -mtune=generic -msse2 -mfpmath=sse +# 50.4 sec. sse not sse2: -m32 -march=i686 -mtune=generic -msse -mfpmath=sse +# +# That's what I found several years ago with a much older compiler. +# The original SSE helped a lot but SSE2 and later made little difference. +# +# direwolf 1.6 - 64 bit target - gcc 7.4.0 +# +# 34.8 sec. with no SIMD instructions enabled. +# 34.8 sec. with SSE +# 34.8 sec. with SSE2 +# 34.2 sec. with SSE2, SSE3 +# 33.5 sec. with SSE2, SSE3, SSE4.1, SSE4.2 +# 33.4 Fedora standard: -mtune=generic +# +# Why do we see such little variation? 64-bit target implies +# SSE, SSE2, SSE3 instructions are available. +# +# Building for a 64 bit target makes it run about 1.5x faster on the same hardware. +# +# The default will be set for maximum portability so packagers won't need to +# to anything special. +# +# +# While ENABLE_GENERIC also had the desired result (for x86_64), I don't think +# it is the right approach. It prevents the detection of the architecture, +# i.e. x86, x86_64, ARM, ARM64. That's why it did not go looking for the various +# SSE instructions. For x86, we would miss out on using SSE. + +if (NOT ENABLE_GENERIC) + if (C_MSVC) + detect_architecture("_M_AMD64" x86_64) + detect_architecture("_M_IX86" x86) + detect_architecture("_M_ARM" ARM) + detect_architecture("_M_ARM64" ARM64) + else() + detect_architecture("__x86_64__" x86_64) + detect_architecture("__i386__" x86) + detect_architecture("__arm__" ARM) + detect_architecture("__aarch64__" ARM64) + endif() +endif() +if (NOT DEFINED ARCHITECTURE) + set(ARCHITECTURE "GENERIC") + set(ARCHITECTURE_GENERIC 1) + add_definitions(-DARCHITECTURE_GENERIC=1) +endif() +message(STATUS "Target architecture: ${ARCHITECTURE}") + +set(TEST_DIR ${PROJECT_SOURCE_DIR}/cmake/cpu_tests) + +# flag that set the minimum cpu flag requirements +# used to create re-distribuitable binary + +if (${ARCHITECTURE} MATCHES "x86_64|x86" AND (FORCE_SSE OR FORCE_SSSE3 OR FORCE_SSE41)) + if (FORCE_SSE) + set(HAS_SSE ON CACHE BOOL "SSE SIMD enabled") + if(C_GCC OR C_CLANG) + if (${ARCHITECTURE} MATCHES "x86_64") + # All 64-bit capable chips support MMX, SSE, SSE2, and SSE3 + # so they are all enabled automatically. We don't want to use + # SSE4, based on build machine capabilites, because the application + # would not run properly on an older CPU. + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mtune=generic" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mtune=generic" ) + else() + # Fedora standard uses -msse2 here. + # I dropped it down to -msse for greater compatibility and little penalty. + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -m32 -march=i686 -mtune=generic -msse -mfpmath=sse" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -m32 -march=i686 -mtune=generic -msse -mfpmath=sse" ) + endif() + message(STATUS "Use SSE SIMD instructions") + add_definitions(-DUSE_SSE) + elseif(C_MSVC) + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:SSE" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:SSE" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + message(STATUS "Use MSVC SSSE3 SIMD instructions") + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSSE3) + endif() + elseif (FORCE_SSSE3) + set(HAS_SSSE3 ON CACHE BOOL "SSSE3 SIMD enabled") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mssse3" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mssse3" ) + message(STATUS "Use SSSE3 SIMD instructions") + add_definitions(-DUSE_SSSE3) + elseif(C_MSVC) + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:SSSE3" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:SSSE3" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + message(STATUS "Use MSVC SSSE3 SIMD instructions") + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSSE3) + endif() + elseif (FORCE_SSE41) + set(HAS_SSSE3 ON CACHE BOOL "SSSE3 SIMD enabled") + set(HAS_SSE4_1 ON CACHE BOOL "Architecture has SSE 4.1 SIMD enabled") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -msse4.1" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -msse4.1" ) + message(STATUS "Use SSE 4.1 SIMD instructions") + add_definitions(-DUSE_SSSE3) + add_definitions(-DUSE_SSE4_1) + elseif(C_MSVC) + # seems that from MSVC 2015 comiler doesn't support those flags + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:SSE4_1" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:SSE4_1" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + message(STATUS "Use SSE 4.1 SIMD instructions") + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSSE3) + add_definitions(-DUSE_SSE4_1) + endif() + endif() +else () + if (${ARCHITECTURE} MATCHES "x86_64|x86") + if(C_MSVC) + try_run(RUN_SSE2 COMPILE_SSE2 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_sse2.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_SSE2 COMPILE_SSE2 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_sse2.cxx" COMPILE_DEFINITIONS -msse2 -O0) + endif() + if(COMPILE_SSE2 AND RUN_SSE2 EQUAL 0) + set(HAS_SSE2 ON CACHE BOOL "Architecture has SSSE2 SIMD enabled") + message(STATUS "Use SSE2 SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -msse2" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -msse2" ) + add_definitions(-DUSE_SSE2) + elseif(C_MSVC) + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:SSE2" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:SSE2" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:SSE2" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:SSE2" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSE2) + endif() + else() + set(HAS_SSE2 OFF CACHE BOOL "Architecture does not have SSSE2 SIMD enabled") + endif() + if(C_MSVC) + try_run(RUN_SSSE3 COMPILE_SSSE3 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_ssse3.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_SSSE3 COMPILE_SSSE3 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_ssse3.cxx" COMPILE_DEFINITIONS -mssse3 -O0) + endif() + if(COMPILE_SSSE3 AND RUN_SSSE3 EQUAL 0) + set(HAS_SSSE3 ON CACHE BOOL "Architecture has SSSE3 SIMD enabled") + message(STATUS "Use SSSE3 SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mssse3" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mssse3" ) + add_definitions(-DUSE_SSSE3) + elseif(C_MSVC) + # seems not present on MSVC 2017 + #set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:SSSE3" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSSE3) + endif() + else() + set(HAS_SSSE3 OFF CACHE BOOL "Architecture does not have SSSE3 SIMD enabled") + endif() + if(C_MSVC) + try_run(RUN_SSE4_1 COMPILE_SSE4_1 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_sse41.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_SSE4_1 COMPILE_SSE4_1 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_sse41.cxx" COMPILE_DEFINITIONS -msse4.1 -O0) + endif() + if(COMPILE_SSE4_1 AND RUN_SSE4_1 EQUAL 0) + set(HAS_SSE4_1 ON CACHE BOOL "Architecture has SSE 4.1 SIMD enabled") + message(STATUS "Use SSE 4.1 SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -msse4.1" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -msse4.1" ) + add_definitions(-DUSE_SSE4_1) + elseif(C_MSVC) + # seems not present on MSVC 2017 + #set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:SSE4_1" ) + #set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:SSE4_1" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSE4_1) + endif() + else() + set(HAS_SSE4_1 OFF CACHE BOOL "Architecture does not have SSE 4.1 SIMD enabled") + endif() + if(C_MSVC) + try_run(RUN_SSE4_2 COMPILE_SSE4_2 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_sse42.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_SSE4_2 COMPILE_SSE4_2 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_sse42.cxx" COMPILE_DEFINITIONS -msse4.2 -O0) + endif() + if(COMPILE_SSE4_2 AND RUN_SSE4_2 EQUAL 0) + set(HAS_SSE4_2 ON CACHE BOOL "Architecture has SSE 4.2 SIMD enabled") + message(STATUS "Use SSE 4.2 SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -msse4.2" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -msse4.2" ) + add_definitions(-DUSE_SSE4_2) + elseif(C_MSVC) + # seems not present on MSVC 2017 + #set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:SSE4_2" ) + #set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:SSE4_2" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_SSE4_2) + endif() + else() + set(HAS_SSE4_2 OFF CACHE BOOL "Architecture does not have SSE 4.2 SIMD enabled") + endif() + if(C_MSVC) + try_run(RUN_AVX COMPILE_AVX "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_avx.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_AVX COMPILE_AVX "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_avx.cxx" COMPILE_DEFINITIONS -mavx -O0) + endif() + if(COMPILE_AVX AND RUN_AVX EQUAL 0) + set(HAS_AVX ON CACHE BOOL "Architecture has AVX SIMD enabled") + message(STATUS "Use AVX SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mavx" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mavx" ) + add_definitions(-DUSE_AVX) + elseif(C_MSVC) + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:AVX" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:AVX" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:AVX" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:AVX" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_AVX) + endif() + else() + set(HAS_AVX OFF CACHE BOOL "Architecture does not have AVX SIMD enabled") + endif() + if(C_MSVC) + try_run(RUN_AVX2 COMPILE_AVX2 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_avx2.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_AVX2 COMPILE_AVX2 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_avx2.cxx" COMPILE_DEFINITIONS -mavx2 -O0) + endif() + if(COMPILE_AVX2 AND RUN_AVX2 EQUAL 0) + set(HAS_AVX2 ON CACHE BOOL "Architecture has AVX2 SIMD enabled") + message(STATUS "Use AVX2 SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mavx2" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mavx2" ) + add_definitions(-DUSE_AVX2) + elseif(C_MSVC) + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:AVX2" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:AVX2" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:AVX2" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:AVX2" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_AVX2) + endif() + else() + set(HAS_AVX2 OFF CACHE BOOL "Architecture does not have AVX2 SIMD enabled") + endif() + if(C_MSVC) + try_run(RUN_AVX512 COMPILE_AVX512 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_avx512.cxx" COMPILE_DEFINITIONS /O0) + else() + try_run(RUN_AVX512 COMPILE_AVX512 "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_x86_avx512.cxx" COMPILE_DEFINITIONS -mavx512f -O0) + endif() + if(COMPILE_AVX512 AND RUN_AVX512 EQUAL 0) + set(HAS_AVX512 ON CACHE BOOL "Architecture has AVX512 SIMD enabled") + message(STATUS "Use AVX512 SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mavx512f" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mavx512f" ) + add_definitions(-DUSE_AVX512) + elseif(C_MSVC) + set( CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /arch:AVX512" ) + set( CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /arch:AVX512" ) + set( CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:AVX512" ) + set( CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} /Oi /GL /Ot /Ox /arch:AVX512" ) + set( CMAKE_EXE_LINKER_FLAGS_RELEASE "${CMAKE_EXE_LINKER_FLAGS_RELEASE} /LTCG" ) + add_definitions (/D "_CRT_SECURE_NO_WARNINGS") + add_definitions(-DUSE_AVX512) + endif() + else() + set(HAS_AVX512 OFF CACHE BOOL "Architecture does not have AVX512 SIMD enabled") + endif() +elseif(ARCHITECTURE_ARM) + if(C_MSVC) + try_run(RUN_NEON COMPILE_NEON "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_arm_neon.cxx" COMPILE_DEFINITIONS /O0) + else() + if(${CMAKE_HOST_SYSTEM_PROCESSOR} STREQUAL ${CMAKE_SYSTEM_PROCESSOR}) + try_run(RUN_NEON COMPILE_NEON "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_arm_neon.cxx" COMPILE_DEFINITIONS -mfpu=neon -O0) + else() + try_compile(COMPILE_NEON "${CMAKE_BINARY_DIR}/tmp" "${TEST_DIR}/test_arm_neon.cxx" COMPILE_DEFINITIONS -mfpu=neon -O0) + set(RUN_NEON 0) + endif() + endif() + if(COMPILE_NEON AND RUN_NEON EQUAL 0) + set(HAS_NEON ON CACHE BOOL "Architecture has NEON SIMD enabled") + message(STATUS "Use NEON SIMD instructions") + if(C_GCC OR C_CLANG) + set( CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -mfpu=neon" ) + set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mfpu=neon" ) + add_definitions(-DUSE_NEON) + endif() + else() + set(HAS_NEON OFF CACHE BOOL "Architecture does not have NEON SIMD enabled") + endif() +elseif(ARCHITECTURE_ARM64) + # Advanced SIMD (aka NEON) is mandatory for AArch64 + set(HAS_NEON ON CACHE BOOL "Architecture has NEON SIMD enabled") + message(STATUS "Use NEON SIMD instructions") + add_definitions(-DUSE_NEON) +endif() +endif() + +# clear binary test folder +FILE(REMOVE_RECURSE ${CMAKE_BINARY_DIR}/tmp) diff --git a/cmake/modules/FindCompiler.cmake b/cmake/modules/FindCompiler.cmake new file mode 100644 index 00000000..fe036e4b --- /dev/null +++ b/cmake/modules/FindCompiler.cmake @@ -0,0 +1,15 @@ +# Clang or AppleClang (see CMP0025) +if(NOT DEFINED C_CLANG AND CMAKE_CXX_COMPILER_ID MATCHES "Clang") + set(C_CLANG 1) +elseif(NOT DEFINED C_GCC AND CMAKE_CXX_COMPILER_ID MATCHES "GNU") + set(C_GCC 1) +elseif(NOT DEFINED C_MSVC AND CMAKE_CXX_COMPILER_ID MATCHES "MSVC") + set(C_MSVC 1) + if(MSVC_VERSION GREATER_EQUAL 1920 AND MSVC_VERSION LESS_EQUAL 1929) + set(VS2019 ON) + elseif(MSVC_VERSION GREATER_EQUAL 1910 AND MSVC_VERSION LESS_EQUAL 1919) + set(VS2017 ON) + elseif(MSVC_VERSION GREATER 1899 AND MSVC_VERSION LESS 1910) + set(VS2015 ON) + endif() +endif() diff --git a/cmake/modules/FindGPSD.cmake b/cmake/modules/FindGPSD.cmake new file mode 100644 index 00000000..d21b3311 --- /dev/null +++ b/cmake/modules/FindGPSD.cmake @@ -0,0 +1,88 @@ +# - Try to find GPSD +# Once done this will define +# +# GPSD_FOUND - system has GPSD +# GPSD_INCLUDE_DIRS - the GPSD include directory +# GPSD_LIBRARIES - Link these to use GPSD +# GPSD_DEFINITIONS - Compiler switches required for using GPSD +# +# Copyright (c) 2006 Andreas Schneider +# +# Redistribution and use is allowed according to the terms of the New +# BSD license. +# For details see the accompanying COPYING-CMAKE-SCRIPTS file. +# + +set(GPSD_ROOT_DIR + "${GPSD_ROOT_DIR}" + CACHE + PATH + "Directory to search for gpsd") + +find_package(PkgConfig QUIET) +if(PKG_CONFIG_FOUND) + pkg_check_modules(PC_GPSD libgps) +endif() + +if (GPSD_LIBRARIES AND GPSD_INCLUDE_DIRS) + # in cache already + set(GPSD_FOUND TRUE) +else (GPSD_LIBRARIES AND GPSD_INCLUDE_DIRS) + find_path(GPSD_INCLUDE_DIRS + NAMES + gps.h + PATHS + /usr/include + /usr/local/include + /opt/local/include + /sw/include + /usr/include/gps + /usr/local/include/gps + /opt/local/include/gps + /sw/include/gps + HINTS + ${PC_GPSD_INCLUDEDIR} + ${GPSD_ROOT_DIR} + ) + + # debian uses version suffixes + # add suffix evey new release + find_library(GPSD_LIBRARIES + NAMES + gps + PATHS + /usr/lib64 + /usr/lib + /usr/local/lib + /opt/local/lib + /sw/lib + HINTS + ${PC_GPSD_LIBDIR} + ${GPSD_ROOT_DIR} + ) + + if (GPSD_INCLUDE_DIRS AND GPSD_LIBRARIES) + set(GPSD_FOUND TRUE) + endif (GPSD_INCLUDE_DIRS AND GPSD_LIBRARIES) + + if (GPSD_FOUND) + if (NOT GPSD_FIND_QUIETLY) + message(STATUS "Found GPSD: ${GPSD_LIBRARIES}") + endif (NOT GPSD_FIND_QUIETLY) + else (GPSD_FOUND) + if (GPSD_FIND_REQUIRED) + message(FATAL_ERROR "Could not find GPSD") + endif (GPSD_FIND_REQUIRED) + endif (GPSD_FOUND) + + # show the GPSD_INCLUDE_DIRS and GPSD_LIBRARIES variables only in the advanced view + mark_as_advanced(GPSD_INCLUDE_DIRS GPSD_LIBRARIES) + +endif (GPSD_LIBRARIES AND GPSD_INCLUDE_DIRS) + +# maybe on CYGWIN gpsd works +if (WIN32) + set(GPSD_FOUND FALSE) + set(GPSD_LIBRARIES "") + set(GPSD_INCLUDE_DIRS "") +endif (WIN32) diff --git a/cmake/modules/FindPortaudio.cmake b/cmake/modules/FindPortaudio.cmake new file mode 100644 index 00000000..9cda3428 --- /dev/null +++ b/cmake/modules/FindPortaudio.cmake @@ -0,0 +1,64 @@ +# - Try to find Portaudio +# Once done this will define +# +# PORTAUDIO_FOUND - system has Portaudio +# PORTAUDIO_INCLUDE_DIRS - the Portaudio include directory +# PORTAUDIO_LIBRARIES - Link these to use Portaudio + +set(PORTAUDIO_ROOT_DIR + "${PORTAUDIO_ROOT_DIR}" + CACHE + PATH + "Directory to search for portaudio") + +find_package(PkgConfig QUIET) +if(PKG_CONFIG_FOUND) + pkg_check_modules(PC_PORTAUDIO portaudio-2.0) +endif() + +find_path(PORTAUDIO_INCLUDE_DIRS + NAMES + portaudio.h + PATHS + /usr/local/include + /usr/include + /opt/local/include + HINTS + ${PC_PORTAUDIO_INCLUDEDIR} + ${PORTAUDIO_ROOT_DIR} + ) + +find_library(PORTAUDIO_LIBRARIES + NAMES + portaudio + PATHS + /usr/local/lib + /usr/lib + /usr/lib64 + /opt/local/lib + HINTS + ${PC_PORTAUDIO_LIBDIR} + ${PORTAUDIO_ROOT_DIR} + ) + +mark_as_advanced(PORTAUDIO_INCLUDE_DIRS PORTAUDIO_LIBRARIES) + +# Found PORTAUDIO, but it may be version 18 which is not acceptable. +if(EXISTS ${PORTAUDIO_INCLUDE_DIRS}/portaudio.h) + include(CheckCXXSourceCompiles) + set(CMAKE_REQUIRED_INCLUDES_SAVED ${CMAKE_REQUIRED_INCLUDES}) + set(CMAKE_REQUIRED_INCLUDES ${PORTAUDIO_INCLUDE_DIRS}) + CHECK_CXX_SOURCE_COMPILES( + "#include \nPaDeviceIndex pa_find_device_by_name(const char *name); int main () {return 0;}" + PORTAUDIO2_FOUND) + set(CMAKE_REQUIRED_INCLUDES ${CMAKE_REQUIRED_INCLUDES_SAVED}) + unset(CMAKE_REQUIRED_INCLUDES_SAVED) + if(PORTAUDIO2_FOUND) + INCLUDE(FindPackageHandleStandardArgs) + FIND_PACKAGE_HANDLE_STANDARD_ARGS(PORTAUDIO DEFAULT_MSG PORTAUDIO_INCLUDE_DIRS PORTAUDIO_LIBRARIES) + else(PORTAUDIO2_FOUND) + message(STATUS + " portaudio.h not compatible (requires API 2.0)") + set(PORTAUDIO_FOUND FALSE) + endif(PORTAUDIO2_FOUND) +endif() diff --git a/cmake/modules/Findhamlib.cmake b/cmake/modules/Findhamlib.cmake new file mode 100644 index 00000000..16ca5685 --- /dev/null +++ b/cmake/modules/Findhamlib.cmake @@ -0,0 +1,67 @@ +# - Try to find Hamlib +# +# HAMLIB_FOUND - system has Hamlib +# HAMLIB_LIBRARIES - location of the library for hamlib +# HAMLIB_INCLUDE_DIRS - location of the include files for hamlib +# +# Requires these CMake modules: +# FindPackageHandleStandardArgs (known included with CMake >=2.6.2) +# +# Original Author: +# 2019 Davide Gerhard + +set(HAMLIB_ROOT_DIR + "${HAMLIB_ROOT_DIR}" + CACHE + PATH + "Directory to search for hamlib") + +find_package(PkgConfig QUIET) +if(PKG_CONFIG_FOUND) + pkg_check_modules(PC_HAMLIB hamlib) +endif() + +find_path(HAMLIB_INCLUDE_DIR + NAMES hamlib/rig.h + PATHS + /usr/include + /usr/local/include + /opt/local/include + HINTS + ${PC_HAMLIB_INCLUDEDIR} + ${HAMLIB_ROOT_DIR} + ) + +find_library(HAMLIB_LIBRARY + NAMES hamlib + PATHS + /usr/lib64/hamlib + /usr/lib/hamlib + /usr/lib64 + /usr/lib + /usr/local/lib64/hamlib + /usr/local/lib/hamlib + /usr/local/lib64 + /usr/local/lib + /opt/local/lib + /opt/local/lib/hamlib + HINTS + ${PC_HAMLIB_LIBDIR} + ${HAMLIB_ROOT_DIR} + + ) + +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(hamlib + DEFAULT_MSG + HAMLIB_LIBRARY + HAMLIB_INCLUDE_DIR + ) + +if(HAMLIB_FOUND) + list(APPEND HAMLIB_LIBRARIES ${HAMLIB_LIBRARY}) + list(APPEND HAMLIB_INCLUDE_DIRS ${HAMLIB_INCLUDE_DIR}) + mark_as_advanced(HAMLIB_ROOT_DIR) +endif() + +mark_as_advanced(HAMLIB_INCLUDE_DIR HAMLIB_LIBRARY) diff --git a/cmake/modules/Findsndio.cmake b/cmake/modules/Findsndio.cmake new file mode 100644 index 00000000..e7292d5d --- /dev/null +++ b/cmake/modules/Findsndio.cmake @@ -0,0 +1,42 @@ +# - Try to find sndio +# +# SNDIO_FOUND - system has sndio +# SNDIO_LIBRARIES - location of the library for sndio +# SNDIO_INCLUDE_DIRS - location of the include files for sndio + +set(SNDIO_ROOT_DIR + "${SNDIO_ROOT_DIR}" + CACHE + PATH + "Directory to search for sndio") + +# no need to check pkg-config + +find_path(SNDIO_INCLUDE_DIRS + NAMES + sndio.h + PATHS + /usr/local/include + /usr/include + /opt/local/include + HINTS + ${SNDIO_ROOT_DIR} + ) + +find_library(SNDIO_LIBRARIES + NAMES + sndio + PATHS + /usr/local/lib + /usr/lib + /usr/lib64 + /opt/local/lib + HINTS + ${SNDIIO_ROOT_DIR} + ) + + +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(SNDIO DEFAULT_MSG SNDIO_INCLUDE_DIRS SNDIO_LIBRARIES) + +mark_as_advanced(SNDIO_INCLUDE_DIRS SNDIO_LIBRARIES) diff --git a/cmake/modules/Findudev.cmake b/cmake/modules/Findudev.cmake new file mode 100644 index 00000000..c8c4b624 --- /dev/null +++ b/cmake/modules/Findudev.cmake @@ -0,0 +1,85 @@ +# - try to find the udev library +# +# Cache Variables: (probably not for direct use in your scripts) +# UDEV_INCLUDE_DIR +# UDEV_SOURCE_DIR +# UDEV_LIBRARY +# +# Non-cache variables you might use in your CMakeLists.txt: +# UDEV_FOUND +# UDEV_INCLUDE_DIRS +# UDEV_LIBRARIES +# +# Requires these CMake modules: +# FindPackageHandleStandardArgs (known included with CMake >=2.6.2) +# +# Original Author: +# 2014 Kevin M. Godby +# +# Distributed under the Boost Software License, Version 1.0. +# (See accompanying file LICENSE_1_0.txt or copy at +# http://www.boost.org/LICENSE_1_0.txt) + +set(UDEV_ROOT_DIR + "${UDEV_ROOT_DIR}" + CACHE + PATH + "Directory to search for udev") + +find_package(PkgConfig QUIET) +if(PKG_CONFIG_FOUND) + pkg_check_modules(PC_LIBUDEV libudev) +endif() + +find_library(UDEV_LIBRARY + NAMES + udev + PATHS + ${PC_LIBUDEV_LIBRARY_DIRS} + ${PC_LIBUDEV_LIBDIR} + /usr/lib64 + /usr/lib + /usr/local/lib + HINTS + "${UDEV_ROOT_DIR}" + PATH_SUFFIXES + lib + ) + +get_filename_component(_libdir "${UDEV_LIBRARY}" PATH) + +find_path(UDEV_INCLUDE_DIR + NAMES + libudev.h + PATHS + /usr/include + /usr/local/include + ${PC_LIBUDEV_INCLUDE_DIRS} + ${PC_LIBUDEV_INCLUDEDIR} + HINTS + "${_libdir}" + "${_libdir}/.." + "${UDEV_ROOT_DIR}" + PATH_SUFFIXES + include + ) + +include(FindPackageHandleStandardArgs) +find_package_handle_standard_args(udev + DEFAULT_MSG + UDEV_LIBRARY + UDEV_INCLUDE_DIR + ) + +if (UDEV_INCLUDE_DIR AND UDEV_LIBRARY) + set(UDEV_FOUND TRUE) +endif (UDEV_INCLUDE_DIR AND UDEV_LIBRARY) + +if(UDEV_FOUND) + list(APPEND UDEV_LIBRARIES ${UDEV_LIBRARY}) + list(APPEND UDEV_INCLUDE_DIRS ${UDEV_INCLUDE_DIR}) + mark_as_advanced(UDEV_ROOT_DIR) +endif() + +mark_as_advanced(UDEV_INCLUDE_DIR + UDEV_LIBRARY) diff --git a/conf/99-direwolf-cmedia.rules b/conf/99-direwolf-cmedia.rules new file mode 100644 index 00000000..94e1828f --- /dev/null +++ b/conf/99-direwolf-cmedia.rules @@ -0,0 +1,36 @@ +# Normally, all of /dev/hidraw* are accessible only by root. +# +# $ ls -l /dev/hidraw* +# crw------- 1 root root 247, 0 Sep 24 09:40 /dev/hidraw0 +# +# An ordinary user, trying to access it will be denied. +# +# Unnecessarily running applications as root is generally a bad idea because it makes it too easy +# to accidentally trash your system. We need to relax the restrictions so ordinary users can use these devices. +# +# If all went well with installation, the /etc/udev/rules.d directory should contain a file called +# 99-direwolf-cmedia.rules containing: +# + +SUBSYSTEM=="hidraw", ATTRS{idVendor}=="0d8c", GROUP="audio", MODE="0660" + +# +# I used the "audio" group, mimicking the permissions on the sound side of the device. +# +# $ ls -l /dev/snd/pcm* +# crw-rw----+ 1 root audio 116, 16 Sep 24 09:40 /dev/snd/pcmC0D0p +# crw-rw----+ 1 root audio 116, 17 Sep 24 09:40 /dev/snd/pcmC0D1p +# +# You should see something similar to this where someone in the "audio" group has read-write access. +# +# $ ls -l /dev/hidraw* +# crw-rw---- 1 root audio 247, 0 Oct 6 19:24 /dev/hidraw0 +# +# Read the User Guide and run the "cm108" application for more information. +# + +# +# Same thing for the "All In One Cable." +# + +SUBSYSTEM=="hidraw", ATTRS{idVendor}=="1209", ATTRS{idProduct}=="7388", GROUP="audio", MODE="0660" diff --git a/conf/CMakeLists.txt b/conf/CMakeLists.txt new file mode 100644 index 00000000..d4a229d7 --- /dev/null +++ b/conf/CMakeLists.txt @@ -0,0 +1,48 @@ +# generate conf per platform +file(READ "${CUSTOM_CONF_DIR}/generic.conf" file_content) + +if(LINUX) + string(REGEX REPLACE "\n%W%[^\n]*" "" file_content "${file_content}") + string(REGEX REPLACE "\n%M%[^\n]*" "" file_content "${file_content}") + string(REGEX REPLACE "\n%L%([^\n]*)" "\n\\1" file_content "${file_content}") +elseif(WIN32 OR CYGWIN) + string(REGEX REPLACE "\n%M%[^\n]*" "" file_content "${file_content}") + string(REGEX REPLACE "\n%L%[^\n]*" "" file_content "${file_content}") + string(REGEX REPLACE "\n%W%([^\n]*)" "\n\\1" file_content "${file_content}") +else() # macOS FreeBSD OpenBSD + string(REGEX REPLACE "\n%W%[^\n]*" "" file_content "${file_content}") + string(REGEX REPLACE "\n%L%[^\n]*" "" file_content "${file_content}") + string(REGEX REPLACE "\n%M%([^\n]*)" "\n\\1" file_content "${file_content}") +endif() + +# remove remark +string(REGEX REPLACE "\n%R%[^\n]*" "" file_content "${file_content}") + +# clear common lines +string(REGEX REPLACE "\n%C%([^\n]*)" "\n\\1" file_content "${file_content}") +string(REGEX REPLACE "^%C%([^\n]*)" "\\1" file_content "${file_content}") + +file(WRITE "${CMAKE_BINARY_DIR}/direwolf.conf" "${file_content}") + +# install udev rules for CM108 +if(LINUX) + install(FILES "${CUSTOM_CONF_DIR}/99-direwolf-cmedia.rules" DESTINATION /etc/udev/rules.d/) +endif() + +install(FILES "${CMAKE_BINARY_DIR}/direwolf.conf" DESTINATION ${INSTALL_CONF_DIR}) +install(FILES "${CUSTOM_CONF_DIR}/sdr.conf" DESTINATION ${INSTALL_CONF_DIR}) + +# Put sample configuration & startup files in home directory. +# This step would be done as ordinary user. +# Some people like to put the direwolf config file in /etc/ax25. +# Note that all of these are also in $(DESTDIR)/share/doc/direwolf/examples/. +if(NOT (WIN32 OR CYGWIN)) + add_custom_target(install-conf + COMMAND ${CMAKE_COMMAND} + -DCUSTOM_BINARY_DIR="${CMAKE_BINARY_DIR}" + -DCUSTOM_CONF_DIR="${CUSTOM_CONF_DIR}" + -DCUSTOM_SCRIPTS_DIR="${CUSTOM_SCRIPTS_DIR}" + -DCUSTOM_TELEMETRY_DIR="${CUSTOM_TELEMETRY_DIR}" + -P "${CMAKE_SOURCE_DIR}/conf/install_conf.cmake" + ) +endif() diff --git a/conf/generic.conf b/conf/generic.conf new file mode 100644 index 00000000..9a19d8a2 --- /dev/null +++ b/conf/generic.conf @@ -0,0 +1,537 @@ +%C%############################################################# +%C%# # +%C%# Configuration file for Dire Wolf # +%C%# # +%L%# Linux version # +%W%# Windows version # +%M%# Macintosh version # +%C%# # +%C%############################################################# +%R% +%R% +%R% The sample config file was getting pretty messy +%R% with the Windows and Linux differences. +%R% It would be a maintenance burden to keep most of +%R% two different versions in sync. +%R% This common source is now used to generate the +%R% two different variations while having only a single +%R% copy of the common parts. +%R% +%R% The first column contains one of the following: +%R% +%R% R remark which is discarded. +%R% C common to both versions. +%R% W Windows version only. +%R% L Linux version only. +%R% M Macintosh version and possibly others (portaudio used). +%R% +%C%# +%C%# Extensive documentation can be found here: +%C%# Stable release - https://github.com/wb2osz/direwolf/tree/master/doc +%C%# Latest development - https://github.com/wb2osz/direwolf/tree/dev/doc +%C%# Additional topics - https://github.com/wb2osz/direwolf-doc +%C%# +%W%# The basic documentation set can also be found in the doc folder. +%L%# The basic documentation set can also be found in +%L%# /usr/local/share/doc/direwolf/ or /usr/share/doc/direwolf/ +%L%# Concise "man" pages are also available for Linux. +%M%# /usr/local/share/doc/direwolf/ or /usr/share/doc/direwolf/ +%M%# Concise "man" pages are also available for Mac OSX. +%C%# +%C%# Questions??? Join the discussion forum: https://groups.io/g/direwolf +%C%# +%C%# +%C%# This sample file does not have examples for all of the possibilities. +%C%# Consult the User Guide for more details on configuration options +%C%# and other documents for more details for different uses. +%C%# +%C%# These are the most likely settings you might change: +%C%# +%C%# (1) MYCALL - call sign and SSID for your station. +%C%# +%C%# Look for lines starting with MYCALL and +%C%# change NOCALL to your own. +%C%# +%C%# (2) PBEACON - enable position beaconing. +%C%# +%C%# Look for lines starting with PBEACON and +%C%# modify for your call, location, etc. +%C%# +%C%# (3) DIGIPEATER - configure digipeating rules. +%C%# +%C%# Look for lines starting with DIGIPEATER. +%C%# Most people will probably use the given example. +%C%# Just remove the "#" from the start of the line +%C%# to enable it. +%C%# +%C%# (4) IGSERVER, IGLOGIN - IGate server and login +%C%# +%C%# Configure an IGate client to relay messages between +%C%# radio and internet servers. +%C%# +%C%# +%C%# The default location is "direwolf.conf" in the current working directory. +%L%# On Linux, the user's home directory will also be searched. +%C%# An alternate configuration file location can be specified with the "-c" command line option. +%C%# +%C%# As you probably guessed by now, # indicates a comment line. +%C%# +%C%# Remove the # at the beginning of a line if you want to use a sample +%C%# configuration that is currently commented out. +%C%# +%C%# Commands are a keyword followed by parameters. +%C%# +%C%# Command key words are case insensitive. i.e. upper and lower case are equivalent. +%C%# +%C%# Command parameters are generally case sensitive. i.e. upper and lower case are different. +%C%# +%C% +%C% +%C%############################################################# +%C%# # +%C%# FIRST AUDIO DEVICE PROPERTIES # +%C%# (Channel 0 + 1 if in stereo) # +%C%# # +%C%############################################################# +%C% +%C%# +%C%# Many people will simply use the default sound device. +%C%# Some might want to use an alternative device by choosing it here. +%C%# +%C%# +%C%# Many examples of radio interfaces and PTT options can be found in: +%C%# https://github.com/wb2osz/direwolf-doc/blob/main/Radio-Interface-Guide.pdf +%C%# +%C%# +%R% ---------- Windows ---------- +%R% +%W%# When the Windows version starts up, it displays something like +%W%# this with the available sound devices and capabilities: +%W%# +%W%# Available audio input devices for receive (*=selected): +%W%# * 0: Microphone (C-Media USB Headpho (channel 2) +%W%# 1: Microphone (Bluetooth SCO Audio +%W%# 2: Microphone (Bluetooth AV Audio) +%W%# * 3: Microphone (Realtek High Defini (channels 0 & 1) +%W%# Available audio output devices for transmit (*=selected): +%W%# * 0: Speakers (C-Media USB Headphone (channel 2) +%W%# 1: Speakers (Bluetooth SCO Audio) +%W%# 2: Realtek Digital Output(Optical) +%W%# 3: Speakers (Bluetooth AV Audio) +%W%# * 4: Speakers (Realtek High Definiti (channels 0 & 1) +%W%# 5: Realtek Digital Output (Realtek +%W%# +%W%# Example: To use the microphone and speaker connections on the +%W%# system board, either of these forms can be used: +%W% +%W%#ADEVICE High +%W%#ADEVICE 3 4 +%W% +%W% +%W%# Example: To use the USB Audio, use a command like this with +%W%# the input and output device numbers. (Remove the # comment character.) +%W%#ADEVICE USB +%W% +%W%# You can also use "-" or "stdin" to pipe stdout from +%W%# some other application such as a software defined radio. +%W%# "stdin" is not an audio device. Don't use this unless you +%W%# understand what this means. Read the User Guide. +%W%# You can also specify "UDP:" and an optional port for input. +%W%# Something different must be specified for output. +%W% +%W%# ADEVICE stdin 0 +%W%# ADEVICE UDP:7355 0 +%W% +%W%# The position in the list can change when devices (e.g. USB) are added and removed. +%W%# You can also specify devices by using part of the name. +%W%# Here is an example of specifying the USB Audio device. +%W%# This is case-sensitive. Upper and lower case are not treated the same. +%W% +%W%#ADEVICE USB +%W% +%W% +%R% ---------- Linux ---------- +%R% +%L%# Linux ALSA is complicated. See User Guide for discussion. +%L%# To use something other than the default, generally use plughw +%L%# and a card number reported by "arecord -l" command. Example: +%L% +%L%# ADEVICE plughw:1,0 +%L% +%L%# You can also use "-" or "stdin" to pipe stdout from +%L%# some other application such as a software defined radio. +%L%# "stdin" is not an audio device. Don't use this unless you +%L%# understand what this means. Read the User Guide. +%L%# You can also specify "UDP:" and an optional port for input. +%L%# Something different must be specified for output. +%L% +%L%# ADEVICE stdin plughw:1,0 +%L%# ADEVICE UDP:7355 default +%L% +%R% ---------- Mac ---------- +%R% +%M%# Macintosh Operating System uses portaudio driver for audio +%M%# input/output. Default device selection not available. User/OP +%M%# must configure the sound input/output option. Note that +%M%# the device names can contain spaces. In this case, the names +%M%# must be enclosed by quotes. +%M%# +%M%# Examples: +%M%# +%M%ADEVICE "Built-in Input" "Built-in Output" +%M% +%M%# ADEVICE "USB Audio Codec:6" "USB Audio Codec:5" +%M%# +%M%# +%M%# You can also use "-" or "stdin" to pipe stdout from +%M%# some other application such as a software defined radio. +%M%# "stdin" is not an audio device. Don't use this unless you +%M%# understand what this means. Read the User Guide. +%M%# You can also specify "UDP:" and an optional port for input. +%M%# Something different must be specified for output. +%M% +%M%# ADEVICE UDP:7355 default +%M%# +%C% +%C%# +%C%# Number of audio channels for this souncard: 1 (mono) or 2 (stereo). +%C%# 1 is the default so there is no need to specify it. +%C%# +%C% +%C%#ACHANNELS 2 +%C% +%C% +%C%############################################################# +%C%# # +%C%# SECOND AUDIO DEVICE PROPERTIES # +%C%# (Channel 2 + 3 if in stereo) # +%C%# # +%C%############################################################# +%C% +%C%#ADEVICE1 ... +%C% +%C% +%C%############################################################# +%C%# # +%C%# THIRD AUDIO DEVICE PROPERTIES # +%C%# (Channel 4 + 5 if in stereo) # +%C%# # +%C%############################################################# +%C% +%C%#ADEVICE2 ... +%C% +%C% +%C%############################################################# +%C%# # +%C%# CHANNEL 0 PROPERTIES # +%C%# # +%C%############################################################# +%C% +%C%CHANNEL 0 +%C% +%C%# +%C%# The following MYCALL, MODEM, PTT, etc. configuration items +%C%# apply to the most recent CHANNEL. +%C%# +%C% +%C%# +%C%# Station identifier for this channel. +%C%# Multiple channels can have the same or different names. +%C%# +%C%# It can be up to 6 letters and digits with an optional ssid. +%C%# The APRS specification requires that it be upper case. +%C%# +%C%# Example (don't use this unless you are me): MYCALL WB2OSZ-5 +%C%# +%C% +%C%MYCALL N0CALL +%C% +%C%# +%C%# Pick a suitable modem speed based on your situation. +%C%# 1200 Most common for VHF/UHF. This is the default if not specified. +%C%# 2400 QPSK compatible with MFJ-2400, and probably PK232-2400 & KPC-2400. +%C%# 300 Low speed for HF SSB. Default tones 1600 & 1800. +%C%# EAS Emergency Alert System (EAS) Specific Area Message Encoding (SAME). +%C%# 9600 G3RUH style - Can't use Microphone and Speaker connections. +%C%# AIS International system for tracking ships on VHF. +%C%# Also uses 9600 bps so Speaker connection won't work. +%C%# +%C%# In most cases you can just specify the speed. Examples: +%C%# +%C% +%C%MODEM 1200 +%C%#MODEM 9600 +%C% +%C%# +%C%# Many options are available for great flexibility. +%C%# See User Guide for details. +%C%# +%C% +%C%# +%C%# Uncomment line below to enable the DTMF decoder for this channel. +%C%# +%C% +%C%#DTMF +%C% +%C%# Push to Talk (PTT) can be confusing because there are so many different cases. +%C%# Radio-Interface-Guide.pdf in https://github.com/wb2osz/direwolf-doc +%C%# goes into detail about the various options. +%C% +%L%# If using a C-Media CM108/CM119 or similar USB Audio Adapter, +%L%# you can use a GPIO pin for PTT control. This is very convenient +%L%# because a single USB connection is used for both audio and PTT. +%L%# Example: +%L% +%L%#PTT CM108 +%L% +%W%# If using a C-Media CM108/CM119 or similar USB Audio Adapter, +%W%# you can use a GPIO pin for PTT control. This is very convenient +%W%# because a single USB connection is used for both audio and PTT. +%W%# Example: +%W% +%W%#PTT CM108 +%W%%C%# +%C%# The transmitter Push to Talk (PTT) control can be wired to a serial port +%C%# with a suitable interface circuit. DON'T connect it directly! +%C%# +%C%# For the PTT command, specify the device and either RTS or DTR. +%C%# RTS or DTR may be preceded by "-" to invert the signal. +%C%# Both can be used for interfaces that want them driven with opposite polarity. +%C%# +%L%# COM1 can be used instead of /dev/ttyS0, COM2 for /dev/ttyS1, and so on. +%L%# +%C% +%C%#PTT COM1 RTS +%C%#PTT COM1 RTS -DTR +%L%#PTT /dev/ttyUSB0 RTS +%L%#PTT /dev/ttyUSB0 RTS -DTR +%C% +%L%# +%L%# On Linux, you can also use general purpose I/O pins if +%L%# your system is configured for user access to them. +%L%# This would apply mostly to microprocessor boards, not a regular PC. +%L%# See separate Raspberry Pi document for more details. +%L%# The number may be preceded by "-" to invert the signal. +%L%# +%L% +%L%#PTT GPIO 25 +%L% +%C%# The Data Carrier Detect (DCD) signal can be sent to most of the same places +%C%# as the PTT signal. This could be used to light up an LED like a normal TNC. +%C% +%C%#DCD COM1 -DTR +%L%#DCD GPIO 24 +%C% +%C% +%C%############################################################# +%C%# # +%C%# CHANNEL 1 PROPERTIES # +%C%# # +%C%############################################################# +%C% +%C%#CHANNEL 1 +%C% +%C%# +%C%# Specify MYCALL, MODEM, PTT, etc. configuration items for +%C%# CHANNEL 1. Repeat for any other channels. +%C% +%C% +%C%############################################################# +%C%# # +%C%# TEXT TO SPEECH COMMAND FILE # +%C%# # +%C%############################################################# +%C% +%W%#SPEECH dwespeak.bat +%L%#SPEECH dwespeak.sh +%C% +%C% +%C%############################################################# +%C%# # +%C%# VIRTUAL TNC SERVER PROPERTIES # +%C%# # +%C%############################################################# +%C% +%C%# +%C%# Dire Wolf acts as a virtual TNC and can communicate with +%C%# client applications by different protocols: +%C%# +%C%# - the "AGW TCPIP Socket Interface" - default port 8000 +%C%# - KISS protocol over TCP socket - default port 8001 +%W%# - KISS TNC via serial port +%L%# - KISS TNC via pseudo terminal (-p command line option) +%C%# +%C% +%C%AGWPORT 8000 +%C%KISSPORT 8001 +%C% +%W%# +%W%# Some applications are designed to operate with only a physical +%W%# TNC attached to a serial port. For these, we provide a virtual serial +%W%# port that appears to be connected to a TNC. +%W%# +%W%# Take a look at the User Guide for instructions to set up +%W%# two virtual serial ports named COM3 and COM4 connected by +%W%# a null modem. +%W%# +%W%# Using the configuration described, Dire Wolf will connect to +%W%# COM3 and the client application will use COM4. +%W%# +%W%# Uncomment following line to use this feature. +%W% +%W%#NULLMODEM COM3 +%W% +%W% +%C%# +%C%# It is sometimes possible to recover frames with a bad FCS. +%C%# This is not a global setting. +%C%# It applies only the the most recent CHANNEL specified. +%C%# +%C%# 0 - Don't try to repair. (default) +%C%# 1 - Attempt to fix single bit error. +%C%# +%C% +%C%#FIX_BITS 0 +%C% +%C%# +%C%############################################################# +%C%# # +%C%# FIXED POSIION BEACONING PROPERTIES # +%C%# # +%C%############################################################# +%C% +%C% +%C%# +%C%# Beaconing is configured with these two commands: +%C%# +%C%# PBEACON - for a position report (usually yourself) +%C%# OBEACON - for an object report (usually some other entity) +%C%# +%C%# Each has a series of keywords and values for options. +%C%# See User Guide for details. +%C%# +%C%# Example: +%C%# +%C%# This results in a broadcast once every 10 minutes. +%C%# Every half hour, it can travel via one digipeater hop. +%C%# The others are kept local. +%C%# +%C% +%C%#PBEACON delay=1 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" via=WIDE1-1 +%C%#PBEACON delay=11 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" +%C%#PBEACON delay=21 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" +%C% +%C%# +%C%# Did you know that APRS comments and messages can contain UTF-8 characters, not only plain ASCII? +%C%# +%C%#PBEACON delay=1 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W comment=" Did you know that APRS comments and messages can contain UTF-8 characters? \xe0\xb8\xa7\xe0\xb8\xb4\xe0\xb8\x97\xe0\xb8\xa2\xe0\xb8\xb8\xe0\xb8\xaa\xe0\xb8\xa1\xe0\xb8\xb1\xe0\xb8\x84\xe0\xb8\xa3\xe0\xb9\x80\xe0\xb8\xa5\xe0\xb9\x88\xe0\xb8\x99" +%C%#PBEACON delay=11 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W comment=" Did you know that APRS comments and messages can contain UTF-8 characters? \xce\xa1\xce\xb1\xce\xb4\xce\xb9\xce\xbf\xce\xb5\xcf\x81\xce\xb1\xcf\x83\xce\xb9\xcf\x84\xce\xb5\xcf\x87\xce\xbd\xce\xb9\xcf\x83\xce\xbc\xcf\x8c\xcf\x82" +%C%#PBEACON delay=21 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W comment=" Did you know that APRS comments and messages can contain UTF-8 characters? \xe3\x82\xa2\xe3\x83\x9e\xe3\x83\x81\xe3\x83\xa5\xe3\x82\xa2\xe7\x84\xa1\xe7\xb7\x9a" +%C%# +%C%# With UTM coordinates instead of latitude and longitude. +%C% +%C%#PBEACON delay=1 every=10 overlay=S symbol="digi" zone=19T easting=307477 northing=4720178 +%C% +%C% +%C%# +%C%# When the destination field is set to "SPEECH" the information part is +%C%# converted to speech rather than transmitted as a data frame. +%C%# +%C% +%C%#CBEACON dest="SPEECH" info="Club meeting tonight at 7 pm." +%C% +%C%# Similar for Morse code. If SSID is specified, it is multiplied +%C%# by 2 to get speed in words per minute (WPM). +%C% +%C%#CBEACON dest="MORSE-6" info="de MYCALL" +%C% +%C% +%C%# +%C%# Modify for your particular situation before removing +%C%# the # comment character from the beginning of appropriate lines above. +%C%# +%C% +%C% +%C%############################################################# +%C%# # +%C%# APRS DIGIPEATER PROPERTIES # +%C%# # +%C%############################################################# +%C% +%C%# +%C%# For most common situations, use something like this by removing +%C%# the "#" from the beginning of the line below. +%C%# +%C% +%C%#DIGIPEAT 0 0 ^WIDE[3-7]-[1-7]$|^TEST$ ^WIDE[12]-[12]$ +%C% +%C%# See User Guide and "APRS-Digipeaters.pdf" for more explanation of what +%C%# this means and how it can be customized for your particular needs. +%C% +%C% +%C%# Traditional connected mode packet radio uses a different +%C%# type of digipeating. See User Guide for details. +%C% +%C%############################################################# +%C%# # +%C%# INTERNET GATEWAY # +%C%# # +%C%############################################################# +%C% +%C%# First you need to specify the name of a Tier 2 server. +%C%# The current preferred way is to use one of these regional rotate addresses: +%C% +%C%# noam.aprs2.net - for North America +%C%# soam.aprs2.net - for South America +%C%# euro.aprs2.net - for Europe and Africa +%C%# asia.aprs2.net - for Asia +%C%# aunz.aprs2.net - for Oceania +%C% +%C%#IGSERVER noam.aprs2.net +%C% +%C%# You also need to specify your login name and passcode. +%C%# Contact the author if you can't figure out how to generate the passcode. +%C% +%C%#IGLOGIN WB2OSZ-5 123456 +%C% +%C%# That's all you need for a receive only IGate which relays +%C%# messages from the local radio channel to the global servers. +%C% +%C%# Some might want to send an IGate client position directly to a server +%C%# without sending it over the air and relying on someone else to +%C%# forward it to an IGate server. This is done by using sendto=IG rather +%C%# than a radio channel number. Overlay R for receive only, T for two way. +%C%# There is no need to send it as often as you would over the radio. +%C% +%C%#PBEACON sendto=IG delay=0:30 every=60:00 symbol="igate" overlay=R lat=42^37.14N long=071^20.83W +%C%#PBEACON sendto=IG delay=0:30 every=60:00 symbol="igate" overlay=T lat=42^37.14N long=071^20.83W +%C% +%C% +%C%# To relay messages from the Internet to radio, you need to add +%C%# one more option with the transmit channel number and a VIA path. +%C% +%C%#IGTXVIA 0 WIDE1-1,WIDE2-1 +%C% +%C% +%C%# Finally, we don't want to flood the radio channel. +%C%# The IGate function will limit the number of packets transmitted +%C%# during 1 minute and 5 minute intervals. If a limit would +%C%# be exceeded, the packet is dropped and message is displayed in red. +%C%# This might be low for APRS Thursday when there is abnormally high activity. +%C% +%C%IGTXLIMIT 6 10 +%C% +%C% +%C%############################################################# +%C%# # +%C%# APRStt GATEWAY # +%C%# # +%C%############################################################# +%C% +%C%# +%C%# Dire Wolf can receive DTMF (commonly known as Touch Tone) +%C%# messages and convert them to packet objects. +%C%# +%C%# See separate "APRStt-Implementation-Notes" document for details. +%C%# + diff --git a/conf/install_conf.cmake b/conf/install_conf.cmake new file mode 100644 index 00000000..af111a76 --- /dev/null +++ b/conf/install_conf.cmake @@ -0,0 +1,23 @@ +if(NOT EXISTS $ENV{HOME}/direwolf.conf) + configure_file("${CUSTOM_BINARY_DIR}/direwolf.conf" $ENV{HOME}) +endif() + +if(NOT EXISTS $ENV{HOME}/sdr.conf) + configure_file("${CUSTOM_CONF_DIR}/sdr.conf" $ENV{HOME}) +endif() + +if(NOT EXISTS $ENV{HOME}/dw-start.sh) + configure_file("${CUSTOM_SCRIPTS_DIR}/dw-start.sh" $ENV{HOME}) +endif() + +if(NOT EXISTS $ENV{HOME}/telem-m0xer-3.txt) + configure_file("${CUSTOM_TELEMETRY_DIR}/telem-m0xer-3.txt" $ENV{HOME}) +endif() + +if(NOT EXISTS $ENV{HOME}/telem-balloon.conf) + configure_file("${CUSTOM_TELEMETRY_DIR}/telem-balloon.conf" $ENV{HOME}) +endif() + +if(NOT EXISTS $ENV{HOME}/telem-volts.conf) + configure_file("${CUSTOM_TELEMETRY_DIR}/telem-volts.conf" $ENV{HOME}) +endif() diff --git a/sdr.conf b/conf/sdr.conf similarity index 100% rename from sdr.conf rename to conf/sdr.conf diff --git a/data/CMakeLists.txt b/data/CMakeLists.txt new file mode 100644 index 00000000..7972cc23 --- /dev/null +++ b/data/CMakeLists.txt @@ -0,0 +1,41 @@ +# +# Update: 1 May 2023 (still 1.7 dev version) +# +# The original intention was to allow an easy way to download the most +# recent versions of some files. +# +# "update-data" would only work once. +# +# These locations are no longer being maintained: +# http://www.aprs.org/aprs11/tocalls.txt -- 14 Dec 2021 +# http://www.aprs.org/symbols/symbols-new.txt -- 17 Mar 2021 +# http://www.aprs.org/symbols/symbolsX.txt -- 25 Nov 2015 +# so there is no reason to provide a capability grab the latest version. +# +# Rather than fixing an obsolete capability, it will just be removed. +# +# The destination field is often used to identify the manufacturer/model. +# These are not hardcoded into Dire Wolf. Instead they are read from +# a file called tocalls.txt at application start up time. +# +# The original permanent symbols are built in but the "new" symbols, +# using overlays, are often updated. These are also read from files. +# + + +include(ExternalProject) + +set(TOCALLS_TXT "tocalls.txt") +set(SYMBOLS-NEW_TXT "symbols-new.txt") +set(SYMBOLSX_TXT "symbolsX.txt") +set(CUSTOM_BINARY_DATA_DIR "${CMAKE_BINARY_DIR}/data") + +# we can also move to a separate cmake file and use file(download) +# see conf/install_conf.cmake as example +file(COPY "${CUSTOM_DATA_DIR}/${TOCALLS_TXT}" DESTINATION "${CUSTOM_BINARY_DATA_DIR}") +file(COPY "${CUSTOM_DATA_DIR}/${SYMBOLS-NEW_TXT}" DESTINATION "${CUSTOM_BINARY_DATA_DIR}") +file(COPY "${CUSTOM_DATA_DIR}/${SYMBOLSX_TXT}" DESTINATION "${CUSTOM_BINARY_DATA_DIR}") + +install(FILES "${CUSTOM_BINARY_DATA_DIR}/${TOCALLS_TXT}" DESTINATION ${INSTALL_DATA_DIR}) +install(FILES "${CUSTOM_BINARY_DATA_DIR}/${SYMBOLS-NEW_TXT}" DESTINATION ${INSTALL_DATA_DIR}) +install(FILES "${CUSTOM_BINARY_DATA_DIR}/${SYMBOLSX_TXT}" DESTINATION ${INSTALL_DATA_DIR}) diff --git a/symbols-new.txt b/data/symbols-new.txt similarity index 90% rename from symbols-new.txt rename to data/symbols-new.txt index 44e2cba0..aa6fc7d2 100644 --- a/symbols-new.txt +++ b/data/symbols-new.txt @@ -1,11 +1,16 @@ -APRS SYMBOL OVERLAY and EXTENSION TABLES in APRS 1.2 3 Apr 2017 ---------------------------------------------------------------------- - -BACKGROUND: Since 1 October 2007, overlay characters (36 per symbol) -are allowed on all symbols. Since the master symbol document, -http://aprs.org/symbols/symbolsX.txt page only has one line per symbol -character this overlay list gives us thousands of new symbol codes. - +APRS SYMBOL OVERLAY and EXTENSION TABLES in APRS 1.2 17 Mar 2021 +------------------------------------------------------------------------ + +BACKGROUND: Since October 2007, overlay characters (36/symbol) are +allowed on all symbols. This allows thousands of uniquely specified +symbols instead of the original 188 (94 primary and 94 alternate). +But the master symbol document, http://aprs.org/symbols/symbolsX.txt, +only has one line per symbol. So this added overlay list below gives +us thousands of new symbol codes. + +17 Mar 21 Added L& for LORA Igate +24 Jun18: Updated CAR symbols +17 Jun18: Added several overlays for RAIL symbol 03 Apr17: Added Methane Hazard symbol "MH" 13 Feb17: Added Ez = Emergency Power (shelter), Cars: P> = Plugin S> = Solar powered. Moved Ham club C- to Buildings Ch. @@ -54,8 +59,8 @@ of unique overlay definitions: \u - Overlay Trucks. "Tu" is a tanker. "Gu" is a gas truck, etc \< - Advisories may now have overlays \8 - Nodes with overlays. "G8" would be 802.11G -\[ - \[ is wall cloud, but overlays are humans. S[ is a skier. -\h - Buildings. \h is a Ham store, "Hh" is Home Depot, etc. +\[ - \[ was wall cloud, but overlays are humans. S[ is a skier. +\h - Was Ham store, Now Overlays are buildings "Hh" Home Depot, etc. 4 Oct 2007. ORIGINAL EXPANSION to OVERLAYS ON ALL SYMBOLS @@ -94,8 +99,9 @@ small subset of alternate symbols. Those original overlayable alternate symbols were labeled with a "#" and called "numbered" symbols. (UIview requires "No." in the symbols.ini file) -STATUS OF OVERLAYS 1 OCTOBER 2007: the APRS symbol set only had a -few remaining unused symbol codes that had not yet been defined: +STATUS OF OVERLAYS 1 OCTOBER 2007: the APRS symbol set was limited +to about 94 symbols and only had a few remaining unused symbol +codes that had not yet been defined: OF THE 94 Primary Symbols. The following were available: 10 symbols (/0 - /9) that mostly look like billiard balls now @@ -107,7 +113,7 @@ OF THE 94 Alternate Symbols. The following were available: 8 series \1 through \8 that can support 36 overlays each 3 reserved series. -ADDITIONAL OVERLAY PROPOSAL: But any of the other 79 alternate +BASIS FOR OVERLAY EXPANSION: But any of the other 79 alternate symbols could all have multiple (36) overlays if they can make sense with the existing underlying basic symbol that we have been using for that basic alternate symbol. That is, any new definition of a @@ -128,7 +134,7 @@ letting that define a new graphic just for that combination. The following tables will attempt to keep track of these and any other useful generic applications of overlay characters. -AMPLIFIED some existing ALTERNATE SYMBOL Overlays: (new Aug 2014) +AMPLIFIED some existing ALTERNATE SYMBOL Overlays: (Aug 2014) change Flooding #W to include Avalanche, Mudslide/Landslide Update #' name to crash & incident sites Update \D (was available) to DEPOT family @@ -178,6 +184,7 @@ MO = Manned Balloon (2015) TO = Teathered (2015) CO = Constant Pressure - Long duration (2015) RO = Rocket bearing Balloon (Rockoon) (2015) +WO = World-round balloon (2018) BOX SYMBOL: #A (and other system inputted symbols) /A = Aid station @@ -197,21 +204,26 @@ BUILDINGS: #h /h = Hospital \h = Ham Store ** <= now used for HAMFESTS Ch = Club (ham radio) +Eh = Electronics Store Fh = HamFest (new Aug 2014) -Hh = Home Depot etc.. +Hh = Hardware Store etc.. CARS: #> (Vehicles) /> = normal car (side view) \> = Top view and symbol POINTS in direction of travel #> = Reserve overlays 1-9 for numbered cars (new Aug 2014) -B> = Battery (was E for electric) +3> = Model 3 (Tesla) +B> = BEV - Battery EV(was E for electric) +D> = DIY - Do it yourself E> = Ethanol (was electric) F> = Fuelcell or hydrogen -H> = Homemade -P> = Plugin-hybrid +H> = Hybrid +L> = Leaf +P> = PHEV - Plugin-hybrid S> = Solar powered T> = Tesla (temporary) -V> = GM Volt (temporary) +V> = Volt (temporary) +X> = Model X CIVIL DEFENSE or TRIANGLE: #c /c = Incident Command Post @@ -262,6 +274,7 @@ FE = (F overlay) Fog was \{ GATEWAYS: #& /& = HF Gateway <= the original primary table definition I& = Igate Generic (please use more specific overlay) +L& - Lora Igate R& = Receive only IGate (do not send msgs back to RF) P& = PSKmail node T& = TX igate with path set to 1 hop only) @@ -332,19 +345,39 @@ I; = Islands on the air S; = Summits on the air W; = WOTA -POWER or ENERGY: #% +POWER and ENERGY: #% /% = DX cluster <= the original primary table definition C% = Coal E% = Emergency (new Aug 2014) -G% = Geothermal +G% = Gas Turbine H% = Hydroelectric N% = Nuclear P% = Portable (new Aug 2014) R% = Renewable (hydrogen etc fuels) S% = Solar -T% = Turbine +T% = Thermal (geo) W% = Wind +RAIL Symbols: #= +/= = generic train (use steam engine shape for quick recognition) +\= = tbd (use same symbol for now) +B= = Bus-rail/trolley/streetcar/guiderail +C= = Commuter +D= = Diesel +E= = Electric +F= = Freight +G= = Gondola +H= = High Speed Rail (& Hyperloop?) +I= = Inclined Rail +L= = eLevated +M= = Monorail +P= = Passenger +S= = Steam +T= = Terminal (station) +U= = sUbway (& Hyperloop?) +X= = eXcursion + + RESTAURANTS: #R \R = Restaurant (generic) 7R = 7/11 diff --git a/symbolsX.txt b/data/symbolsX.txt similarity index 100% rename from symbolsX.txt rename to data/symbolsX.txt diff --git a/tocalls.txt b/data/tocalls.txt similarity index 58% rename from tocalls.txt rename to data/tocalls.txt index f69b6917..169c9868 100644 --- a/tocalls.txt +++ b/data/tocalls.txt @@ -1,26 +1,40 @@ -APRS TO-CALL VERSION NUMBERS 06 Feb 2017 -------------------------------------------------------------------- - WB4APR -06 Feb 17 added APIExx for W7KMV's PiAPRS system -25 Jan 17 added APSFxx F5OPV embedded devices - was APZ40 -16 Dec 16 added APYSxx for W2GMD's Python APRS -14 Nov 16 Added APINxx for PinPoint by AB0WV -09 Nov 16 added APNICx for SQ5EKU http://sq5eku.blogspot.com/ -24 Oct 16 added APTKPT TrackPoint N0LP, removed APZTKP -24 Aug 16 added APK004 for Kenwood THD-74 -29 Apr 16 added APFPRS for FreeDV by Jeroen PE1RXQ -25 Feb 16 Added APCDS0 for Leon Lessing ZS6LMG's cell tracker -21 Jan 16 added APDNOx for APRSduino by DO3SWW -18 Nov 15 Added APSTPO for N0AGI -03 Nov 15 Updated APAND1 and APDRxx for androids -26 Oct 15 added APZ247 for UPRS -09 Sep 15 added APHTxx for HMTracker by IU0AAC -06 Aug 15 added APMTxx for LZ1PPL for tracker -27 Apr 15 added APZMAJ for Martyn M1MAJ DeLorme inReach Tracker -21 Apr 15 added APB2MF & APR2MF DL2MF - MF2APRS Radiosonde -06 Apr 15 added APAVT5 SainSonic AP510 - a 1watt tracker -. . . . . ... -11 Jan 12 added APYTxx for YagTracker and updated Yaesu APY008/350 + +APRS TO-CALL VERSION NUMBERS 14 Dec 2021 +--------------------------------------------------------------------- + WB4APR + + +07 Jun 23 Added APK005 for Kenwood TH-D75 +14 Dec 21 Added APATAR ATA-R APRS Digipeater by TA7W/OH2UDS and TA6AEU +26 Sep 21 Added APRRDZ EPS32 https://github.com/dl9rdz/rdz_ttgo_sonde +18 Sep 21 Added APCSS for AMSAT Cubesat Simulator https://cubesatsim.org +16 Sep 21 Added APY05D for Yaesu FT5D series +04 Sep 21 APLOxx LoRa KISS TNC/Tracker https://github.com/SQ9MDD/TTGO-T-Beam-LoRa-APRS +24 Aug 21 Added APLSxx SARIMESH http://www.sarimesh.net +22 Aug 21 Added APE2Ax for VA3NNW's Email-2-APRS ap +30 Jun 21 Added APCNxx for carNET by DG5OAW +14 Jun 21 Added APN2xx for NOSaprs JNOS 2.0 - VE4KLM +24 Apr 21 Added APMPAD for DF1JSL's WXBot clone and extension +20 Apr 21 Added APLCxx for APRScube by DL3DCW +19 Apr 21 Added APVMxx for DRCC-DVM Voice (Digital Radio China Club) +13 Apr 21 Added APIxxx for all Dstar ICOMS (APRS via DPRS) +23 MAr 20 Added APW9xx For 9A9Y Weather Tracker +16 Feb 21 Added API970 for I com 9700 + +2020 Added APHBLx,APIZCI,APLGxx,APLTxx,APNVxx,APY300,APESPG,APESPW + APGDTx,APOSWx,APOSBx,APBT62,APCLUB,APMQxx +2019 Added APTPNx,APJ8xx,APBSDx,APNKMX,APAT51,APMGxx,APTCMA, + APATxx,APQTHx,APLIGx +2018 added APRARX,APELKx,APGBLN,APBKxx,APERSx,APTCHE +2017 Added APHWxx,APDVxx,APPICO,APBMxx,APP6xx,APTAxx,APOCSG,APCSMS, + APPMxx,APOFF,APDTMF,APRSON,APDIGI,APSAT,APTBxx,APIExx, + APSFxx +2016 added APYSxx,APINxx,APNICx,APTKPT,APK004,APFPRS,APCDS0,APDNOx +2015 Added APSTPO,APAND1,APDRxx,APZ247,APHTxx,APMTxx,APZMAJ + APB2MF,APR2MF,APAVT5 + + + In APRS, the AX.25 Destination address is not used for packet routing as is normally done in AX.25. So APRS uses it for two @@ -31,6 +45,13 @@ bytes of the field are available to indicate the software version number or application. The following applications have requested a TOCALL number series: +Authors with similar alphabetic requirements are encouraged to share +their address space with other software. Work out agreements amongst +yourselves and keep me informed. + + + + APn 3rd digit is a number AP1WWX TAPR T-238+ WX station AP1MAJ Martyn M1MAJ DeLorme inReach Tracker @@ -45,19 +66,30 @@ a TOCALL number series: APAHxx AHub APAND1 APRSdroid (pre-release) http://aprsdroid.org/ APAMxx Altus Metrum GPS trackers + APATAR ATA-R APRS Digipeater by TA7W/OH2UDS and TA6AEU + APAT8x for Anytone. 81 for 878 HT + APAT51 for Anytone AT-D578UV APRS mobile radio APAVT5 SainSonic AP510 which is a 1watt tracker APAWxx AGWPE APB APBxxx Beacons or Rabbit TCPIP micros? APB2MF DL2MF - MF2APRS Radiosonde for balloons APBLxx BigRedBee BeeLine APBLO MOdel Rocketry K7RKT - APBPQx John G8BPQ Digipeater/IGate + APBKxx PY5BK Bravo Tracker in Brazil + APBPQx John G8BPQ Digipeater/IGate + APBMxx BrandMeister DMR Server for R3ABM + APBSDx HamBSD https://hambsd.org/ + APBT62 BTech DMR 6x2 APC APCxxx Cellular applications APCBBx VE7UDP Blackberry Applications APCDS0 Leon Lessing ZS6LMG's cell tracker APCLEY EYTraker GPRS/GSM tracker by ZS6EY - APCLWX EYWeather GPRS/GSM WX station by ZS6EY APCLEZ Telit EZ10 GSM application ZS6CEY + APCLUB Brazil APRS network + APCLWX EYWeather GPRS/GSM WX station by ZS6EY + APCNxx for carNET by DG5OAW + APCSMS for Cosmos (used for sending commands @USNA) + APCSS for AMSAT cubesats https://cubesatsim.org APCWP8 John GM7HHB, WinphoneAPRS APCYxx Cybiko applications APD APD4xx UP4DAR platform @@ -66,6 +98,8 @@ a TOCALL number series: APDGxx D-Star Gateways by G4KLX ircDDB APDHxx WinDV (DUTCH*Star DV Node for Windows) APDInn DIXPRS - Bela, HA5DI + APDIGI Used by PSAT2 to indicate the digi is ON + APDIGI digi ON for PSAT2 and QIKCOM-2 APDKxx KI4LKF g2_ircddb Dstar gateway software APDNOx APRSduino by DO3SWW APDOxx ON8JL Standalone DStar Node @@ -73,25 +107,49 @@ a TOCALL number series: APDRxx APRSdroid Android App http://aprsdroid.org/ APDSXX SP9UOB for dsDigi and ds-tracker APDTxx APRStouch Tone (DTMF) + APDTMF digi off mode on QIKCOM2 and DTMF ON APDUxx U2APRS by JA7UDE + APDVxx OE6PLD's SSTV with APRS status exchange APDWxx DireWolf, WB2OSZ APE APExxx Telemetry devices + APE2Ax VA3NNW's Email-2-APRS ap APECAN Pecan Pico APRS Balloon Tracker + APELKx WB8ELK balloons APERXQ Experimental tracker by PE1RXQ + APERSx Runner tracking by Jason,KG7YKZ + APESPG ESP SmartBeacon APRS-IS Client + APESPW ESP Weather Station APRS-IS Client APF APFxxx Firenet APFGxx Flood Gage (KP4DJT) APFIxx for APRS.FI OH7LZB, Hessu APFPRS for FreeDV by Jeroen PE1RXQ APG APGxxx Gates, etc APGOxx for AA3NJ PDA application + APGBLN for NW5W's GoBalloon + APGDTx for VK4FAST's Graphic Data Terminal APH APHKxx for LA1BR tracker/digipeater APHAXn SM2APRS by PY2UEP + APHBLx for DMR Gateway by Eric - KF7EEL APHTxx HMTracker by IU0AAC - API APICQx for ICQ + APHWxx for use in "HamWAN + API API282 for ICOM IC-2820 + API31 for ICOM ID-31 + API410 for ICOM ID-4100 + API51 for ICOM ID-51 + API510 for ICOM ID-5100 + API710 for ICOM IC-7100 + API80 for ICOM IC-80 + API880 for ICOM ID-880 + API910 for ICOM IC-9100 + API92 for ICOM IC-92 + API970 for ICOM 9700 + APICQx for ICQ APICxx HA9MCQ's Pic IGate APIExx W7KMV's PiAPRS system APINxx PinPoint by AB0WV - APJ APJAxx JavAPRS + APIZCI hymTR IZCI Tracker by TA7W/OH2UDS and TA6AEU + APJ APJ8xx Jordan / KN4CRD JS8Call application + APJAxx JavAPRS APJExx JeAPRS APJIxx jAPRSIgate APJSxx javAPRSSrvr @@ -99,16 +157,26 @@ a TOCALL number series: APK APK0xx Kenwood TH-D7's APK003 Kenwood TH-D72 APK004 Kenwood TH-D74 + APK005 Kenwood TH-D75 APK1xx Kenwood D700's APK102 Kenwood D710 APKRAM KRAMstuff.com - Mark. G7LEU - APL APLQRU Charlie - QRU Server + APL APLCxx APRScube by DL3DCW + APLGxx LoRa Gateway/Digipeater OE5BPA + APLIGx LightAPRS - TA2MUN and TA9OHC + APLOxx LoRa KISS TNC/Tracker + APLQRU Charlie - QRU Server APLMxx WA0TQG transceiver controller + APLSxx SARIMESH ( http://www.sarimesh.net ) + APLTxx LoRa Tracker - OE5BPA APM APMxxx MacAPRS, - APMGxx MiniGate - Alex, AB0TJ + APMGxx PiCrumbs and MiniGate - Alex, AB0TJ APMIxx SQ3PLX http://microsat.com.pl/ + APMPAD DF1JSL's WXBot clone and extension + APMQxx Ham Radio of Things WB2OSZ APMTxx LZ1PPL for tracker APN APNxxx Network nodes, digis, etc + APN2xx NOSaprs for JNOS 2.0 - VE4KLM APN3xx Kantronics KPC-3 rom versions APN9xx Kantronics KPC-9612 Roms APNAxx WB6ZSU's APRServe @@ -117,23 +185,34 @@ a TOCALL number series: APNK01 Kenwood D700 (APK101) type APNK80 KAM version 8.0 APNKMP KAM+ + APNKMX KAM-XL APNMxx MJF TNC roms APNPxx Paccom TNC roms APNTxx SV2AGW's TNT tnc as a digi APNUxx UIdigi + APNVxx SQ8L's VP digi and Nodes APNXxx TNC-X (K6DBG) APNWxx SQ3FYK.com WX/Digi and SQ3PLX http://microsat.com.pl/ APO APRSpoint + APOFF Used by PSAT and PSAT2 to indicate the digi is OFF APOLUx for OSCAR satellites for AMSAT-LU by LU9DO APOAxx OpenAPRS - Greg Carter - APOTxx Open Track + APOCSG For N0AGI's APRS to POCSAG project APOD1w Open Track with 1 wire WX + APOSBx openSPOT3 by HA2NON at sharkrf.com + APOSWx openSPOT2 + APOTxx Open Track APOU2k Open Track for Ultimeter APOZxx www.KissOZ.dk Tracker. OZ1EKD and OZ7HVO - APP APPTxx KetaiTracker by JF6LZE, Takeki (msg capable) + APP APP6xx for APRSlib + APPICx DB1NTO' PicoAPRS + APPMxx DL1MX's RTL-SDR pytohon Igate + APPTxx KetaiTracker by JF6LZE, Takeki (msg capable) APQ APQxxx Earthquake data + APQTHx W8WJB's QTH.app APR APR8xx APRSdos versions 800+ APR2MF DL2MF - MF2APRS Radiosonde WX reporting + APRARX VK5QI's radiosonde tracking APRDxx APRSdata, APRSdr APRGxx aprsg igate software, OH2GVE APRHH2 HamHud 2 @@ -141,6 +220,7 @@ a TOCALL number series: APRNOW W5GGW ipad application APRRTx RPC electronics APRS Generic, (obsolete. Digis should use APNxxx instead) + APRSON Used by PSAT to indicate the DIGI is ON APRXxx >40 APRSmax APRXxx <39 for OH2MQK's igate APRTLM used in MIM's and Mic-lites, etc @@ -148,6 +228,7 @@ a TOCALL number series: APRSTx APRStt (Touch tone) APS APSxxx APRS+SA, etc APSARx ZL4FOX's SARTRACK + APSAT digi ON for QIKCOM-1 APSCxx aprsc APRS-IS core server (OH7LZB, OH2MQK) APSFxx F5OPV embedded devices - was APZ40 APSK63 APRS Messenger -over-PSK63 @@ -158,8 +239,12 @@ a TOCALL number series: APT APT2xx Tiny Track II APT3xx Tiny Track III APTAxx K4ATM's tiny track + APTBxx TinyAPRS by BG5HHP Was APTAxx till Sep 2017 + APTCHE PU3IKE in Brazil TcheTracker/Tcheduino + APTCMA CAPI tracker - PU1CMA Brazil APTIGR TigerTrack APTKPT TrackPoint N0LP + APTPNx TARPN Packet Node Tracker by KN4ORB http://tarpn.net/ APTTxx Tiny Track APTWxx Byons WXTrac APTVxx for ATV/APRN and SSTV applications @@ -168,38 +253,47 @@ a TOCALL number series: APU3xx UIview terminal program APUDRx NW Digital Radio's UDR (APRS/Dstar) APV APVxxx Voice over Internet applications + APVMxx DRCC-DVM Digital Voice (Digital Radio China Club) APVRxx for IRLP APVLxx for I-LINK APVExx for ECHO link APW APWxxx WinAPRS, etc + APW9xx 9A9Y Weather Tracker APWAxx APRSISCE Android version APWSxx DF4IAN's WS2300 WX station APWMxx APRSISCE KJ4ERJ APWWxx APRSISCE win32 version APX APXnnn Xastir APXRnn Xrouter - APY APYxxx Yeasu + APY APYxxx Yaesu Radios APY008 Yaesu VX-8 series - APY350 Yaesu FTM-350 series - APYTxx for YagTracker - APYSxx for W2GMD's Python APRS + APY01D Yaesu FT1D series + APY02D Yaesu FT2D series + APY03D Yaesu FT3D series + APY05D Yaesu FT5D series + APY100 Yaesu FTM-100D series + APY300 Yaesu FTM-300D series + APY350 Yaesu FTM-350 series + APY400 Yaesu FTM-400D series APZ APZxxx Experimental + APZ200 old versions of JNOS APZ247 for UPRS NR0Q APZ0xx Xastir (old versions. See APX) APZMAJ Martyn M1MAJ DeLorme inReach Tracker + APZMDM github/codec2_talkie - product code not registered APZMDR for HaMDR trackers - hessu * hes.iki.fi] APZPAD Smart Palm APZTKP TrackPoint, Nick N0LP (Balloon tracking)(depricated) APZWIT MAP27 radio (Mountain Rescue) EI7IG APZWKR GM1WKR NetSked application + + -Authors with similar alphabetic requirements are encouraged to share -their address space with other software. Work out agreements amongst -yourselves and keep me informed. - + + -REGISTERED ALTNETS: -------------------- +REGISTERED TOCALL ALTNETS: +-------------------------- ALTNETS are uses of the AX-25 tocall to distinguish specialized traffic that may be flowing on the APRS-IS, but that are not intended @@ -222,9 +316,11 @@ The following is a list of ALTNETS that may be of interest to other users. This list is by no means complete, since ANY combination of characters other than APxxxx are considered an ALTNET. But this list can give consisntecy to ALTNETS that may be using the global APRS-IS -and need some special recognition: - - TEST - A generic ALTNET for use during testing - PSKAPR - PSKmail . But it is not AX.25 anyway +and need some special recognition. Here are some ideas: + + -de WB4APR, Bob + SATERN - Salvation Army Altnet + AFMARS - Airforce Mars + AMARS - Army Mars + \ No newline at end of file diff --git a/debian/README.Debian b/debian/README.Debian new file mode 100644 index 00000000..853f55f0 --- /dev/null +++ b/debian/README.Debian @@ -0,0 +1,5 @@ +In order to start direwolf as a service the configuration file +/etc/direwolf.conf needs to exist. Otherwise attempting to start the service +returns an 'Assertion failed' error. An example configuration file which may be +used as a model can be found in +/usr/share/doc/direwolf/examples/direwolf.conf.gz diff --git a/debian/changelog b/debian/changelog new file mode 120000 index 00000000..cf547089 --- /dev/null +++ b/debian/changelog @@ -0,0 +1 @@ +../CHANGES.md \ No newline at end of file diff --git a/debian/compat b/debian/compat new file mode 100644 index 00000000..9a037142 --- /dev/null +++ b/debian/compat @@ -0,0 +1 @@ +10 \ No newline at end of file diff --git a/debian/control b/debian/control new file mode 100644 index 00000000..106663b5 --- /dev/null +++ b/debian/control @@ -0,0 +1,30 @@ +Source: direwolf +Maintainer: Debian Hamradio Maintainers +Uploaders: Iain R. Learmonth +Section: hamradio +Priority: optional +Build-Depends: debhelper (>= 9), + libasound2-dev, + libgps-dev, + libhamlib-dev, + dh-systemd +Standards-Version: 4.1.0 +Vcs-Browser: https://anonscm.debian.org/cgit/pkg-hamradio/direwolf.git/ +Vcs-Git: https://anonscm.debian.org/git/pkg-hamradio/direwolf.git +Homepage: https://github.com/wb2osz/direwolf + +Package: direwolf +Architecture: alpha amd64 arm64 armel armhf i386 mipsel ppc64el sh4 x32 +Depends: ${shlibs:Depends}, + ${misc:Depends}, + adduser, + libhamlib2 +Suggests: gpsd, libhamlib-utils +Breaks: direwolf-docs (<< 1.1-1) +Replaces: direwolf-docs (<< 1.1-1) +Description: Soundcard TNC for APRS + Dire Wolf is a software "soundcard" modem/TNC and APRS encoder/decoder. It can + be used stand-alone to receive APRS messages, as a digipeater, APRStt gateway, + or Internet Gateway (IGate). It can also be used as a virtual TNC for other + applications such as APRSIS32, UI-View32, Xastir, APRS-TW, YAAC, UISS, Linux + AX25, SARTrack, and many others. \ No newline at end of file diff --git a/debian/copyright b/debian/copyright new file mode 100644 index 00000000..b546bf71 --- /dev/null +++ b/debian/copyright @@ -0,0 +1,176 @@ +Format: http://www.debian.org/doc/packaging-manuals/copyright-format/1.0/ +Upstream-Name: direwolf +Files-Excluded: doc/*.pdf +Source: https://github.com/wb2osz/direwolf +Comment: + The files in misc/ are copied directly from the Cygwin source code. These are + listed here as dual licensed as they are both part of the Cygwin distribution + and originally part of BSD. See misc/README-dire-wolf.txt for more information. + . + Please see ftp-master's comments on this here: + https://lists.debian.org/debian-hams/2014/09/msg00063.html + https://lists.debian.org/debian-hams/2014/10/msg00003.html + +Files: * +Copyright: (C) 2011-2014 John Langner WB2OSZ +License: GPL-2+ + +Files: geotranz/* +Copyright: National Geospatial-Intelligence Agency +License: Permissive-NGA + +Files: regex/* +Copyright: (C) 2002, 2003, 2005 Free Software Foundation, Inc. +License: LGPL-2.1+ + +Files: misc/strcasestr.c +Copyright: + (C) 1990, 1993 The Regents of the University of California + (C) RedHat +License: BSD-4-clause or GPL-2+ + +Files: misc/strtok_r.c misc/strsep.c +Copyright: + (C) 1988 Regents of the University of California + (C) RedHat +License: BSD-3-clause or GPL-2+ + +Files: debian/* +Copyright: (C) 2014 Iain R. Learmonth +License: GPL-2+ + +License: BSD-3-clause + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions + are met: + . + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + . + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + . + 3. Neither the name of the University nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + . + THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + SUCH DAMAGE. + +License: BSD-4-clause + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions + are met: + . + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + . + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + . + 3. All advertising materials mentioning features or use of this software + must display the following acknowledgement: + This product includes software developed by the University of + California, Berkeley and its contributors. + . + 4. Neither the name of the University nor the names of its contributors + may be used to endorse or promote products derived from this software + without specific prior written permission. + . + THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + SUCH DAMAGE. + +License: GPL-2+ + 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, see . + . + On Debian systems, a copy of the full license text is available in + /usr/share/common-licenses/GPL-2. + +License: LGPL-2.1+ + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Lesser General Public + License as published by the Free Software Foundation; either + version 2.1 of the License, or (at your option) any later version. + . + This library 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 + Lesser General Public License for more details. + . + You should have received a copy of the GNU Lesser General Public + License along with this library; if not, write to the Free Software + Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + . + On Debian systems, a copy of the full license text is available in + /usr/share/common-licenses/LGPL-2.1. + +License: Permissive-NGA + 1. The GEOTRANS source code ("the software") is provided free of charge by the + National Geospatial-Intelligence Agency (NGA) of the United States Department + of Defense. Although NGA makes no copyright claim under Title 17 U.S.C., NGA + claims copyrights in the source code under other legal regimes. NGA hereby + grants to each user of the software a license to use and distribute the + software, and develop derivative works. + . + 2. NGA requests that products developed using the software credit the source of + the software with the following statement, "The product was developed using + GEOTRANS, a product of the National Geospatial-Intelligence Agency (NGA) and + U.S. Army Engineering Research and Development Center." Do not use the name + GEOTRANS for any derived work. + . + 3. Warranty Disclaimer: The software was developed to meet only the internal + requirements of the National Geospatial-Intelligence Agency (NGA). The software + is provided "as is," and no warranty, express or implied, including but not + limited to the implied warranties of merchantability and fitness for particular + purpose or arising by statute or otherwise in law or from a course of dealing + or usage in trade, is made by NGA as to the accuracy and functioning of the + software. + . + 4. NGA and its personnel are not required to provide technical support or + general assistance with respect to public use of the software. Government + customers may contact NGA. + . + 5. Neither NGA nor its personnel will be liable for any claims, losses, or + damages arising from or connected with the use of the software. The user agrees + to hold harmless the United States National Geospatial-Intelligence Agency + (NGA). The user's sole and exclusive remedy is to stop using the software. + . + 6. Please be advised that pursuant to the United States Code, 10 U.S.C. 425, + the name of the National Geospatial-Intelligence Agency, the initials "NGA", + the seal of the National Geospatial-Intelligence Agency, or any colorable + imitation thereof shall not be used to imply approval, endorsement, or + authorization of a product without prior written permission from United States + Secretary of Defense. Do not create the impression that NGA, the Secretary of + Defense or the Director of National Intelligence has endorsed any product + derived from GEOTRANS. \ No newline at end of file diff --git a/debian/direwolf.postinst b/debian/direwolf.postinst new file mode 100644 index 00000000..e42b9f83 --- /dev/null +++ b/debian/direwolf.postinst @@ -0,0 +1,33 @@ +#!/bin/sh + +set -e + +. /usr/share/debconf/confmodule + +add_group_if_missing() { + if ! getent group direwolf >/dev/null; then + addgroup --system --force-badname direwolf || true + fi +} + +add_user_if_missing() { + if ! id -u direwolf > /dev/null 2>&1; then + mkdir -m 02750 -p /var/lib/direwolf + adduser --system --home /var/lib/direwolf \ + --disabled-password \ + --force-badname direwolf \ + --ingroup direwolf + adduser direwolf dialout + chown direwolf:direwolf /var/lib/direwolf + fi +} + +add_group_if_missing +add_user_if_missing + +db_stop + +#DEBHELPER# + +exit 0 + diff --git a/debian/direwolf.postrm b/debian/direwolf.postrm new file mode 100644 index 00000000..886af3d2 --- /dev/null +++ b/debian/direwolf.postrm @@ -0,0 +1,19 @@ +#!/bin/sh + +set -e + +case "$1" in + purge) + rm -rf /var/lib/direwolf/ + ;; + remove|upgrade|failed-upgrade|abort-install|abort-upgrade|disappear) + ;; + *) + echo "postrm called with unknown argument \`$1'" >&2 + exit 1 +esac + +#DEBHELPER# + +exit 0 + diff --git a/debian/rules b/debian/rules new file mode 100644 index 00000000..b8c22228 --- /dev/null +++ b/debian/rules @@ -0,0 +1,7 @@ +#!/usr/bin/make -f + +%: + dh $@ --parallel + +override_dh_auto_configure: + dh_auto_configure -- -DFORCE_SSE=1 diff --git a/debian/source/format b/debian/source/format new file mode 100644 index 00000000..46ebe026 --- /dev/null +++ b/debian/source/format @@ -0,0 +1 @@ +3.0 (quilt) \ No newline at end of file diff --git a/demod_afsk.c b/demod_afsk.c deleted file mode 100644 index e8252db9..00000000 --- a/demod_afsk.c +++ /dev/null @@ -1,1158 +0,0 @@ -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2011, 2012, 2013, 2014, 2015 John Langner, WB2OSZ -// -// 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, see . -// - - -// #define DEBUG1 1 /* display debugging info */ - -// #define DEBUG3 1 /* print carrier detect changes. */ - -// #define DEBUG4 1 /* capture AFSK demodulator output to log files */ - -// #define DEBUG5 1 /* capture 9600 output to log files */ - - -/*------------------------------------------------------------------ - * - * Module: demod_afsk.c - * - * Purpose: Demodulator for Audio Frequency Shift Keying (AFSK). - * - * Input: Audio samples from either a file or the "sound card." - * - * Outputs: Calls hdlc_rec_bit() for each bit demodulated. - * - *---------------------------------------------------------------*/ - -#include "direwolf.h" - -#include -#include -#include -#include -#include -#include -#include -#include - -#include "audio.h" -#include "tune.h" -#include "fsk_demod_state.h" -#include "fsk_gen_filter.h" -#include "hdlc_rec.h" -#include "textcolor.h" -#include "demod_afsk.h" -#include "dsp.h" - -#define MIN(a,b) ((a)<(b)?(a):(b)) -#define MAX(a,b) ((a)>(b)?(a):(b)) - - -#ifndef GEN_FFF - -/* Quick approximation to sqrt(x*x+y*y) */ -/* No benefit for regular PC. */ -/* Should help with microcomputer platform. */ - - -__attribute__((hot)) __attribute__((always_inline)) -static inline float z (float x, float y) -{ - x = fabsf(x); - y = fabsf(y); - - if (x > y) { - return (x * .941246f + y * .41f); - } - else { - return (y * .941246f + x * .41f); - } -} - -/* Add sample to buffer and shift the rest down. */ - -__attribute__((hot)) __attribute__((always_inline)) -static inline void push_sample (float val, float *buff, int size) -{ - memmove(buff+1,buff,(size-1)*sizeof(float)); - buff[0] = val; -} - - -/* FIR filter kernel. */ - -__attribute__((hot)) __attribute__((always_inline)) -static inline float convolve (const float *__restrict__ data, const float *__restrict__ filter, int filter_size) -{ - float sum = 0.0f; - int j; - - -//#pragma GCC ivdep // ignored until gcc 4.9 - for (j=0; j= *ppeak) { - *ppeak = in * fast_attack + *ppeak * (1.0f - fast_attack); - } - else { - *ppeak = in * slow_decay + *ppeak * (1.0f - slow_decay); - } - - if (in <= *pvalley) { - *pvalley = in * fast_attack + *pvalley * (1.0f - fast_attack); - } - else { - *pvalley = in * slow_decay + *pvalley * (1.0f - slow_decay); - } - - if (*ppeak > *pvalley) { - return ((in - 0.5f * (*ppeak + *pvalley)) / (*ppeak - *pvalley)); - } - return (0.0f); -} - -#endif // ifndef GEN_FFF - - -/* - * for multi-slicer experiment. - */ - -#define MIN_G 0.5f -#define MAX_G 4.0f - -/* TODO: static */ float space_gain[MAX_SUBCHANS]; - - - -/*------------------------------------------------------------------ - * - * Name: demod_afsk_init - * - * Purpose: Initialization for an AFSK demodulator. - * Select appropriate parameters and set up filters. - * - * Inputs: samples_per_sec - * baud - * mark_freq - * space_freq - * - * D - Pointer to demodulator state for given channel. - * - * Outputs: D->ms_filter_size - * D->m_sin_table[] - * D->m_cos_table[] - * D->s_sin_table[] - * D->s_cos_table[] - * - * Returns: None. - * - * Bugs: This doesn't do much error checking so don't give it - * anything crazy. - * - *----------------------------------------------------------------*/ - -void demod_afsk_init (int samples_per_sec, int baud, int mark_freq, - int space_freq, char profile, struct demodulator_state_s *D) -{ - - int j; - - memset (D, 0, sizeof(struct demodulator_state_s)); - D->num_slicers = 1; - -#if DEBUG1 - dw_printf ("demod_afsk_init (rate=%d, baud=%d, mark=%d, space=%d, profile=%c\n", - samples_per_sec, baud, mark_freq, space_freq, profile); -#endif - -#ifdef TUNE_PROFILE - profile = TUNE_PROFILE; -#endif - - - if (profile == 'F') { - - if (baud != DEFAULT_BAUD || - mark_freq != DEFAULT_MARK_FREQ || - space_freq!= DEFAULT_SPACE_FREQ || - samples_per_sec != DEFAULT_SAMPLES_PER_SEC) { - - text_color_set(DW_COLOR_INFO); - dw_printf ("Note: Decoder 'F' works only for %d baud, %d/%d tones, %d samples/sec.\n", - DEFAULT_BAUD, DEFAULT_MARK_FREQ, DEFAULT_SPACE_FREQ, DEFAULT_SAMPLES_PER_SEC); - dw_printf ("Using Decoder 'A' instead.\n"); - profile = 'A'; - } - } - - D->profile = profile; // so we know whether to take fast path later. - - switch (profile) { - - case 'A': - case 'F': - - /* Original. 52 taps, truncated bandpass, IIR lowpass */ - /* 'F' is the fast version for low end processors. */ - /* It is a special case that works only for a particular */ - /* baud rate, tone pair, and sampling rate. */ - - D->use_prefilter = 0; - - D->ms_filter_len_bits = 1.415; /* 52 @ 44100, 1200 */ - D->ms_window = BP_WINDOW_TRUNCATED; - - //D->bp_window = BP_WINDOW_TRUNCATED; - - D->lpf_use_fir = 0; - D->lpf_iir = 0.195; - - D->agc_fast_attack = 0.250; - D->agc_slow_decay = 0.00012; - D->hysteresis = 0.005; - - D->pll_locked_inertia = 0.700; - D->pll_searching_inertia = 0.580; - break; - - case 'B': - - /* Original bandpass. Use FIR lowpass instead. */ - - D->use_prefilter = 0; - - D->ms_filter_len_bits = 1.415; /* 52 @ 44100, 1200 */ - D->ms_window = BP_WINDOW_TRUNCATED; - - //D->bp_window = BP_WINDOW_TRUNCATED; - - D->lpf_use_fir = 1; - D->lpf_baud = 1.09; - D->lp_filter_len_bits = D->ms_filter_len_bits; - D->lp_window = BP_WINDOW_TRUNCATED; - - D->agc_fast_attack = 0.370; - D->agc_slow_decay = 0.00014; - D->hysteresis = 0.003; - - D->pll_locked_inertia = 0.620; - D->pll_searching_inertia = 0.350; - break; - - case 'C': - - /* Cosine window, 76 taps for bandpass, FIR lowpass. */ - - D->use_prefilter = 0; - - D->ms_filter_len_bits = 2.068; /* 76 @ 44100, 1200 */ - D->ms_window = BP_WINDOW_COSINE; - - //D->bp_window = BP_WINDOW_COSINE; - - D->lpf_use_fir = 1; - D->lpf_baud = 1.09; - D->lp_filter_len_bits = D->ms_filter_len_bits; - D->lp_window = BP_WINDOW_TRUNCATED; - - D->agc_fast_attack = 0.495; - D->agc_slow_decay = 0.00022; - D->hysteresis = 0.005; - - D->pll_locked_inertia = 0.620; - D->pll_searching_inertia = 0.350; - break; - - case 'D': - - /* Prefilter, Cosine window, FIR lowpass. Tweeked for 300 baud. */ - - D->use_prefilter = 1; /* first, a bandpass filter. */ - D->prefilter_baud = 0.87; - D->pre_filter_len_bits = 1.857; - D->pre_window = BP_WINDOW_COSINE; - - D->ms_filter_len_bits = 1.857; /* 91 @ 44100/3, 300 */ - D->ms_window = BP_WINDOW_COSINE; - - //D->bp_window = BP_WINDOW_COSINE; - - D->lpf_use_fir = 1; - D->lpf_baud = 1.10; - D->lp_filter_len_bits = D->ms_filter_len_bits; - D->lp_window = BP_WINDOW_TRUNCATED; - - D->agc_fast_attack = 0.495; - D->agc_slow_decay = 0.00022; - D->hysteresis = 0.027; - - D->pll_locked_inertia = 0.620; - D->pll_searching_inertia = 0.350; - break; - - case 'E': - - /* 1200 baud - Started out similar to C but add prefilter. */ - /* Version 1.2 */ - /* Enhancements: */ - /* + Add prefilter. Previously used for 300 baud D, but not 1200. */ - /* + Prefilter length now independent of M/S filters. */ - /* + Lowpass filter length now independent of M/S filters. */ - /* + Allow mixed window types. */ - - //D->bp_window = BP_WINDOW_COSINE; /* The name says BP but it is used for all of them. */ - - D->use_prefilter = 1; /* first, a bandpass filter. */ - D->prefilter_baud = 0.23; - D->pre_filter_len_bits = 156 * 1200. / 44100.; - D->pre_window = BP_WINDOW_TRUNCATED; - - D->ms_filter_len_bits = 74 * 1200. / 44100.; - D->ms_window = BP_WINDOW_COSINE; - - D->lpf_use_fir = 1; - D->lpf_baud = 1.18; - D->lp_filter_len_bits = 63 * 1200. / 44100.; - D->lp_window = BP_WINDOW_TRUNCATED; - - //D->agc_fast_attack = 0.300; - //D->agc_slow_decay = 0.000185; - D->agc_fast_attack = 0.820; - D->agc_slow_decay = 0.000214; - D->hysteresis = 0.01; - - //D->pll_locked_inertia = 0.57; - //D->pll_searching_inertia = 0.33; - D->pll_locked_inertia = 0.74; - D->pll_searching_inertia = 0.50; - break; - - case 'G': - - /* 1200 baud - Started out same as E but add 3 way interleave. */ - /* Version 1.3 - EXPERIMENTAL - Needs more fine tuning. */ - - //D->bp_window = BP_WINDOW_COSINE; /* The name says BP but it is used for all of them. */ - - D->use_prefilter = 1; /* first, a bandpass filter. */ - D->prefilter_baud = 0.15; - D->pre_filter_len_bits = 128 * 1200. / (44100. / 3.); - D->pre_window = BP_WINDOW_TRUNCATED; - - D->ms_filter_len_bits = 25 * 1200. / (44100. / 3.); - D->ms_window = BP_WINDOW_COSINE; - - D->lpf_use_fir = 1; - D->lpf_baud = 1.16; - D->lp_filter_len_bits = 21 * 1200. / (44100. / 3.); - D->lp_window = BP_WINDOW_TRUNCATED; - - D->agc_fast_attack = 0.130; - D->agc_slow_decay = 0.00013; - D->hysteresis = 0.01; - - D->pll_locked_inertia = 0.73; - D->pll_searching_inertia = 0.64; - break; - - default: - - text_color_set(DW_COLOR_ERROR); - dw_printf ("Invalid filter profile = %c\n", profile); - exit (1); - } - -#ifdef TUNE_PRE_WINDOW - D->pre_window = TUNE_PRE_WINDOW; -#endif -#ifdef TUNE_MS_WINDOW - D->ms_window = TUNE_MS_WINDOW; -#endif -#ifdef TUNE_LP_WINDOW - D->lp_window = TUNE_LP_WINDOW; -#endif - - -#if defined(TUNE_AGC_FAST) && defined(TUNE_AGC_SLOW) - D->agc_fast_attack = TUNE_AGC_FAST; - D->agc_slow_decay = TUNE_AGC_SLOW; -#endif -#ifdef TUNE_HYST - D->hysteresis = TUNE_HYST; -#endif -#if defined(TUNE_PLL_LOCKED) && defined(TUNE_PLL_SEARCHING) - D->pll_locked_inertia = TUNE_PLL_LOCKED; - D->pll_searching_inertia = TUNE_PLL_SEARCHING; -#endif -#ifdef TUNE_LPF_BAUD - D->lpf_baud = TUNE_LPF_BAUD; -#endif -#ifdef TUNE_PRE_BAUD - D->prefilter_baud = TUNE_PRE_BAUD; -#endif - - -/* - * Calculate constants used for timing. - * The audio sample rate must be at least a few times the data rate. - */ - - D->pll_step_per_sample = (int) round((TICKS_PER_PLL_CYCLE * (double)baud) / ((double)samples_per_sec)); - -/* - * Convert number of bit times to number of taps. - */ - - D->pre_filter_size = (int) round( D->pre_filter_len_bits * (float)samples_per_sec / (float)baud ); - D->ms_filter_size = (int) round( D->ms_filter_len_bits * (float)samples_per_sec / (float)baud ); - D->lp_filter_size = (int) round( D->lp_filter_len_bits * (float)samples_per_sec / (float)baud ); - -/* Experiment with other sizes. */ - -#ifdef TUNE_PRE_FILTER_SIZE - D->pre_filter_size = TUNE_PRE_FILTER_SIZE; -#endif -#ifdef TUNE_MS_FILTER_SIZE - D->ms_filter_size = TUNE_MS_FILTER_SIZE; -#endif -#ifdef TUNE_LP_FILTER_SIZE - D->lp_filter_size = TUNE_LP_FILTER_SIZE; -#endif - - //assert (D->pre_filter_size >= 4); - assert (D->ms_filter_size >= 4); - //assert (D->lp_filter_size >= 4); - - if (D->pre_filter_size > MAX_FILTER_SIZE) - { - text_color_set (DW_COLOR_ERROR); - dw_printf ("Calculated filter size of %d is too large.\n", D->pre_filter_size); - dw_printf ("Decrease the audio sample rate or increase the baud rate or\n"); - dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", - MAX_FILTER_SIZE); - exit (1); - } - - if (D->ms_filter_size > MAX_FILTER_SIZE) - { - text_color_set (DW_COLOR_ERROR); - dw_printf ("Calculated filter size of %d is too large.\n", D->ms_filter_size); - dw_printf ("Decrease the audio sample rate or increase the baud rate or\n"); - dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", - MAX_FILTER_SIZE); - exit (1); - } - - if (D->lp_filter_size > MAX_FILTER_SIZE) - { - text_color_set (DW_COLOR_ERROR); - dw_printf ("Calculated filter size of %d is too large.\n", D->pre_filter_size); - dw_printf ("Decrease the audio sample rate or increase the baud rate or\n"); - dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", - MAX_FILTER_SIZE); - exit (1); - } - -/* - * Optionally apply a bandpass ("pre") filter to attenuate - * frequencies outside the range of interest. - * This was first used for the "D" profile for 300 baud - * which uses narrow shift. We expect it to have significant - * benefit for a narrow shift. - * In version 1.2, we will also try it with 1200 baud "E" as - * an experiment to see how much it actually helps. - */ - - if (D->use_prefilter) { - float f1, f2; - - f1 = MIN(mark_freq,space_freq) - D->prefilter_baud * baud; - f2 = MAX(mark_freq,space_freq) + D->prefilter_baud * baud; -#if 0 - text_color_set(DW_COLOR_DEBUG); - dw_printf ("Generating prefilter %.0f to %.0f Hz.\n", f1, f2); -#endif - f1 = f1 / (float)samples_per_sec; - f2 = f2 / (float)samples_per_sec; - - //gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_size, BP_WINDOW_HAMMING); - //gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_size, BP_WINDOW_BLACKMAN); - //gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_size, BP_WINDOW_COSINE); - //gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_size, D->bp_window); - gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_size, D->pre_window); - } - -/* - * Filters for detecting mark and space tones. - */ - -#if DEBUG1 - text_color_set(DW_COLOR_DEBUG); - dw_printf ("%s: \n", __FILE__); - dw_printf ("%d baud, %d samples_per_sec\n", baud, samples_per_sec); - dw_printf ("AFSK %d & %d Hz\n", mark_freq, space_freq); - dw_printf ("spll_step_per_sample = %d = 0x%08x\n", D->pll_step_per_sample, D->pll_step_per_sample); - dw_printf ("D->ms_filter_size = %d = 0x%08x\n", D->ms_filter_size, D->ms_filter_size); - dw_printf ("\n"); - dw_printf ("Mark\n"); - dw_printf (" j shape M sin M cos \n"); -#endif - - float Gs = 0, Gc = 0; - - for (j=0; jms_filter_size; j++) { - float am; - float center; - float shape = 1.0f; /* Shape is an attempt to smooth out the */ - /* abrupt edges in hopes of reducing */ - /* overshoot and ringing. */ - /* My first thought was to use a cosine shape. */ - /* Should investigate Hamming and Blackman */ - /* windows mentioned in the literature. */ - /* http://en.wikipedia.org/wiki/Window_function */ - - center = 0.5f * (D->ms_filter_size - 1); - am = ((float)(j - center) / (float)samples_per_sec) * ((float)mark_freq) * (2.0f * (float)M_PI); - - shape = window (D->ms_window, D->ms_filter_size, j); - - D->m_sin_table[j] = sinf(am) * shape; - D->m_cos_table[j] = cosf(am) * shape; - - Gs += D->m_sin_table[j] * sinf(am); - Gc += D->m_cos_table[j] * cosf(am); - -#if DEBUG1 - dw_printf ("%6d %6.2f %6.2f %6.2f\n", j, shape, D->m_sin_table[j], D->m_cos_table[j]) ; -#endif - } - - -/* Normalize for unity gain */ - -#if DEBUG1 - dw_printf ("Before normalizing, Gs = %.2f, Gc = %.2f\n", Gs, Gc) ; -#endif - for (j=0; jms_filter_size; j++) { - D->m_sin_table[j] = D->m_sin_table[j] / Gs; - D->m_cos_table[j] = D->m_cos_table[j] / Gc; - } - - -#if DEBUG1 - text_color_set(DW_COLOR_DEBUG); - - dw_printf ("Space\n"); - dw_printf (" j shape S sin S cos\n"); -#endif - Gs = 0; - Gc = 0; - - for (j=0; jms_filter_size; j++) { - float as; - float center; - float shape = 1.0f; - - center = 0.5 * (D->ms_filter_size - 1); - as = ((float)(j - center) / (float)samples_per_sec) * ((float)space_freq) * (2.0f * (float)M_PI); - - shape = window (D->ms_window, D->ms_filter_size, j); - - D->s_sin_table[j] = sinf(as) * shape; - D->s_cos_table[j] = cosf(as) * shape; - - Gs += D->s_sin_table[j] * sinf(as); - Gc += D->s_cos_table[j] * cosf(as); - -#if DEBUG1 - dw_printf ("%6d %6.2f %6.2f %6.2f\n", j, shape, D->s_sin_table[j], D->s_cos_table[j] ) ; -#endif - } - - -/* Normalize for unity gain */ - -#if DEBUG1 - dw_printf ("Before normalizing, Gs = %.2f, Gc = %.2f\n", Gs, Gc) ; -#endif - for (j=0; jms_filter_size; j++) { - D->s_sin_table[j] = D->s_sin_table[j] / Gs; - D->s_cos_table[j] = D->s_cos_table[j] / Gc; - } - -/* - * Now the lowpass filter. - * I thought we'd want a cutoff of about 0.5 the baud rate - * but it turns out about 1.1x is better. Still investigating... - */ - - if (D->lpf_use_fir) { - float fc; - fc = baud * D->lpf_baud / (float)samples_per_sec; - gen_lowpass (fc, D->lp_filter, D->lp_filter_size, D->lp_window); - } - -/* - * A non-whole number of cycles results in a DC bias. - * Let's see if it helps to take it out. - * Actually makes things worse: 20 fewer decoded. - * Might want to try again after EXPERIMENTC. - */ - -#if 0 -#ifndef AVOID_FLOATING_POINT - -failed experiment - - dc_bias = 0; - for (j=0; jms_filter_size; j++) { - dc_bias += D->m_sin_table[j]; - } - for (j=0; jms_filter_size; j++) { - D->m_sin_table[j] -= dc_bias / D->ms_filter_size; - } - - dc_bias = 0; - for (j=0; jms_filter_size; j++) { - dc_bias += D->m_cos_table[j]; - } - for (j=0; jms_filter_size; j++) { - D->m_cos_table[j] -= dc_bias / D->ms_filter_size; - } - - - dc_bias = 0; - for (j=0; jms_filter_size; j++) { - dc_bias += D->s_sin_table[j]; - } - for (j=0; jms_filter_size; j++) { - D->s_sin_table[j] -= dc_bias / D->ms_filter_size; - } - - dc_bias = 0; - for (j=0; jms_filter_size; j++) { - dc_bias += D->s_cos_table[j]; - } - for (j=0; jms_filter_size; j++) { - D->s_cos_table[j] -= dc_bias / D->ms_filter_size; - } - -#endif -#endif - -/* - * In version 1.2 we try another experiment. - * Try using multiple slicing points instead of the traditional AGC. - */ - - space_gain[0] = MIN_G; - float step = powf(10.0, log10f(MAX_G/MIN_G) / (MAX_SUBCHANS-1)); - for (j=1; j= 0 && chan < MAX_CHANS); - assert (subchan >= 0 && subchan < MAX_SUBCHANS); - -/* - * Filters use last 'filter_size' samples. - * - * First push the older samples down. - * - * Finally, put the most recent at the beginning. - * - * Future project? Can we do better than shifting each time? - */ - - /* Scale to nice number, TODO: range -1.0 to +1.0, not 2. */ - - fsam = sam / 16384.0f; - - //abs_fsam = fsam >= 0.0f ? fsam : -fsam; - - -/* - * Optional bandpass filter before the mark/space discriminator. - */ - - if (D->use_prefilter) { - float cleaner; - - push_sample (fsam, D->raw_cb, D->pre_filter_size); - cleaner = convolve (D->raw_cb, D->pre_filter, D->pre_filter_size); - push_sample (cleaner, D->ms_in_cb, D->ms_filter_size); - } - else { - push_sample (fsam, D->ms_in_cb, D->ms_filter_size); - } - -/* - * Next we have bandpass filters for the mark and space tones. - * - * This takes a lot of computation. - * It's not a problem on a typical (Intel x86 based) PC. - * Dire Wolf takes only about 2 or 3% of the CPU time. - * - * It might be too much for a little microcomputer to handle. - * - * Here we have an optimized case for the default values. - */ - - - -// TODO1.2: is this right or do we need to store profile in the modulator info? - - - if (D->profile == toupper(FFF_PROFILE)) { - - /* ========== Faster for default values on slower processors. ========== */ - - m_sum1 = CALC_M_SUM1(D->ms_in_cb); - m_sum2 = CALC_M_SUM2(D->ms_in_cb); - m_amp = z(m_sum1,m_sum2); - - s_sum1 = CALC_S_SUM1(D->ms_in_cb); - s_sum2 = CALC_S_SUM2(D->ms_in_cb); - s_amp = z(s_sum1,s_sum2); - } - else { - - /* ========== General case to handle all situations. ========== */ - -/* - * find amplitude of "Mark" tone. - */ - m_sum1 = convolve (D->ms_in_cb, D->m_sin_table, D->ms_filter_size); - m_sum2 = convolve (D->ms_in_cb, D->m_cos_table, D->ms_filter_size); - - m_amp = sqrtf(m_sum1 * m_sum1 + m_sum2 * m_sum2); - -/* - * Find amplitude of "Space" tone. - */ - s_sum1 = convolve (D->ms_in_cb, D->s_sin_table, D->ms_filter_size); - s_sum2 = convolve (D->ms_in_cb, D->s_cos_table, D->ms_filter_size); - - s_amp = sqrtf(s_sum1 * s_sum1 + s_sum2 * s_sum2); - - /* ========== End of general case. ========== */ - } - - -/* - * Apply some low pass filtering BEFORE the AGC to remove - * overshoot, ringing, and other bad stuff. - * - * A simple IIR filter is faster but FIR produces better results. - * - * It is a balancing act between removing high frequency components - * from the tone dectection while letting the data thru. - */ - - if (D->lpf_use_fir) { - - push_sample (m_amp, D->m_amp_cb, D->lp_filter_size); - m_amp = convolve (D->m_amp_cb, D->lp_filter, D->lp_filter_size); - - push_sample (s_amp, D->s_amp_cb, D->lp_filter_size); - s_amp = convolve (D->s_amp_cb, D->lp_filter, D->lp_filter_size); - } - else { - - /* Original, but faster, IIR. */ - - m_amp = D->lpf_iir * m_amp + (1.0f - D->lpf_iir) * D->m_amp_prev; - D->m_amp_prev = m_amp; - - s_amp = D->lpf_iir * s_amp + (1.0f - D->lpf_iir) * D->s_amp_prev; - D->s_amp_prev = s_amp; - } - -/* - * Version 1.2: Try new approach to capturing the amplitude for display. - * This is same as the AGC above without the normalization step. - * We want decay to be substantially slower to get a longer - * range idea of the received audio. - */ - - if (m_amp >= D->alevel_mark_peak) { - D->alevel_mark_peak = m_amp * D->quick_attack + D->alevel_mark_peak * (1.0f - D->quick_attack); - } - else { - D->alevel_mark_peak = m_amp * D->sluggish_decay + D->alevel_mark_peak * (1.0f - D->sluggish_decay); - } - - if (s_amp >= D->alevel_space_peak) { - D->alevel_space_peak = s_amp * D->quick_attack + D->alevel_space_peak * (1.0f - D->quick_attack); - } - else { - D->alevel_space_peak = s_amp * D->sluggish_decay + D->alevel_space_peak * (1.0f - D->sluggish_decay); - } - - -/* - * Which tone is stronger? - * - * In an ideal world, simply compare. In my first naive attempt, that - * worked perfectly with perfect signals. In the real world, we don't - * have too many perfect signals. - * - * Here is an excellent explanation: - * http://www.febo.com/packet/layer-one/transmit.html - * - * Under real conditions, we find that the higher tone has a - * considerably smaller amplitude due to the passband characteristics - * of the transmitter and receiver. To make matters worse, it - * varies considerably from one station to another. - * - * The two filters also have different amounts of DC bias. - * - * My solution was to apply automatic gain control (AGC) to the mark and space - * levels. This works by looking at the minimum and maximum outputs - * for each filter and scaling the results to be roughly in the -0.5 to +0.5 range. - * Results were excellent after tweaking the attack and decay times. - * - * 4X6IZ took a different approach. See QEX Jul-Aug 2012. - * - * He ran two different demodulators in parallel. One of them boosted the higher - * frequency tone by 6 dB. Any duplicates were removed. This produced similar results. - * He also used a bandpass filter before the mark/space filters. - * I haven't tried this combination yet for 1200 baud. - * - * First, let's take a look at Track 1 of the TNC test CD. Here the receiver - * has a flat response. We find the mark/space strength ratios very from 0.53 to 1.38 - * with a median of 0.81. This in in line with expections because most - * transmitters add pre-emphasis to boost the higher audio frequencies. - * Track 2 should more closely resemble what comes out of the speaker on a typical - * transceiver. Here we see a ratio from 1.73 to 3.81 with a median of 2.48. - * - * This is similar to my observations of local signals, from the speaker. - * The amplitude ratio varies from 1.48 to 3.41 with a median of 2.70. - * - * Rather than only two filters, let's try slicing the data in more places. - */ - - /* Fast attack and slow decay. */ - /* Numbers were obtained by trial and error from actual */ - /* recorded less-than-optimal signals. */ - - /* See fsk_demod_agc.h for more information. */ - - m_norm = agc (m_amp, D->agc_fast_attack, D->agc_slow_decay, &(D->m_peak), &(D->m_valley)); - s_norm = agc (s_amp, D->agc_fast_attack, D->agc_slow_decay, &(D->s_peak), &(D->s_valley)); - - if (D->num_slicers <= 1) { - - /* Normal case of one demodulator to one HDLC decoder. */ - /* Demodulator output is difference between response from two filters. */ - /* AGC should generally keep this around -1 to +1 range. */ - - demod_out = m_norm - s_norm; - - /* Try adding some Hysteresis. */ - /* (Not to be confused with Hysteria.) */ - - if (demod_out > D->hysteresis) { - demod_data = 1; - } - else if (demod_out < (- (D->hysteresis))) { - demod_data = 0; - } - else { - demod_data = D->slicer[subchan].prev_demod_data; - } - nudge_pll (chan, subchan, 0, demod_data, D); - } - else { - int slice; - - for (slice=0; slicenum_slicers; slice++) { - demod_data = m_amp > s_amp * space_gain[slice]; - nudge_pll (chan, subchan, slice, demod_data, D); - } - } - - -#if DEBUG4 - - if (chan == 0) { - if (hdlc_rec_gathering (chan, subchan)) { - char fname[30]; - - - if (demod_log_fp == NULL) { - seq++; - snprintf (fname, sizeof(fname), "demod/%04d.csv", seq); - if (seq == 1) mkdir ("demod", 0777); - - demod_log_fp = fopen (fname, "w"); - text_color_set(DW_COLOR_DEBUG); - dw_printf ("Starting demodulator log file %s\n", fname); - fprintf (demod_log_fp, "Audio, Mark, Space, Demod, Data, Clock\n"); - } - fprintf (demod_log_fp, "%.3f, %.3f, %.3f, %.3f, %.2f, %.2f\n", fsam + 3.5, m_norm + 2, s_norm + 2, - (m_norm - s_norm) / 2 + 1.5, - demod_data ? .9 : .55, - (D->data_clock_pll & 0x80000000) ? .1 : .45); - } - else { - if (demod_log_fp != NULL) { - fclose (demod_log_fp); - demod_log_fp = NULL; - } - } - } - -#endif - - -} /* end demod_afsk_process_sample */ - - -__attribute__((hot)) -inline static void nudge_pll (int chan, int subchan, int slice, int demod_data, struct demodulator_state_s *D) -{ - -/* - * Finally, a PLL is used to sample near the centers of the data bits. - * - * D points to a demodulator for a channel/subchannel pair so we don't - * have to keep recalculating it. - * - * D->data_clock_pll is a SIGNED 32 bit variable. - * When it overflows from a large positive value to a negative value, we - * sample a data bit from the demodulated signal. - * - * Ideally, the the demodulated signal transitions should be near - * zero we we sample mid way between the transitions. - * - * Nudge the PLL by removing some small fraction from the value of - * data_clock_pll, pushing it closer to zero. - * - * This adjustment will never change the sign so it won't cause - * any erratic data bit sampling. - * - * If we adjust it too quickly, the clock will have too much jitter. - * If we adjust it too slowly, it will take too long to lock on to a new signal. - * - * Be a little more agressive about adjusting the PLL - * phase when searching for a signal. Don't change it as much when - * locked on to a signal. - * - * I don't think the optimal value will depend on the audio sample rate - * because this happens for each transition from the demodulator. - */ - - D->slicer[slice].prev_d_c_pll = D->slicer[slice].data_clock_pll; - D->slicer[slice].data_clock_pll += D->pll_step_per_sample; - - //text_color_set(DW_COLOR_DEBUG); - // dw_printf ("prev = %lx, new data clock pll = %lx\n" D->prev_d_c_pll, D->data_clock_pll); - - if (D->slicer[slice].data_clock_pll < 0 && D->slicer[slice].prev_d_c_pll > 0) { - - /* Overflow. */ - - hdlc_rec_bit (chan, subchan, slice, demod_data, 0, -1); - } - - if (demod_data != D->slicer[slice].prev_demod_data) { - - if (hdlc_rec_gathering (chan, subchan, slice)) { - D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_locked_inertia); - } - else { - D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_searching_inertia); - } - } - -/* - * Remember demodulator output so we can compare next time. - */ - D->slicer[slice].prev_demod_data = demod_data; - -} /* end nudge_pll */ - - -#endif /* GEN_FFF */ - - -/* end demod_afsk.c */ diff --git a/demod_psk.h b/demod_psk.h deleted file mode 100644 index 0f5830d8..00000000 --- a/demod_psk.h +++ /dev/null @@ -1,7 +0,0 @@ - -/* demod_psk.h */ - - -void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char profile, struct demodulator_state_s *D); - -void demod_psk_process_sample (int chan, int subchan, int sam, struct demodulator_state_s *D); diff --git a/direwolf-block-diagram.png b/direwolf-block-diagram.png new file mode 100644 index 00000000..ad339b7e Binary files /dev/null and b/direwolf-block-diagram.png differ diff --git a/direwolf.spec b/direwolf.spec deleted file mode 100644 index 3c64e624..00000000 --- a/direwolf.spec +++ /dev/null @@ -1,62 +0,0 @@ -Name: direwolf -Version: 1.1b1 -Release: 1%{?dist} -Summary: Soundcard based AX.25 TNC - -Group: Applications/Communications -License: GPLv2 -URL: http://home.comcast.net/~wb2osz -Source0: http://home.comcast.net/~wb2osz/Version%201.1/direwolf-%{version}.tgz -Packager: David Ranch (KI6ZHD) -Distribution: RedHat Linux - -Patch0: direwolf-makefile7.patch - -BuildRequires: automake -BuildRequires: alsa-lib-devel - - -%description -Dire Wolf is a software "soundcard" modem/TNC and APRS encoder/decoder. It can -be used stand-alone to receive APRS messages, as a digipeater, APRStt gateway, -or Internet Gateway (IGate). It can also be used as a virtual TNC for other -applications such as APRSIS32, UI-View32, Xastir, APRS-TW, YAAC, UISS, -Linux AX25, SARTrack, RMS Express, and many others. - -%prep - -%setup -q -n %{name}-%{version} -%patch0 -p0 - -%build -make -f Makefile.linux tocalls-symbols -make %{?_smp_mflags} -f Makefile.linux - - -%install -make -f Makefile.linux install DESTDIR=$RPM_BUILD_ROOT -make -f Makefile.linux install-conf DESTDIR=$RPM_BUILD_ROOT - -# Install icon -mkdir -p ${RPM_BUILD_ROOT}%{_datadir}/pixmaps/ -cp dw-icon.png ${RPM_BUILD_ROOT}%{_datadir}/pixmaps/ -mv symbols-new.txt ${RPM_BUILD_ROOT}%{_docdir}/%{name}/ -mv symbolsX.txt ${RPM_BUILD_ROOT}%{_docdir}/%{name}/ -mv tocalls.txt ${RPM_BUILD_ROOT}%{_docdir}/%{name}/ -desktop-file-install \ - --dir=${RPM_BUILD_ROOT}%{_datadir}/applications direwolf.desktop - - -%files -%{_sysconfdir}/ax25/direwolf.conf -%{_bindir}/* -%{_datadir}/pixmaps/dw-icon.png -%{_datadir}/applications/%{name}.desktop -%{_datadir}/direwolf/* -%{_docdir}/%{name}/* - - - -%changelog -* Sat Dec 20 2014 David Ranch - 1.1b1-1 -- new spec file diff --git a/direwolf.txt b/direwolf.txt deleted file mode 100644 index c8577395..00000000 --- a/direwolf.txt +++ /dev/null @@ -1,532 +0,0 @@ -C############################################################# -C# # -C# Configuration file for Dire Wolf # -C# # -L# Linux version # -W# Windows version # -C# # -C############################################################# -R -R -R The sample config file was getting pretty messy -R with the Windows and Linux differences. -R It would be a maintenance burden to keep most of -R two different versions in sync. -R This common source is now used to generate the -R two different variations while having only a single -R copy of the common parts. -R -R The first column contains one of the following: -R -R R remark which is discarded. -R C common to both versions. -R W Windows version only. -R L Linux version only. -R -C# -C# Consult the User Guide for more details on configuration options. -C# -C# -C# These are the most likely settings you might change: -C# -C# (1) MYCALL - call sign and SSID for your station. -C# -C# Look for lines starting with MYCALL and -C# change NOCALL to your own. -C# -C# (2) PBEACON - enable position beaconing. -C# -C# Look for lines starting with PBEACON and -C# modify for your call, location, etc. -C# -C# (3) DIGIPEATER - configure digipeating rules. -C# -C# Look for lines starting with DIGIPEATER. -C# Most people will probably use the given example. -C# Just remove the "#" from the start of the line -C# to enable it. -C# -C# (4) IGSERVER, IGLOGIN - IGate server and login -C# -C# Configure an IGate client to relay messages between -C# radio and internet servers. -C# -C# -C# The default location is "direwolf.conf" in the current working directory. -L# On Linux, the user's home directory will also be searched. -C# An alternate configuration file location can be specified with the "-c" command line option. -C# -C# As you probably guessed by now, # indicates a comment line. -C# -C# Remove the # at the beginning of a line if you want to use a sample -C# configuration that is currently commented out. -C# -C# Commands are a keyword followed by parameters. -C# -C# Command key words are case insensitive. i.e. upper and lower case are equivalent. -C# -C# Command parameters are generally case sensitive. i.e. upper and lower case are different. -C# -C -C -C############################################################# -C# # -C# FIRST AUDIO DEVICE PROPERTIES # -C# (Channel 0 + 1 if in stereo) # -C# # -C############################################################# -C -C# -C# Many people will simply use the default sound device. -C# Some might want to use an alternative device by chosing it here. -C# -W# When the Windows version starts up, it displays something like -W# this with the available sound devices and capabilities: -W# -W# Available audio input devices for receive (*=selected): -W# * 0: Microphone (C-Media USB Headpho (channel 2) -W# 1: Microphone (Bluetooth SCO Audio -W# 2: Microphone (Bluetooth AV Audio) -W# * 3: Microphone (Realtek High Defini (channels 0 & 1) -W# Available audio output devices for transmit (*=selected): -W# * 0: Speakers (C-Media USB Headphone (channel 2) -W# 1: Speakers (Bluetooth SCO Audio) -W# 2: Realtek Digital Output(Optical) -W# 3: Speakers (Bluetooth AV Audio) -W# * 4: Speakers (Realtek High Definiti (channels 0 & 1) -W# 5: Realtek Digital Output (Realtek -W# -W# Example: To use the microphone and speaker connections on the -W# system board, either of these forms can be used: -W -W#ADEVICE High -W#ADEVICE 3 4 -W -W -W# Example: To use the USB Audio, use a command like this with -W# the input and output device numbers. (Remove the # comment character.) -W#ADEVICE USB -W -W# The position in the list can change when devices (e.g. USB) are added and removed. -W# You can also specify devices by using part of the name. -W# Here is an example of specifying the USB Audio device. -W# This is case-sensitive. Upper and lower case are not treated the same. -W -W#ADEVICE USB -W -W -L# Linux ALSA is complicated. See User Guide for discussion. -L# To use something other than the default, generally use plughw -L# and a card number reported by "arecord -l" command. Example: -L -L# ADEVICE plughw:1,0 -L -L# Starting with version 1.0, you can also use "-" or "stdin" to -L# pipe stdout from some other application such as a software defined -L# radio. You can also specify "UDP:" and an optional port for input. -L# Something different must be specified for output. -L -W# ADEVICE - 0 -W# ADEVICE UDP:7355 0 -L# ADEVICE - plughw:1,0 -L# ADEVICE UDP:7355 default -L -L -C -C# -C# Number of audio channels for this souncard: 1 or 2. -C# -C -CACHANNELS 1 -C#ACHANNELS 2 -C -C -C############################################################# -C# # -C# SECOND AUDIO DEVICE PROPERTIES # -C# (Channel 2 + 3 if in stereo) # -C# # -C############################################################# -C -C#ADEVICE1 ... -C -C -C############################################################# -C# # -C# THIRD AUDIO DEVICE PROPERTIES # -C# (Channel 4 + 5 if in stereo) # -C# # -C############################################################# -C -C#ADEVICE2 ... -C -C -C############################################################# -C# # -C# CHANNEL 0 PROPERTIES # -C# # -C############################################################# -C -CCHANNEL 0 -C -C# -C# The following MYCALL, MODEM, PTT, etc. configuration items -C# apply to the most recent CHANNEL. -C# -C -C# -C# Station identifier for this channel. -C# Multiple channels can have the same or different names. -C# -C# It can be up to 6 letters and digits with an optional ssid. -C# The APRS specification requires that it be upper case. -C# -C# Example (don't use this unless you are me): MYCALL WB2OSZ-5 -C# -C -CMYCALL N0CALL -C -C# -C# Pick a suitable modem speed based on your situation. -C# 1200 Most common for VHF/UHF. Default if not specified. -C# 300 Low speed for HF SSB. -C# 9600 High speed - Can't use Microphone and Speaker connections. -C# -C# In the simplest form, just specify the speed. -C# -C -CMODEM 1200 -C#MODEM 300 -C#MODEM 9600 -C -C# -C# These are the defaults should be fine for most cases. In special situations, -C# you might want to specify different AFSK tones or the baseband mode which does -C# not use AFSK. -C# -C#MODEM 1200 1200:2200 -C#MODEM 300 1600:1800 -C#MODEM 9600 0:0 -C# -C# -C# On HF SSB, you might want to use multiple demodulators on slightly different -C# frequencies to compensate for stations off frequency. Here we have 7 different -C# demodulators at 30 Hz intervals. This takes a lot of CPU power so you will -C# probably need to reduce the audio sampling rate with the /n option. -C -C#MODEM 300 1600:1800 7@30 /4 -C -C -C# -C# Uncomment line below to enable the DTMF decoder for this channel. -C# -C -C#DTMF -C -C# -C# If not using a VOX circuit, the transmitter Push to Talk (PTT) -C# control is usually wired to a serial port with a suitable interface circuit. -C# DON'T connect it directly! -C# -C# For the PTT command, specify the device and either RTS or DTR. -C# RTS or DTR may be preceded by "-" to invert the signal. -C# Both can be used for interfaces that want them driven with opposite polarity. -C# -L# COM1 can be used instead of /dev/ttyS0, COM2 for /dev/ttyS1, and so on. -L# -C -C#PTT COM1 RTS -C#PTT COM1 RTS -DTR -L#PTT /dev/ttyUSB0 RTS -C -L# -L# On Linux, you can also use general purpose I/O pins if -L# your system is configured for user access to them. -L# This would apply mostly to microprocessor boards, not a regular PC. -L# See separate Raspberry Pi document for more details. -L# The number may be preceded by "-" to invert the signal. -L# -L -L#PTT GPIO 25 -L -C# The Data Carrier Detect (DCD) signal can be sent to the same places -C# as the PTT signal. This could be used to light up an LED like a normal TNC. -C -C#DCD COM1 -DTR -L#DCD GPIO 24 -C -C -C############################################################# -C# # -C# CHANNEL 1 PROPERTIES # -C# # -C############################################################# -C -C#CHANNEL 1 -C -C# -C# Specify MYCALL, MODEM, PTT, etc. configuration items for -C# CHANNEL 1. Repeat for any other channels. -C -C -C############################################################# -C# # -C# TEXT TO SPEECH COMMAND FILE # -C# # -C############################################################# -C -W#SPEECH dwespeak.bat -L#SPEECH dwespeak.sh -C -C -C############################################################# -C# # -C# VIRTUAL TNC SERVER PROPERTIES # -C# # -C############################################################# -C -C# -C# Dire Wolf acts as a virtual TNC and can communicate with -C# client applications by different protocols: -C# -C# - the "AGW TCPIP Socket Interface" - default port 8000 -C# - KISS protocol over TCP socket - default port 8001 -W# - KISS TNC via serial port -L# - KISS TNC via pseudo terminal (-p command line option) -C# -C -CAGWPORT 8000 -CKISSPORT 8001 -C -W# -W# Some applications are designed to operate with only a physical -W# TNC attached to a serial port. For these, we provide a virtual serial -W# port that appears to be connected to a TNC. -W# -W# Take a look at the User Guide for instructions to set up -W# two virtual serial ports named COM3 and COM4 connected by -W# a null modem. -W# -W# Using the configuration described, Dire Wolf will connect to -W# COM3 and the client application will use COM4. -W# -W# Uncomment following line to use this feature. -W -W#NULLMODEM COM3 -W -W -C# -C# It is sometimes possible to recover frames with a bad FCS. -C# This applies to all channels. -C# -C# 0 [NONE] - Don't try to repair. -C# 1 [SINGLE] - Attempt to fix single bit error. (default) -C# 2 [DOUBLE] - Also attempt to fix two adjacent bits. -C# ... see User Guide for more values and in-depth discussion. -C# -C -C#FIX_BITS 0 -C -C# -C############################################################# -C# # -C# BEACONING PROPERTIES # -C# # -C############################################################# -C -C -C# -C# Beaconing is configured with these two commands: -C# -C# PBEACON - for a position report (usually yourself) -C# OBEACON - for an object report (usually some other entity) -C# -C# Each has a series of keywords and values for options. -C# See User Guide for details. -C# -C# Example: -C# -C# This results in a broadcast once every 10 minutes. -C# Every half hour, it can travel via two digipeater hops. -C# The others are kept local. -C# -C -C#PBEACON delay=1 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" via=WIDE1-1,WIDE2-1 -C#PBEACON delay=11 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" -C#PBEACON delay=21 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" -C -C -C# With UTM coordinates instead of latitude and longitude. -C -C#PBEACON delay=1 every=10 overlay=S symbol="digi" zone=19T easting=307477 northing=4720178 -C -C -C# -C# When the destination field is set to "SPEECH" the information part is -C# converted to speech rather than transmitted as a data frame. -C# -C -C#CBEACON dest="SPEECH" info="Club meeting tonight at 7 pm." -C -C -C# -C# Modify for your particular situation before removing -C# the # comment character from the beginning of appropriate lines above. -C# -C -C -C############################################################# -C# # -C# DIGIPEATER PROPERTIES # -C# # -C############################################################# -C -C# -C# For most common situations, use something like this by removing -C# the "#" from the beginning of the line below. -C# -C -C#DIGIPEAT 0 0 ^WIDE[3-7]-[1-7]$|^TEST$ ^WIDE[12]-[12]$ TRACE -C -C# See User Guide for more explanation of what this means and how -C# it can be customized for your particular needs. -C -C# Filtering can be used to limit was is digipeated. -C# For example, only weather weather reports, received on channel 0, -C# will be retransmitted on channel 1. -C# -C -C#FILTER 0 1 t/wn -C -C -C############################################################# -C# # -C# INTERNET GATEWAY # -C# # -C############################################################# -C -C# First you need to specify the name of a Tier 2 server. -C# The current preferred way is to use one of these regional rotate addresses: -C -C# noam.aprs2.net - for North America -C# soam.aprs2.net - for South America -C# euro.aprs2.net - for Europe and Africa -C# asia.aprs2.net - for Asia -C# aunz.aprs2.net - for Oceania -C -C#IGSERVER noam.aprs2.net -C -C# You also need to specify your login name and passcode. -C# Contact the author if you can't figure out how to generate the passcode. -C -C#IGLOGIN WB2OSZ-5 123456 -C -C# That's all you need for a receive only IGate which relays -C# messages from the local radio channel to the global servers. -C -C# Some might want to send an IGate client position directly to a server -C# without sending it over the air and relying on someone else to -C# forward it to an IGate server. This is done by using sendto=IG rather -C# than a radio channel number. Overlay R for receive only, T for two way. -C -C#PBEACON sendto=IG delay=0:30 every=60:00 symbol="igate" overlay=R lat=42^37.14N long=071^20.83W -C#PBEACON sendto=IG delay=0:30 every=60:00 symbol="igate" overlay=T lat=42^37.14N long=071^20.83W -C -C -C# To relay messages from the Internet to radio, you need to add -C# one more option with the transmit channel number and a VIA path. -C -C#IGTXVIA 0 WIDE1-1 -C -C# You might want to apply a filter for what packets will be obtained from the server. -C# Read about filters here: http://www.aprs-is.net/javaprsfilter.aspx -C# Example, positions and objects within 50 km of my location: -C -C#IGFILTER m/50 -C -C# That is known as a server-side filter. It is processed by the IGate server. -C# You can also apply local filtering to limit what will be transmitted on the -C# RF side. For example, transmit only "messages" on channel 0 and weather -C# reports on channel 1. -C -C#FILTER IG 0 t/m -C#FILTER IG 1 t/wn -C -C# Finally, we don't want to flood the radio channel. -C# The IGate function will limit the number of packets transmitted -C# during 1 minute and 5 minute intervals. If a limit would -C# be exceeded, the packet is dropped and message is displayed in red. -C -CIGTXLIMIT 6 10 -C -C -C############################################################# -C# # -C# APRStt GATEWAY # -C# # -C############################################################# -C -C# -C# Dire Wolf can receive DTMF (commonly known as Touch Tone) -C# messages and convert them to packet objects. -C# -C# See separate "APRStt-Implementation-Notes" document for details. -C# -C -C# -C# Sample gateway configuration based on: -C# -C# http://www.aprs.org/aprstt/aprstt-coding24.txt -C# http://www.aprs.org/aprs-jamboree-2013.html -C# -C -C# Define specific points. -C -CTTPOINT B01 37^55.37N 81^7.86W -CTTPOINT B7495088 42.605237 -71.34456 -CTTPOINT B934 42.605237 -71.34456 -C -CTTPOINT B901 42.661279 -71.364452 -CTTPOINT B902 42.660411 -71.364419 -CTTPOINT B903 42.659046 -71.364452 -CTTPOINT B904 42.657578 -71.364602 -C -C -C# For location at given bearing and distance from starting point. -C -CTTVECTOR B5bbbddd 37^55.37N 81^7.86W 0.01 mi -C -C# For location specified by x, y coordinates. -C -CTTGRID Byyyxxx 37^50.00N 81^00.00W 37^59.99N 81^09.99W -C -C# UTM location for Lowell-Dracut-Tyngsborough State Forest. -C -CTTUTM B6xxxyyy 19T 10 300000 4720000 -C -C -C -C# Location for the corral. -C -CTTCORRAL 37^55.50N 81^7.00W 0^0.02N -C -C# Compact messages - Fixed locations xx and object yyy where -C# Object numbers 100 - 199 = bicycle -C# Object numbers 200 - 299 = fire truck -C# Others = dog -C -CTTMACRO xx1yy B9xx*AB166*AA2B4C5B3B0A1yy -CTTMACRO xx2yy B9xx*AB170*AA3C4C7C3B0A2yy -CTTMACRO xxyyy B9xx*AB180*AA3A6C4A0Ayyy -C -CTTMACRO z Cz -C -C# Receive on channel 0, Transmit object reports on channel 1 with optional via path. -C -C#TTOBJ 0 1 WIDE1-1 -C -C# Advertise gateway position with beacon. -C -C# OBEACON DELAY=0:15 EVERY=10:00 VIA=WIDE1-1 OBJNAME=WB2OSZ-tt SYMBOL=APRStt LAT=42^37.14N LONG=71^20.83W COMMENT="APRStt Gateway" -C -C diff --git a/doc/2400-4800-PSK-for-APRS-Packet-Radio.pdf b/doc/2400-4800-PSK-for-APRS-Packet-Radio.pdf index 077c5ff3..4efd364b 100644 Binary files a/doc/2400-4800-PSK-for-APRS-Packet-Radio.pdf and b/doc/2400-4800-PSK-for-APRS-Packet-Radio.pdf differ diff --git a/doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf b/doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf index 6730138e..966ea52d 100644 Binary files a/doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf and b/doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf differ diff --git a/doc/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf b/doc/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf index 5fecd4c3..85fafb30 100644 Binary files a/doc/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf and b/doc/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf differ diff --git a/doc/AIS-Reception.pdf b/doc/AIS-Reception.pdf new file mode 100644 index 00000000..c868d7ce Binary files /dev/null and b/doc/AIS-Reception.pdf differ diff --git a/doc/APRStt-Implementation-Notes.pdf b/doc/APRStt-Implementation-Notes.pdf index 7fae3aed..3e6b8fb0 100644 Binary files a/doc/APRStt-Implementation-Notes.pdf and b/doc/APRStt-Implementation-Notes.pdf differ diff --git a/doc/AX25_plus_FEC_equals_FX25.pdf b/doc/AX25_plus_FEC_equals_FX25.pdf new file mode 100644 index 00000000..3113a1bc Binary files /dev/null and b/doc/AX25_plus_FEC_equals_FX25.pdf differ diff --git a/doc/Bluetooth-KISS-TNC.pdf b/doc/Bluetooth-KISS-TNC.pdf new file mode 100644 index 00000000..6969334f Binary files /dev/null and b/doc/Bluetooth-KISS-TNC.pdf differ diff --git a/doc/CMakeLists.txt b/doc/CMakeLists.txt new file mode 100644 index 00000000..d8b6343f --- /dev/null +++ b/doc/CMakeLists.txt @@ -0,0 +1,21 @@ + +install(FILES "${CUSTOM_DOC_DIR}/README.md" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/2400-4800-PSK-for-APRS-Packet-Radio.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/AIS-Reception.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/APRS-Telemetry-Toolkit.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/APRStt-Implementation-Notes.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/APRStt-interface-for-SARTrack.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/APRStt-Listening-Example.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/AX25_plus_FEC_equals_FX25.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Bluetooth-KISS-TNC.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Going-beyond-9600-baud.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Raspberry-Pi-APRS.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Raspberry-Pi-APRS-Tracker.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Raspberry-Pi-SDR-IGate.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Successful-APRS-IGate-Operation.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/User-Guide.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/WA8LMF-TNC-Test-CD-Results.pdf" DESTINATION ${INSTALL_DOC_DIR}) +install(FILES "${CUSTOM_DOC_DIR}/Why-is-9600-only-twice-as-fast-as-1200.pdf" DESTINATION ${INSTALL_DOC_DIR}) diff --git a/doc/README.md b/doc/README.md index 734f3a01..9f44684e 100644 --- a/doc/README.md +++ b/doc/README.md @@ -2,13 +2,20 @@ Click on the document name to view in your web browser or the link following to download the PDF file. + +## Slide Show ## + +Brief summary of packet radio / APRS history and the capbilities of Dire Wolf. + +[Power Point presentation](https://github.com/wb2osz/direwolf-presentation) -- Why not give a talk at a local club meeting? + ## Essential Reading ## -- [**User Guide**](User-Guide.pdf) [ [*download*](../../../raw/dev/doc/User-Guide.pdf) ] +- [**User Guide**](User-Guide.pdf) [ [*download*](../../../raw/master/doc/User-Guide.pdf) ] This is your primary source of information about installation, operation, and configuration. -- [**Raspberry Pi APRS**](Raspberry-Pi-APRS.pdf) [ [*download*](../../../raw/dev/doc/Raspberry-Pi-APRS.pdf) ] +- [**Raspberry Pi APRS**](Raspberry-Pi-APRS.pdf) [ [*download*](../../../raw/master/doc/Raspberry-Pi-APRS.pdf) ] The Raspberry Pi has some special considerations that make it different from other generic Linux systems. @@ -19,37 +26,60 @@ Click on the document name to view in your web browser or the link following to These dive into more detail for specialized topics or typical usage scenarios. + + +- [**AX.25 + FEC = FX.25**](AX25_plus_FEC_equals_FX25.pdf) [ [*download*](../../../raw/dev/doc/AX25_plus_FEC_equals_FX25.pdf) ] + + What can you do if your radio signal isn’t quite strong enough to get through reliably? Move to higher ground? Get a better antenna? More power? Use very narrow bandwidth and very slow data? + + Sometimes those are not options. Another way to improve communication reliability is to add redundant information so the message will still get through even if small parts are missing. FX.25 adds forward error correction (FEC) which maintaining complete compatibility with older equipment. + + +- [**AX.25 Throughput: Why is 9600 bps Packet Radio only twice as fast as 1200?**](Why-is-9600-only-twice-as-fast-as-1200.pdf) [ [*download*](../../../raw/dev/doc/Why-is-9600-only-twice-as-fast-as-1200.pdf) ] + + Simply switching to a higher data rate will probably result in great disappointment. You might expect it to be 8 times faster but it can turn out to be only twice as fast. + + In this document, we look at why a large increase in data bit rate can produce a much smaller increase in throughput. We will explore techniques that can be used to make large improvements and drastically speed up large data transfer. + + + + - [**Successful APRS IGate Operation**](Successful-APRS-IGate-Operation.pdf) [ [*download*](../../../raw/dev/doc/Successful-APRS-IGate-Operation.pdf) ] + Dire Wolf can serve as a gateway between the APRS radio network and APRS-IS servers on the Internet. This explains how it all works, proper configuration, and troubleshooting. -- [**APRStt Implementation Notes**](APRStt-Implementation-Notes.pdf) [ [*download*](../../../raw/dev/doc/APRStt-Implementation-Notes.pdf) ] +- [**Bluetooth KISS TNC**](Bluetooth-KISS-TNC.pdf) [ [*download*](../../../raw/master/doc/Bluetooth-KISS-TNC.pdf) ] + + Eliminate the cable between your TNC and application. Use Bluetooth instead. + +- [**APRStt Implementation Notes**](APRStt-Implementation-Notes.pdf) [ [*download*](../../../raw/master/doc/APRStt-Implementation-Notes.pdf) ] Very few hams have portable equipment for APRS but nearly everyone has a handheld radio that can send DTMF tones. APRStt allows a user, equipped with only DTMF (commonly known as Touch Tone) generation capability, to enter information into the global APRS data network. This document explains how the APRStt concept was implemented in the Dire Wolf application. -- [**APRStt Interface for SARTrack**](APRStt-interface-for-SARTrack.pdf) [ [*download*](../../../raw/dev/doc/APRStt-interface-for-SARTrack.pdf) ] +- [**APRStt Interface for SARTrack**](APRStt-interface-for-SARTrack.pdf) [ [*download*](../../../raw/master/doc/APRStt-interface-for-SARTrack.pdf) ] This example illustrates how APRStt can be integrated with other applications such as SARTrack, APRSISCE/32, YAAC, or Xastir. -- [**APRStt Listening Example**](APRStt-Listening-Example.pdf) [ [*download*](../../../raw/dev/doc/APRStt-Listening-Example.pdf) ] +- [**APRStt Listening Example**](APRStt-Listening-Example.pdf) [ [*download*](../../../raw/master/doc/APRStt-Listening-Example.pdf) ] WB4APR described a useful application for the [QIKCOM-2 Satallite Transponder](http://www.tapr.org/pipermail/aprssig/2015-November/045035.html). Don’t have your own QIKCOM-2 Satellite Transponder? No Problem. You can do the same thing with an ordinary computer and the APRStt gateway built into Dire Wolf. Here’s how. -- [**Raspberry Pi APRS Tracker**](Raspberry-Pi-APRS-Tracker.pdf) [ [*download*](../../../raw/dev/doc/Raspberry-Pi-APRS-Tracker.pdf) ] +- [**Raspberry Pi APRS Tracker**](Raspberry-Pi-APRS-Tracker.pdf) [ [*download*](../../../raw/master/doc/Raspberry-Pi-APRS-Tracker.pdf) ] Build a tracking device which transmits position from a GPS receiver. -- [**Raspberry Pi SDR IGate**](Raspberry-Pi-SDR-IGate.pdf) [ [*download*](../../../raw/dev/doc/Raspberry-Pi-SDR-IGate.pdf) ] +- [**Raspberry Pi SDR IGate**](Raspberry-Pi-SDR-IGate.pdf) [ [*download*](../../../raw/master/doc/Raspberry-Pi-SDR-IGate.pdf) ] It's easy to build a receive-only APRS Internet Gateway (IGate) with only a Raspberry Pi and a software defined radio (RTL-SDR) dongle. Here’s how. -- [**APRS Telemetry Toolkit**](APRS-Telemetry-Toolkit.pdf) [ [*download*](../../../raw/dev/doc/APRS-Telemetry-Toolkit.pdf) ] +- [**APRS Telemetry Toolkit**](APRS-Telemetry-Toolkit.pdf) [ [*download*](../../../raw/master/doc/APRS-Telemetry-Toolkit.pdf) ] Describes scripts and methods to generate telemetry. Includes a complete example of attaching an analog to @@ -58,21 +88,41 @@ These dive into more detail for specialized topics or typical usage scenarios. -- [**2400 & 4800 bps PSK for APRS / Packet Radio**](2400-4800-PSK-for-APRS-Packet-Radio.pdf) [ [*download*](../../../raw/dev/doc/2400-4800-PSK-for-APRS-Packet-Radio.pdf) ] +- [**2400 & 4800 bps PSK for APRS / Packet Radio**](2400-4800-PSK-for-APRS-Packet-Radio.pdf) [ [*download*](../../../raw/master/doc/2400-4800-PSK-for-APRS-Packet-Radio.pdf) ] Double or quadruple your data rate by sending multiple bits at the same time. -- [**Going beyond 9600 baud**](Going-beyond-9600-baud.pdf) [ [*download*](../../../raw/dev/doc/Going-beyond-9600-baud.pdf) ] +- [**Going beyond 9600 baud**](Going-beyond-9600-baud.pdf) [ [*download*](../../../raw/master/doc/Going-beyond-9600-baud.pdf) ] Why stop at 9600 baud? Go faster if your soundcard and radio can handle it. +- [**AIS Reception**](AIS-Reception.pdf) [ [*download*](../../../raw/dev/doc/AIS-Reception.pdf) ] + + + AIS is an international tracking system for ships. Messages can contain position, speed, course, name, destination, status, vessel dimensions, and many other types of information. Learn how to receive these signals with an ordindary ham transceiver and display the ship locations with APRS applications or [OpenCPN](https://opencpn.org). + +- **[EAS to APRS message converter](https://github.com/wb2osz/eas2aprs)** + + + The [U.S. National Weather Service](https://www.weather.gov/nwr/) (NWS) operates more than 1,000 VHF FM radio stations that continuously transmit weather information. These stations also transmit special warnings about severe weather, disasters (natural & manmade), and public safety. + + Alerts are sent in a digital form known as Emergency Alert System (EAS) Specific Area Message Encoding (SAME). [You can hear a sample here](https://en.wikipedia.org/wiki/Specific_Area_Message_Encoding). + + It is possible to buy radios that decode these messages but what fun is that? We are ham radio operators so we want to build our own from stuff that we already have sitting around. + ## Miscellaneous ## +- **[Ham Radio of Things (HRoT)](https://github.com/wb2osz/hrot)** -- [**A Better APRS Packet Demodulator, part 1, 1200 baud**](A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf) [ [*download*](../../../raw/dev/doc/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf) ] + + Now that billions of computers and mobile phones (which are handheld computers) are all connected by the Internet, the large growth is expected from the “Internet of Things.” What is a “thing?” It could be a temperature sensor, garage door opener, motion detector, flood water level, smoke alarm, antenna rotator, coffee maker, lights, home thermostat, â€Ļ, just about anything you might want to monitor or control. + + There have been other occasional mentions of merging Ham Radio with the Internet of Things but only ad hoc incompatible narrowly focused applications. Here is a proposal for a standardized more flexible method so different systems can communicate with each other. + +- [**A Better APRS Packet Demodulator, part 1, 1200 baud**](A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf) [ [*download*](../../../raw/master/doc/A-Better-APRS-Packet-Demodulator-Part-1-1200-baud.pdf) ] Sometimes it's a little mystifying why an APRS / AX.25 Packet TNC will decode some signals @@ -84,7 +134,7 @@ and a couple things that can be done about it. -- [**A Better APRS Packet Demodulator, part 2, 9600 baud**](A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf) [ [*download*](../../../raw/dev/doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf) ] +- [**A Better APRS Packet Demodulator, part 2, 9600 baud**](A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf) [ [*download*](../../../raw/master/doc/A-Better-APRS-Packet-Demodulator-Part-2-9600-baud.pdf) ] In the first part of this series we discussed 1200 baud audio frequency shift keying (AFSK). The mismatch between FM transmitter pre-emphasis and the @@ -93,26 +143,34 @@ and a couple things that can be done about it. This makes it more difficult to demodulate them accurately. 9600 baud operation is an entirely different animal. ... -- [**WA8LMF TNC Test CD Results a.k.a. Battle of the TNCs**](WA8LMF-TNC-Test-CD-Results.pdf) [ [*download*](../../../raw/dev/doc/WA8LMF-TNC-Test-CD-Results.pdf) ] +- [**WA8LMF TNC Test CD Results a.k.a. Battle of the TNCs**](WA8LMF-TNC-Test-CD-Results.pdf) [ [*download*](../../../raw/master/doc/WA8LMF-TNC-Test-CD-Results.pdf) ] How can we compare how well the TNCs perform under real world conditions? The de facto standard of measurement is the number of packets decoded from [WA8LMF’s TNC Test CD](http://wa8lmf.net/TNCtest/index.htm). Many have published the number of packets they have been able to decode from this test. Here they are, all gathered in one place, for your reading pleasure. -- [**A Closer Look at the WA8LMF TNC Test CD**](A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf) [ [*download*](../../../raw/dev/doc/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf) ] +- [**A Closer Look at the WA8LMF TNC Test CD**](A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf) [ [*download*](../../../raw/master/doc/A-Closer-Look-at-the-WA8LMF-TNC-Test-CD.pdf) ] Here, we take a closer look at some of the frames on the TNC Test CD in hopes of gaining some insights into why some are easily decoded and others are more difficult. There are a lot of ugly signals out there. Many can be improved by decreasing the transmit volume. Others are just plain weird and you have to wonder how they are being generated. + +## Additional Documentation for Dire Wolf Software TNC # + + +When there was little documentation, it was all added to the source code repository [https://github.com/wb2osz/direwolf/tree/master/doc](https://github.com/wb2osz/direwolf/tree/master/doc) + +The growing number of documentation files and revisions are making the source code repository very large which means long download times. Additional documentation, not tied to a specific release, is now being added to [https://github.com/wb2osz/direwolf-doc](https://github.com/wb2osz/direwolf-doc) + ## Questions? Experiences to share? ## Here are some good places to ask questions and share your experiences: -- [Dire Wolf packet TNC](https://groups.yahoo.com/neo/groups/direwolf_packet/info) +- [Dire Wolf Software TNC](https://groups.io/g/direwolf) -- [Raspberry Pi 4 Ham Radio](https://groups.yahoo.com/neo/groups/Raspberry_Pi_4-Ham_RADIO/info) +- [Raspberry Pi 4 Ham Radio](https://groups.io/g/RaspberryPi-4-HamRadio) -- [linuxham](https://groups.yahoo.com/neo/groups/linuxham/info) +- [linuxham](https://groups.io/g/linuxham) - [TAPR aprssig](http://www.tapr.org/pipermail/aprssig/) diff --git a/doc/Raspberry-Pi-APRS-Tracker.pdf b/doc/Raspberry-Pi-APRS-Tracker.pdf index 0d61dab5..c0c8c0be 100644 Binary files a/doc/Raspberry-Pi-APRS-Tracker.pdf and b/doc/Raspberry-Pi-APRS-Tracker.pdf differ diff --git a/doc/Raspberry-Pi-APRS.pdf b/doc/Raspberry-Pi-APRS.pdf index 86b266cb..344c3de6 100644 Binary files a/doc/Raspberry-Pi-APRS.pdf and b/doc/Raspberry-Pi-APRS.pdf differ diff --git a/doc/Raspberry-Pi-SDR-IGate.pdf b/doc/Raspberry-Pi-SDR-IGate.pdf index 5f0d302f..b4c84f18 100644 Binary files a/doc/Raspberry-Pi-SDR-IGate.pdf and b/doc/Raspberry-Pi-SDR-IGate.pdf differ diff --git a/doc/Successful-APRS-IGate-Operation.pdf b/doc/Successful-APRS-IGate-Operation.pdf index ce8b3cb1..9a51ef58 100644 Binary files a/doc/Successful-APRS-IGate-Operation.pdf and b/doc/Successful-APRS-IGate-Operation.pdf differ diff --git a/doc/User-Guide.pdf b/doc/User-Guide.pdf index 848bcc11..319f882f 100644 Binary files a/doc/User-Guide.pdf and b/doc/User-Guide.pdf differ diff --git a/doc/WA8LMF-TNC-Test-CD-Results.pdf b/doc/WA8LMF-TNC-Test-CD-Results.pdf index 7f883acd..d9af1a3f 100644 Binary files a/doc/WA8LMF-TNC-Test-CD-Results.pdf and b/doc/WA8LMF-TNC-Test-CD-Results.pdf differ diff --git a/doc/Why-is-9600-only-twice-as-fast-as-1200.pdf b/doc/Why-is-9600-only-twice-as-fast-as-1200.pdf new file mode 100644 index 00000000..829aa648 Binary files /dev/null and b/doc/Why-is-9600-only-twice-as-fast-as-1200.pdf differ diff --git a/dsp.h b/dsp.h deleted file mode 100644 index 1f5aaa52..00000000 --- a/dsp.h +++ /dev/null @@ -1,10 +0,0 @@ - -/* dsp.h */ - -// TODO: put prefixes on these names. - -float window (bp_window_t type, int size, int j); - -void gen_lowpass (float fc, float *lp_filter, int filter_size, bp_window_t wtype); - -void gen_bandpass (float f1, float f2, float *bp_filter, int filter_size, bp_window_t wtype); \ No newline at end of file diff --git a/dtime_now.c b/dtime_now.c deleted file mode 100644 index c78fba86..00000000 --- a/dtime_now.c +++ /dev/null @@ -1,61 +0,0 @@ - -#include "direwolf.h" -#include "textcolor.h" -#include "dtime_now.h" - - -/* Current time in seconds but more resolution than time(). */ - -/* We don't care what date a 0 value represents because we */ -/* only use this to calculate elapsed real time. */ - - - -#include - -#ifdef __APPLE__ -#include -#endif - - - - - -double dtime_now (void) -{ - double result; - -#if __WIN32__ - /* 64 bit integer is number of 100 nanosecond intervals from Jan 1, 1601. */ - - FILETIME ft; - - GetSystemTimeAsFileTime (&ft); - - result = ((( (double)ft.dwHighDateTime * (256. * 256. * 256. * 256.) + - (double)ft.dwLowDateTime ) / 10000000.) - 11644473600.); -#else - /* tv_sec is seconds from Jan 1, 1970. */ - - struct timespec ts; - -#ifdef __APPLE__ - struct timeval tp; - gettimeofday(&tp, NULL); - ts.tv_nsec = tp.tv_usec * 1000; - ts.tv_sec = tp.tv_sec; -#else - clock_gettime (CLOCK_REALTIME, &ts); -#endif - - result = ((double)(ts.tv_sec) + (double)(ts.tv_nsec) * 0.000000001); - -#endif - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("dtime_now() returns %.3f\n", result ); -#endif - - return (result); -} diff --git a/dtime_now.h b/dtime_now.h deleted file mode 100644 index 5d7c39a5..00000000 --- a/dtime_now.h +++ /dev/null @@ -1,3 +0,0 @@ - - -extern double dtime_now (void); \ No newline at end of file diff --git a/dw-icon.rc b/dw-icon.rc deleted file mode 100644 index ce34b403..00000000 --- a/dw-icon.rc +++ /dev/null @@ -1 +0,0 @@ -MAINICON ICON "dw-icon.ico" \ No newline at end of file diff --git a/LICENSE-other.txt b/external/LICENSE similarity index 100% rename from LICENSE-other.txt rename to external/LICENSE diff --git a/external/geotranz/CMakeLists.txt b/external/geotranz/CMakeLists.txt new file mode 100644 index 00000000..576d8b82 --- /dev/null +++ b/external/geotranz/CMakeLists.txt @@ -0,0 +1,17 @@ +# UTM, USNG, MGRS conversions + +set(GEOTRANZ_LIBRARIES geotranz CACHE INTERNAL "geotranz") + +list(APPEND geotranz_SOURCES + error_string.c + mgrs.c + polarst.c + tranmerc.c + ups.c + usng.c + utm.c + ) + +add_library(geotranz STATIC + ${geotranz_SOURCES} + ) diff --git a/geotranz/README-FIRST.txt b/external/geotranz/README-FIRST.txt similarity index 100% rename from geotranz/README-FIRST.txt rename to external/geotranz/README-FIRST.txt diff --git a/geotranz/error_string.c b/external/geotranz/error_string.c similarity index 100% rename from geotranz/error_string.c rename to external/geotranz/error_string.c diff --git a/geotranz/error_string.h b/external/geotranz/error_string.h similarity index 100% rename from geotranz/error_string.h rename to external/geotranz/error_string.h diff --git a/geotranz/mgrs.c b/external/geotranz/mgrs.c similarity index 99% rename from geotranz/mgrs.c rename to external/geotranz/mgrs.c index 9fb82899..84454abb 100644 --- a/geotranz/mgrs.c +++ b/external/geotranz/mgrs.c @@ -400,7 +400,7 @@ long Make_MGRS_String (char* MGRS, if (Zone) i = sprintf (MGRS+i,"%2.2ld",Zone); else - strncpy(MGRS, " ", 2); // 2 spaces + strcpy(MGRS, " "); // 2 spaces - Should i be set to 2? for (j=0;j<3;j++) MGRS[i++] = alphabet[Letters[j]]; @@ -868,7 +868,7 @@ long Convert_MGRS_To_Geodetic (char* MGRS, */ { /* Convert_MGRS_To_Geodetic */ long zone; - char hemisphere; + char hemisphere = '?'; double easting; double northing; long zone_exists; @@ -1260,9 +1260,9 @@ long Convert_MGRS_To_UPS ( char *MGRS, double false_northing; /* False northing for 3rd letter */ double grid_easting; /* easting for 100,000 meter grid square */ double grid_northing; /* northing for 100,000 meter grid square */ - long zone; + long zone = 0; long letters[MGRS_LETTERS]; - long in_precision; + long in_precision = 0; int index = 0; long error_code = MGRS_NO_ERROR; diff --git a/geotranz/mgrs.h b/external/geotranz/mgrs.h similarity index 99% rename from geotranz/mgrs.h rename to external/geotranz/mgrs.h index 79a1c28e..bd0453a1 100644 --- a/geotranz/mgrs.h +++ b/external/geotranz/mgrs.h @@ -236,7 +236,7 @@ extern "C" { * The function Convert_MGRS_To_UPS converts an MGRS coordinate string * to UPS (hemisphere, easting, and northing) coordinates, according * to the current ellipsoid parameters. If any errors occur, the error - * code(s) are returned by the function, otherwide UPS_NO_ERROR is returned. + * code(s) are returned by the function, otherwise UPS_NO_ERROR is returned. * * MGRS : MGRS coordinate string (input) * Hemisphere : Hemisphere either 'N' or 'S' (output) diff --git a/geotranz/polarst.c b/external/geotranz/polarst.c similarity index 100% rename from geotranz/polarst.c rename to external/geotranz/polarst.c diff --git a/geotranz/polarst.h b/external/geotranz/polarst.h similarity index 100% rename from geotranz/polarst.h rename to external/geotranz/polarst.h diff --git a/geotranz/readme.txt b/external/geotranz/readme.txt similarity index 100% rename from geotranz/readme.txt rename to external/geotranz/readme.txt diff --git a/geotranz/releasenotes.txt b/external/geotranz/releasenotes.txt similarity index 100% rename from geotranz/releasenotes.txt rename to external/geotranz/releasenotes.txt diff --git a/geotranz/tranmerc.c b/external/geotranz/tranmerc.c similarity index 100% rename from geotranz/tranmerc.c rename to external/geotranz/tranmerc.c diff --git a/geotranz/tranmerc.h b/external/geotranz/tranmerc.h similarity index 100% rename from geotranz/tranmerc.h rename to external/geotranz/tranmerc.h diff --git a/geotranz/ups.c b/external/geotranz/ups.c similarity index 100% rename from geotranz/ups.c rename to external/geotranz/ups.c diff --git a/geotranz/ups.h b/external/geotranz/ups.h similarity index 100% rename from geotranz/ups.h rename to external/geotranz/ups.h diff --git a/geotranz/usng.c b/external/geotranz/usng.c similarity index 99% rename from geotranz/usng.c rename to external/geotranz/usng.c index 3db24b32..fdd2fba7 100644 --- a/geotranz/usng.c +++ b/external/geotranz/usng.c @@ -367,7 +367,7 @@ long Make_USNG_String (char* USNG, if (Zone) i = sprintf (USNG+i,"%2.2ld",Zone); else - strncpy(USNG, " ", 2); // 2 spaces + strcpy(USNG, " "); // 2 spaces - Should i be set to 2? for (j=0;j<3;j++) USNG[i++] = alphabet[Letters[j]]; @@ -782,7 +782,7 @@ long Convert_USNG_To_Geodetic (char* USNG, */ { /* Convert_USNG_To_Geodetic */ long zone; - char hemisphere; + char hemisphere = '?'; double easting; double northing; long zone_exists; @@ -1178,9 +1178,9 @@ long Convert_USNG_To_UPS ( char *USNG, double false_northing; /* False northing for 3rd letter */ double grid_easting; /* easting for 100,000 meter grid square */ double grid_northing; /* northing for 100,000 meter grid square */ - long zone; + long zone = 0; long letters[USNG_LETTERS]; - long in_precision; + long in_precision = 0; int index = 0; long error_code = USNG_NO_ERROR; diff --git a/geotranz/usng.h b/external/geotranz/usng.h similarity index 100% rename from geotranz/usng.h rename to external/geotranz/usng.h diff --git a/geotranz/utm.c b/external/geotranz/utm.c similarity index 100% rename from geotranz/utm.c rename to external/geotranz/utm.c diff --git a/geotranz/utm.h b/external/geotranz/utm.h similarity index 100% rename from geotranz/utm.h rename to external/geotranz/utm.h diff --git a/external/hidapi/CMakeLists.txt b/external/hidapi/CMakeLists.txt new file mode 100644 index 00000000..c9dd66f1 --- /dev/null +++ b/external/hidapi/CMakeLists.txt @@ -0,0 +1,21 @@ +set(HIDAPI_LIBRARIES "" CACHE INTERNAL "") + +if(WIN32 OR CYGWIN) # windows + + set(HIDAPI_LIBRARIES hidapi CACHE INTERNAL "hidapi") + + list(APPEND hidapi_SOURCES + # Functions for accessing HID devices on Windows. + # These were copied from https://github.com/libusb/hidapi + ${CUSTOM_HIDAPI_DIR}/hid.c + ) + + add_library(hidapi STATIC + ${hidapi_SOURCES} + ) + + set_target_properties(hidapi + PROPERTIES COMPILE_FLAGS "-Dbool=int -Dtrue=1 -Dfalse=0 -DUSE_HIDAPI_STATIC" + ) + +endif() diff --git a/external/hidapi/LICENSE-bsd.txt b/external/hidapi/LICENSE-bsd.txt new file mode 100644 index 00000000..538cdf95 --- /dev/null +++ b/external/hidapi/LICENSE-bsd.txt @@ -0,0 +1,26 @@ +Copyright (c) 2010, Alan Ott, Signal 11 Software +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + * Neither the name of Signal 11 Software nor the names of its + contributors may be used to endorse or promote products derived from + this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE +ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE +LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +POSSIBILITY OF SUCH DAMAGE. diff --git a/external/hidapi/LICENSE-gpl3.txt b/external/hidapi/LICENSE-gpl3.txt new file mode 100644 index 00000000..94a9ed02 --- /dev/null +++ b/external/hidapi/LICENSE-gpl3.txt @@ -0,0 +1,674 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + 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 . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright (C) + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. diff --git a/external/hidapi/LICENSE-orig.txt b/external/hidapi/LICENSE-orig.txt new file mode 100644 index 00000000..e3f33808 --- /dev/null +++ b/external/hidapi/LICENSE-orig.txt @@ -0,0 +1,9 @@ + HIDAPI - Multi-Platform library for + communication with HID devices. + + Copyright 2009, Alan Ott, Signal 11 Software. + All Rights Reserved. + + This software may be used by anyone for any reason so + long as the copyright notice in the source files + remains intact. diff --git a/external/hidapi/LICENSE.txt b/external/hidapi/LICENSE.txt new file mode 100644 index 00000000..e1676d4c --- /dev/null +++ b/external/hidapi/LICENSE.txt @@ -0,0 +1,13 @@ +HIDAPI can be used under one of three licenses. + +1. The GNU General Public License, version 3.0, in LICENSE-gpl3.txt +2. A BSD-Style License, in LICENSE-bsd.txt. +3. The more liberal original HIDAPI license. LICENSE-orig.txt + +The license chosen is at the discretion of the user of HIDAPI. For example: +1. An author of GPL software would likely use HIDAPI under the terms of the +GPL. + +2. An author of commercial closed-source software would likely use HIDAPI +under the terms of the BSD-style license or the original HIDAPI license. + diff --git a/external/hidapi/README.md b/external/hidapi/README.md new file mode 100644 index 00000000..6ced0f91 --- /dev/null +++ b/external/hidapi/README.md @@ -0,0 +1 @@ +This is from https://github.com/libusb/hidapi diff --git a/external/hidapi/hid.c b/external/hidapi/hid.c new file mode 100644 index 00000000..e483cd4f --- /dev/null +++ b/external/hidapi/hid.c @@ -0,0 +1,1091 @@ +/******************************************************* + HIDAPI - Multi-Platform library for + communication with HID devices. + + Alan Ott + Signal 11 Software + + 8/22/2009 + + Copyright 2009, All Rights Reserved. + + At the discretion of the user of this library, + this software may be licensed under the terms of the + GNU General Public License v3, a BSD-Style license, or the + original HIDAPI license as outlined in the LICENSE.txt, + LICENSE-gpl3.txt, LICENSE-bsd.txt, and LICENSE-orig.txt + files located at the root of the source distribution. + These files may also be found in the public source + code repository located at: + https://github.com/libusb/hidapi . +********************************************************/ + +#include + +#ifndef _NTDEF_ +typedef LONG NTSTATUS; +#endif + +#ifdef __MINGW32__ +#include +#include +#endif + +#ifdef __CYGWIN__ +#include +#define _wcsdup wcsdup +#endif + +/* The maximum number of characters that can be passed into the + HidD_Get*String() functions without it failing.*/ +#define MAX_STRING_WCHARS 0xFFF + +/*#define HIDAPI_USE_DDK*/ + +#ifdef __cplusplus +extern "C" { +#endif + #include + #include + #ifdef HIDAPI_USE_DDK + #include + #endif + + /* Copied from inc/ddk/hidclass.h, part of the Windows DDK. */ + #define HID_OUT_CTL_CODE(id) \ + CTL_CODE(FILE_DEVICE_KEYBOARD, (id), METHOD_OUT_DIRECT, FILE_ANY_ACCESS) + #define IOCTL_HID_GET_FEATURE HID_OUT_CTL_CODE(100) + #define IOCTL_HID_GET_INPUT_REPORT HID_OUT_CTL_CODE(104) + +#ifdef __cplusplus +} /* extern "C" */ +#endif + +#include +#include + + +#include "hidapi.h" + +#undef MIN +#define MIN(x,y) ((x) < (y)? (x): (y)) + +#ifdef _MSC_VER + /* Thanks Microsoft, but I know how to use strncpy(). */ + #pragma warning(disable:4996) +#endif + +#ifdef __cplusplus +extern "C" { +#endif + +static struct hid_api_version api_version = { + .major = HID_API_VERSION_MAJOR, + .minor = HID_API_VERSION_MINOR, + .patch = HID_API_VERSION_PATCH +}; + +#ifndef HIDAPI_USE_DDK + /* Since we're not building with the DDK, and the HID header + files aren't part of the SDK, we have to define all this + stuff here. In lookup_functions(), the function pointers + defined below are set. */ + typedef struct _HIDD_ATTRIBUTES{ + ULONG Size; + USHORT VendorID; + USHORT ProductID; + USHORT VersionNumber; + } HIDD_ATTRIBUTES, *PHIDD_ATTRIBUTES; + + typedef USHORT USAGE; + typedef struct _HIDP_CAPS { + USAGE Usage; + USAGE UsagePage; + USHORT InputReportByteLength; + USHORT OutputReportByteLength; + USHORT FeatureReportByteLength; + USHORT Reserved[17]; + USHORT fields_not_used_by_hidapi[10]; + } HIDP_CAPS, *PHIDP_CAPS; + typedef void* PHIDP_PREPARSED_DATA; + #define HIDP_STATUS_SUCCESS 0x110000 + + typedef BOOLEAN (__stdcall *HidD_GetAttributes_)(HANDLE device, PHIDD_ATTRIBUTES attrib); + typedef BOOLEAN (__stdcall *HidD_GetSerialNumberString_)(HANDLE device, PVOID buffer, ULONG buffer_len); + typedef BOOLEAN (__stdcall *HidD_GetManufacturerString_)(HANDLE handle, PVOID buffer, ULONG buffer_len); + typedef BOOLEAN (__stdcall *HidD_GetProductString_)(HANDLE handle, PVOID buffer, ULONG buffer_len); + typedef BOOLEAN (__stdcall *HidD_SetFeature_)(HANDLE handle, PVOID data, ULONG length); + typedef BOOLEAN (__stdcall *HidD_GetFeature_)(HANDLE handle, PVOID data, ULONG length); + typedef BOOLEAN (__stdcall *HidD_GetInputReport_)(HANDLE handle, PVOID data, ULONG length); + typedef BOOLEAN (__stdcall *HidD_GetIndexedString_)(HANDLE handle, ULONG string_index, PVOID buffer, ULONG buffer_len); + typedef BOOLEAN (__stdcall *HidD_GetPreparsedData_)(HANDLE handle, PHIDP_PREPARSED_DATA *preparsed_data); + typedef BOOLEAN (__stdcall *HidD_FreePreparsedData_)(PHIDP_PREPARSED_DATA preparsed_data); + typedef NTSTATUS (__stdcall *HidP_GetCaps_)(PHIDP_PREPARSED_DATA preparsed_data, HIDP_CAPS *caps); + typedef BOOLEAN (__stdcall *HidD_SetNumInputBuffers_)(HANDLE handle, ULONG number_buffers); + + static HidD_GetAttributes_ HidD_GetAttributes; + static HidD_GetSerialNumberString_ HidD_GetSerialNumberString; + static HidD_GetManufacturerString_ HidD_GetManufacturerString; + static HidD_GetProductString_ HidD_GetProductString; + static HidD_SetFeature_ HidD_SetFeature; + static HidD_GetFeature_ HidD_GetFeature; + static HidD_GetInputReport_ HidD_GetInputReport; + static HidD_GetIndexedString_ HidD_GetIndexedString; + static HidD_GetPreparsedData_ HidD_GetPreparsedData; + static HidD_FreePreparsedData_ HidD_FreePreparsedData; + static HidP_GetCaps_ HidP_GetCaps; + static HidD_SetNumInputBuffers_ HidD_SetNumInputBuffers; + + static HMODULE lib_handle = NULL; + static BOOLEAN initialized = FALSE; +#endif /* HIDAPI_USE_DDK */ + +struct hid_device_ { + HANDLE device_handle; + BOOL blocking; + USHORT output_report_length; + size_t input_report_length; + USHORT feature_report_length; + unsigned char *feature_buf; + void *last_error_str; + DWORD last_error_num; + BOOL read_pending; + char *read_buf; + OVERLAPPED ol; + OVERLAPPED write_ol; +}; + +static hid_device *new_hid_device() +{ + hid_device *dev = (hid_device*) calloc(1, sizeof(hid_device)); + dev->device_handle = INVALID_HANDLE_VALUE; + dev->blocking = TRUE; + dev->output_report_length = 0; + dev->input_report_length = 0; + dev->feature_report_length = 0; + dev->feature_buf = NULL; + dev->last_error_str = NULL; + dev->last_error_num = 0; + dev->read_pending = FALSE; + dev->read_buf = NULL; + memset(&dev->ol, 0, sizeof(dev->ol)); + dev->ol.hEvent = CreateEvent(NULL, FALSE, FALSE /*initial state f=nonsignaled*/, NULL); + memset(&dev->write_ol, 0, sizeof(dev->write_ol)); + dev->write_ol.hEvent = CreateEvent(NULL, FALSE, FALSE /*inital state f=nonsignaled*/, NULL); + + return dev; +} + +static void free_hid_device(hid_device *dev) +{ + CloseHandle(dev->ol.hEvent); + CloseHandle(dev->write_ol.hEvent); + CloseHandle(dev->device_handle); + LocalFree(dev->last_error_str); + free(dev->feature_buf); + free(dev->read_buf); + free(dev); +} + +static void register_error(hid_device *dev, const char *op) +{ + WCHAR *ptr, *msg; + (void)op; // unreferenced param + FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | + FORMAT_MESSAGE_FROM_SYSTEM | + FORMAT_MESSAGE_IGNORE_INSERTS, + NULL, + GetLastError(), + MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), + (LPWSTR)&msg, 0/*sz*/, + NULL); + + /* Get rid of the CR and LF that FormatMessage() sticks at the + end of the message. Thanks Microsoft! */ + ptr = msg; + while (*ptr) { + if (*ptr == '\r') { + *ptr = 0x0000; + break; + } + ptr++; + } + + /* Store the message off in the Device entry so that + the hid_error() function can pick it up. */ + LocalFree(dev->last_error_str); + dev->last_error_str = msg; +} + +#ifndef HIDAPI_USE_DDK +static int lookup_functions() +{ + lib_handle = LoadLibraryA("hid.dll"); + if (lib_handle) { +#if defined(__GNUC__) +# pragma GCC diagnostic push +//# pragma GCC diagnostic ignored "-Wcast-function-type" +#endif +#define RESOLVE(x) x = (x##_)GetProcAddress(lib_handle, #x); if (!x) return -1; + RESOLVE(HidD_GetAttributes); + RESOLVE(HidD_GetSerialNumberString); + RESOLVE(HidD_GetManufacturerString); + RESOLVE(HidD_GetProductString); + RESOLVE(HidD_SetFeature); + RESOLVE(HidD_GetFeature); + RESOLVE(HidD_GetInputReport); + RESOLVE(HidD_GetIndexedString); + RESOLVE(HidD_GetPreparsedData); + RESOLVE(HidD_FreePreparsedData); + RESOLVE(HidP_GetCaps); + RESOLVE(HidD_SetNumInputBuffers); +#undef RESOLVE +#if defined(__GNUC__) +# pragma GCC diagnostic pop +#endif + } + else + return -1; + + return 0; +} +#endif + +static HANDLE open_device(const char *path, BOOL open_rw) +{ + HANDLE handle; + DWORD desired_access = (open_rw)? (GENERIC_WRITE | GENERIC_READ): 0; + DWORD share_mode = FILE_SHARE_READ|FILE_SHARE_WRITE; + + handle = CreateFileA(path, + desired_access, + share_mode, + NULL, + OPEN_EXISTING, + FILE_FLAG_OVERLAPPED,/*FILE_ATTRIBUTE_NORMAL,*/ + 0); + + return handle; +} + +HID_API_EXPORT const struct hid_api_version* HID_API_CALL hid_version() +{ + return &api_version; +} + +HID_API_EXPORT const char* HID_API_CALL hid_version_str() +{ + return HID_API_VERSION_STR; +} + +int HID_API_EXPORT hid_init(void) +{ +#ifndef HIDAPI_USE_DDK + if (!initialized) { + if (lookup_functions() < 0) { + hid_exit(); + return -1; + } + initialized = TRUE; + } +#endif + return 0; +} + +int HID_API_EXPORT hid_exit(void) +{ +#ifndef HIDAPI_USE_DDK + if (lib_handle) + FreeLibrary(lib_handle); + lib_handle = NULL; + initialized = FALSE; +#endif + return 0; +} + +struct hid_device_info HID_API_EXPORT * HID_API_CALL hid_enumerate(unsigned short vendor_id, unsigned short product_id) +{ + BOOL res; + struct hid_device_info *root = NULL; /* return object */ + struct hid_device_info *cur_dev = NULL; + + /* Windows objects for interacting with the driver. */ + GUID InterfaceClassGuid = {0x4d1e55b2, 0xf16f, 0x11cf, {0x88, 0xcb, 0x00, 0x11, 0x11, 0x00, 0x00, 0x30} }; + SP_DEVINFO_DATA devinfo_data; + SP_DEVICE_INTERFACE_DATA device_interface_data; + SP_DEVICE_INTERFACE_DETAIL_DATA_A *device_interface_detail_data = NULL; + HDEVINFO device_info_set = INVALID_HANDLE_VALUE; + int device_index = 0; + int i; + + if (hid_init() < 0) + return NULL; + + /* Initialize the Windows objects. */ + memset(&devinfo_data, 0x0, sizeof(devinfo_data)); + devinfo_data.cbSize = sizeof(SP_DEVINFO_DATA); + device_interface_data.cbSize = sizeof(SP_DEVICE_INTERFACE_DATA); + + /* Get information for all the devices belonging to the HID class. */ + device_info_set = SetupDiGetClassDevsA(&InterfaceClassGuid, NULL, NULL, DIGCF_PRESENT | DIGCF_DEVICEINTERFACE); + + /* Iterate over each device in the HID class, looking for the right one. */ + + for (;;) { + HANDLE write_handle = INVALID_HANDLE_VALUE; + DWORD required_size = 0; + HIDD_ATTRIBUTES attrib; + + res = SetupDiEnumDeviceInterfaces(device_info_set, + NULL, + &InterfaceClassGuid, + device_index, + &device_interface_data); + + if (!res) { + /* A return of FALSE from this function means that + there are no more devices. */ + break; + } + + /* Call with 0-sized detail size, and let the function + tell us how long the detail struct needs to be. The + size is put in &required_size. */ + res = SetupDiGetDeviceInterfaceDetailA(device_info_set, + &device_interface_data, + NULL, + 0, + &required_size, + NULL); + + /* Allocate a long enough structure for device_interface_detail_data. */ + device_interface_detail_data = (SP_DEVICE_INTERFACE_DETAIL_DATA_A*) malloc(required_size); + device_interface_detail_data->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA_A); + + /* Get the detailed data for this device. The detail data gives us + the device path for this device, which is then passed into + CreateFile() to get a handle to the device. */ + res = SetupDiGetDeviceInterfaceDetailA(device_info_set, + &device_interface_data, + device_interface_detail_data, + required_size, + NULL, + NULL); + + if (!res) { + /* register_error(dev, "Unable to call SetupDiGetDeviceInterfaceDetail"); + Continue to the next device. */ + goto cont; + } + + /* Make sure this device is of Setup Class "HIDClass" and has a + driver bound to it. */ + for (i = 0; ; i++) { + char driver_name[256]; + + /* Populate devinfo_data. This function will return failure + when there are no more interfaces left. */ + res = SetupDiEnumDeviceInfo(device_info_set, i, &devinfo_data); + if (!res) + goto cont; + + res = SetupDiGetDeviceRegistryPropertyA(device_info_set, &devinfo_data, + SPDRP_CLASS, NULL, (PBYTE)driver_name, sizeof(driver_name), NULL); + if (!res) + goto cont; + + if ((strcmp(driver_name, "HIDClass") == 0) || + (strcmp(driver_name, "Mouse") == 0) || + (strcmp(driver_name, "Keyboard") == 0)) { + /* See if there's a driver bound. */ + res = SetupDiGetDeviceRegistryPropertyA(device_info_set, &devinfo_data, + SPDRP_DRIVER, NULL, (PBYTE)driver_name, sizeof(driver_name), NULL); + if (res) + break; + } + } + + //wprintf(L"HandleName: %s\n", device_interface_detail_data->DevicePath); + + /* Open a handle to the device */ + write_handle = open_device(device_interface_detail_data->DevicePath, FALSE); + + /* Check validity of write_handle. */ + if (write_handle == INVALID_HANDLE_VALUE) { + /* Unable to open the device. */ + //register_error(dev, "CreateFile"); + goto cont_close; + } + + + /* Get the Vendor ID and Product ID for this device. */ + attrib.Size = sizeof(HIDD_ATTRIBUTES); + HidD_GetAttributes(write_handle, &attrib); + //wprintf(L"Product/Vendor: %x %x\n", attrib.ProductID, attrib.VendorID); + + /* Check the VID/PID to see if we should add this + device to the enumeration list. */ + if ((vendor_id == 0x0 || attrib.VendorID == vendor_id) && + (product_id == 0x0 || attrib.ProductID == product_id)) { + + #define WSTR_LEN 512 + const char *str; + struct hid_device_info *tmp; + PHIDP_PREPARSED_DATA pp_data = NULL; + HIDP_CAPS caps; + NTSTATUS nt_res; + wchar_t wstr[WSTR_LEN]; /* TODO: Determine Size */ + size_t len; + + /* VID/PID match. Create the record. */ + tmp = (struct hid_device_info*) calloc(1, sizeof(struct hid_device_info)); + if (cur_dev) { + cur_dev->next = tmp; + } + else { + root = tmp; + } + cur_dev = tmp; + + /* Get the Usage Page and Usage for this device. */ + res = HidD_GetPreparsedData(write_handle, &pp_data); + if (res) { + nt_res = HidP_GetCaps(pp_data, &caps); + if (nt_res == HIDP_STATUS_SUCCESS) { + cur_dev->usage_page = caps.UsagePage; + cur_dev->usage = caps.Usage; + } + + HidD_FreePreparsedData(pp_data); + } + + /* Fill out the record */ + cur_dev->next = NULL; + str = device_interface_detail_data->DevicePath; + if (str) { + len = strlen(str); + cur_dev->path = (char*) calloc(len+1, sizeof(char)); + strncpy(cur_dev->path, str, len+1); + cur_dev->path[len] = '\0'; + } + else + cur_dev->path = NULL; + + /* Serial Number */ + wstr[0]= 0x0000; + res = HidD_GetSerialNumberString(write_handle, wstr, sizeof(wstr)); + wstr[WSTR_LEN-1] = 0x0000; + if (res) { + cur_dev->serial_number = _wcsdup(wstr); + } + + /* Manufacturer String */ + wstr[0]= 0x0000; + res = HidD_GetManufacturerString(write_handle, wstr, sizeof(wstr)); + wstr[WSTR_LEN-1] = 0x0000; + if (res) { + cur_dev->manufacturer_string = _wcsdup(wstr); + } + + /* Product String */ + wstr[0]= 0x0000; + res = HidD_GetProductString(write_handle, wstr, sizeof(wstr)); + wstr[WSTR_LEN-1] = 0x0000; + if (res) { + cur_dev->product_string = _wcsdup(wstr); + } + + /* VID/PID */ + cur_dev->vendor_id = attrib.VendorID; + cur_dev->product_id = attrib.ProductID; + + /* Release Number */ + cur_dev->release_number = attrib.VersionNumber; + + /* Interface Number. It can sometimes be parsed out of the path + on Windows if a device has multiple interfaces. See + http://msdn.microsoft.com/en-us/windows/hardware/gg487473 or + search for "Hardware IDs for HID Devices" at MSDN. If it's not + in the path, it's set to -1. */ + cur_dev->interface_number = -1; + if (cur_dev->path) { + char *interface_component = strstr(cur_dev->path, "&mi_"); + if (interface_component) { + char *hex_str = interface_component + 4; + char *endptr = NULL; + cur_dev->interface_number = strtol(hex_str, &endptr, 16); + if (endptr == hex_str) { + /* The parsing failed. Set interface_number to -1. */ + cur_dev->interface_number = -1; + } + } + } + } + +cont_close: + CloseHandle(write_handle); +cont: + /* We no longer need the detail data. It can be freed */ + free(device_interface_detail_data); + + device_index++; + + } + + /* Close the device information handle. */ + SetupDiDestroyDeviceInfoList(device_info_set); + + return root; + +} + +void HID_API_EXPORT HID_API_CALL hid_free_enumeration(struct hid_device_info *devs) +{ + /* TODO: Merge this with the Linux version. This function is platform-independent. */ + struct hid_device_info *d = devs; + while (d) { + struct hid_device_info *next = d->next; + free(d->path); + free(d->serial_number); + free(d->manufacturer_string); + free(d->product_string); + free(d); + d = next; + } +} + + +HID_API_EXPORT hid_device * HID_API_CALL hid_open(unsigned short vendor_id, unsigned short product_id, const wchar_t *serial_number) +{ + /* TODO: Merge this functions with the Linux version. This function should be platform independent. */ + struct hid_device_info *devs, *cur_dev; + const char *path_to_open = NULL; + hid_device *handle = NULL; + + devs = hid_enumerate(vendor_id, product_id); + cur_dev = devs; + while (cur_dev) { + if (cur_dev->vendor_id == vendor_id && + cur_dev->product_id == product_id) { + if (serial_number) { + if (cur_dev->serial_number && wcscmp(serial_number, cur_dev->serial_number) == 0) { + path_to_open = cur_dev->path; + break; + } + } + else { + path_to_open = cur_dev->path; + break; + } + } + cur_dev = cur_dev->next; + } + + if (path_to_open) { + /* Open the device */ + handle = hid_open_path(path_to_open); + } + + hid_free_enumeration(devs); + + return handle; +} + +HID_API_EXPORT hid_device * HID_API_CALL hid_open_path(const char *path) +{ + hid_device *dev; + HIDP_CAPS caps; + PHIDP_PREPARSED_DATA pp_data = NULL; + BOOLEAN res; + NTSTATUS nt_res; + + if (hid_init() < 0) { + return NULL; + } + + dev = new_hid_device(); + + /* Open a handle to the device */ + dev->device_handle = open_device(path, TRUE); + + /* Check validity of write_handle. */ + if (dev->device_handle == INVALID_HANDLE_VALUE) { + /* System devices, such as keyboards and mice, cannot be opened in + read-write mode, because the system takes exclusive control over + them. This is to prevent keyloggers. However, feature reports + can still be sent and received. Retry opening the device, but + without read/write access. */ + dev->device_handle = open_device(path, FALSE); + + /* Check the validity of the limited device_handle. */ + if (dev->device_handle == INVALID_HANDLE_VALUE) { + /* Unable to open the device, even without read-write mode. */ + register_error(dev, "CreateFile"); + goto err; + } + } + + /* Set the Input Report buffer size to 64 reports. */ + res = HidD_SetNumInputBuffers(dev->device_handle, 64); + if (!res) { + register_error(dev, "HidD_SetNumInputBuffers"); + goto err; + } + + /* Get the Input Report length for the device. */ + res = HidD_GetPreparsedData(dev->device_handle, &pp_data); + if (!res) { + register_error(dev, "HidD_GetPreparsedData"); + goto err; + } + nt_res = HidP_GetCaps(pp_data, &caps); + if (nt_res != HIDP_STATUS_SUCCESS) { + register_error(dev, "HidP_GetCaps"); + goto err_pp_data; + } + dev->output_report_length = caps.OutputReportByteLength; + dev->input_report_length = caps.InputReportByteLength; + dev->feature_report_length = caps.FeatureReportByteLength; + HidD_FreePreparsedData(pp_data); + + dev->read_buf = (char*) malloc(dev->input_report_length); + + return dev; + +err_pp_data: + HidD_FreePreparsedData(pp_data); +err: + free_hid_device(dev); + return NULL; +} + +int HID_API_EXPORT HID_API_CALL hid_write(hid_device *dev, const unsigned char *data, size_t length) +{ + DWORD bytes_written = 0; + int function_result = -1; + BOOL res; + BOOL overlapped = FALSE; + + unsigned char *buf; + + /* Make sure the right number of bytes are passed to WriteFile. Windows + expects the number of bytes which are in the _longest_ report (plus + one for the report number) bytes even if the data is a report + which is shorter than that. Windows gives us this value in + caps.OutputReportByteLength. If a user passes in fewer bytes than this, + create a temporary buffer which is the proper size. */ + if (length >= dev->output_report_length) { + /* The user passed the right number of bytes. Use the buffer as-is. */ + buf = (unsigned char *) data; + } else { + /* Create a temporary buffer and copy the user's data + into it, padding the rest with zeros. */ + buf = (unsigned char *) malloc(dev->output_report_length); + memcpy(buf, data, length); + memset(buf + length, 0, dev->output_report_length - length); + length = dev->output_report_length; + } + + res = WriteFile(dev->device_handle, buf, (DWORD) length, NULL, &dev->write_ol); + + if (!res) { + if (GetLastError() != ERROR_IO_PENDING) { + /* WriteFile() failed. Return error. */ + register_error(dev, "WriteFile"); + goto end_of_function; + } + overlapped = TRUE; + } + + if (overlapped) { + /* Wait for the transaction to complete. This makes + hid_write() synchronous. */ + res = WaitForSingleObject(dev->write_ol.hEvent, 1000); + if (res != WAIT_OBJECT_0) { + /* There was a Timeout. */ + register_error(dev, "WriteFile/WaitForSingleObject Timeout"); + goto end_of_function; + } + + /* Get the result. */ + res = GetOverlappedResult(dev->device_handle, &dev->write_ol, &bytes_written, FALSE/*wait*/); + if (res) { + function_result = bytes_written; + } + else { + /* The Write operation failed. */ + register_error(dev, "WriteFile"); + goto end_of_function; + } + } + +end_of_function: + if (buf != data) + free(buf); + + return function_result; +} + + +int HID_API_EXPORT HID_API_CALL hid_read_timeout(hid_device *dev, unsigned char *data, size_t length, int milliseconds) +{ + DWORD bytes_read = 0; + size_t copy_len = 0; + BOOL res = FALSE; + BOOL overlapped = FALSE; + + /* Copy the handle for convenience. */ + HANDLE ev = dev->ol.hEvent; + + if (!dev->read_pending) { + /* Start an Overlapped I/O read. */ + dev->read_pending = TRUE; + memset(dev->read_buf, 0, dev->input_report_length); + ResetEvent(ev); + res = ReadFile(dev->device_handle, dev->read_buf, (DWORD) dev->input_report_length, &bytes_read, &dev->ol); + + if (!res) { + if (GetLastError() != ERROR_IO_PENDING) { + /* ReadFile() has failed. + Clean up and return error. */ + CancelIo(dev->device_handle); + dev->read_pending = FALSE; + goto end_of_function; + } + overlapped = TRUE; + } + } + else { + overlapped = TRUE; + } + + if (overlapped) { + if (milliseconds >= 0) { + /* See if there is any data yet. */ + res = WaitForSingleObject(ev, milliseconds); + if (res != WAIT_OBJECT_0) { + /* There was no data this time. Return zero bytes available, + but leave the Overlapped I/O running. */ + return 0; + } + } + + /* Either WaitForSingleObject() told us that ReadFile has completed, or + we are in non-blocking mode. Get the number of bytes read. The actual + data has been copied to the data[] array which was passed to ReadFile(). */ + res = GetOverlappedResult(dev->device_handle, &dev->ol, &bytes_read, TRUE/*wait*/); + } + /* Set pending back to false, even if GetOverlappedResult() returned error. */ + dev->read_pending = FALSE; + + if (res && bytes_read > 0) { + if (dev->read_buf[0] == 0x0) { + /* If report numbers aren't being used, but Windows sticks a report + number (0x0) on the beginning of the report anyway. To make this + work like the other platforms, and to make it work more like the + HID spec, we'll skip over this byte. */ + bytes_read--; + copy_len = length > bytes_read ? bytes_read : length; + memcpy(data, dev->read_buf+1, copy_len); + } + else { + /* Copy the whole buffer, report number and all. */ + copy_len = length > bytes_read ? bytes_read : length; + memcpy(data, dev->read_buf, copy_len); + } + } + +end_of_function: + if (!res) { + register_error(dev, "GetOverlappedResult"); + return -1; + } + + return (int) copy_len; +} + +int HID_API_EXPORT HID_API_CALL hid_read(hid_device *dev, unsigned char *data, size_t length) +{ + return hid_read_timeout(dev, data, length, (dev->blocking)? -1: 0); +} + +int HID_API_EXPORT HID_API_CALL hid_set_nonblocking(hid_device *dev, int nonblock) +{ + dev->blocking = !nonblock; + return 0; /* Success */ +} + +int HID_API_EXPORT HID_API_CALL hid_send_feature_report(hid_device *dev, const unsigned char *data, size_t length) +{ + BOOL res = FALSE; + unsigned char *buf; + size_t length_to_send; + + /* Windows expects at least caps.FeatureReportByteLength bytes passed + to HidD_SetFeature(), even if the report is shorter. Any less sent and + the function fails with error ERROR_INVALID_PARAMETER set. Any more + and HidD_SetFeature() silently truncates the data sent in the report + to caps.FeatureReportByteLength. */ + if (length >= dev->feature_report_length) { + buf = (unsigned char *) data; + length_to_send = length; + } else { + if (dev->feature_buf == NULL) + dev->feature_buf = (unsigned char *) malloc(dev->feature_report_length); + buf = dev->feature_buf; + memcpy(buf, data, length); + memset(buf + length, 0, dev->feature_report_length - length); + length_to_send = dev->feature_report_length; + } + + res = HidD_SetFeature(dev->device_handle, (PVOID)buf, (DWORD) length_to_send); + + if (!res) { + register_error(dev, "HidD_SetFeature"); + return -1; + } + + return (int) length; +} + + +int HID_API_EXPORT HID_API_CALL hid_get_feature_report(hid_device *dev, unsigned char *data, size_t length) +{ + BOOL res; +#if 0 + res = HidD_GetFeature(dev->device_handle, data, length); + if (!res) { + register_error(dev, "HidD_GetFeature"); + return -1; + } + return 0; /* HidD_GetFeature() doesn't give us an actual length, unfortunately */ +#else + DWORD bytes_returned; + + OVERLAPPED ol; + memset(&ol, 0, sizeof(ol)); + + res = DeviceIoControl(dev->device_handle, + IOCTL_HID_GET_FEATURE, + data, (DWORD) length, + data, (DWORD) length, + &bytes_returned, &ol); + + if (!res) { + if (GetLastError() != ERROR_IO_PENDING) { + /* DeviceIoControl() failed. Return error. */ + register_error(dev, "Send Feature Report DeviceIoControl"); + return -1; + } + } + + /* Wait here until the write is done. This makes + hid_get_feature_report() synchronous. */ + res = GetOverlappedResult(dev->device_handle, &ol, &bytes_returned, TRUE/*wait*/); + if (!res) { + /* The operation failed. */ + register_error(dev, "Send Feature Report GetOverLappedResult"); + return -1; + } + + /* bytes_returned does not include the first byte which contains the + report ID. The data buffer actually contains one more byte than + bytes_returned. */ + bytes_returned++; + + return bytes_returned; +#endif +} + + +int HID_API_EXPORT HID_API_CALL hid_get_input_report(hid_device *dev, unsigned char *data, size_t length) +{ + BOOL res; +#if 0 + res = HidD_GetInputReport(dev->device_handle, data, length); + if (!res) { + register_error(dev, "HidD_GetInputReport"); + return -1; + } + return length; +#else + DWORD bytes_returned; + + OVERLAPPED ol; + memset(&ol, 0, sizeof(ol)); + + res = DeviceIoControl(dev->device_handle, + IOCTL_HID_GET_INPUT_REPORT, + data, (DWORD) length, + data, (DWORD) length, + &bytes_returned, &ol); + + if (!res) { + if (GetLastError() != ERROR_IO_PENDING) { + /* DeviceIoControl() failed. Return error. */ + register_error(dev, "Send Input Report DeviceIoControl"); + return -1; + } + } + + /* Wait here until the write is done. This makes + hid_get_feature_report() synchronous. */ + res = GetOverlappedResult(dev->device_handle, &ol, &bytes_returned, TRUE/*wait*/); + if (!res) { + /* The operation failed. */ + register_error(dev, "Send Input Report GetOverLappedResult"); + return -1; + } + + /* bytes_returned does not include the first byte which contains the + report ID. The data buffer actually contains one more byte than + bytes_returned. */ + bytes_returned++; + + return bytes_returned; +#endif +} + +void HID_API_EXPORT HID_API_CALL hid_close(hid_device *dev) +{ + if (!dev) + return; + CancelIo(dev->device_handle); + free_hid_device(dev); +} + +int HID_API_EXPORT_CALL HID_API_CALL hid_get_manufacturer_string(hid_device *dev, wchar_t *string, size_t maxlen) +{ + BOOL res; + + res = HidD_GetManufacturerString(dev->device_handle, string, sizeof(wchar_t) * (DWORD) MIN(maxlen, MAX_STRING_WCHARS)); + if (!res) { + register_error(dev, "HidD_GetManufacturerString"); + return -1; + } + + return 0; +} + +int HID_API_EXPORT_CALL HID_API_CALL hid_get_product_string(hid_device *dev, wchar_t *string, size_t maxlen) +{ + BOOL res; + + res = HidD_GetProductString(dev->device_handle, string, sizeof(wchar_t) * (DWORD) MIN(maxlen, MAX_STRING_WCHARS)); + if (!res) { + register_error(dev, "HidD_GetProductString"); + return -1; + } + + return 0; +} + +int HID_API_EXPORT_CALL HID_API_CALL hid_get_serial_number_string(hid_device *dev, wchar_t *string, size_t maxlen) +{ + BOOL res; + + res = HidD_GetSerialNumberString(dev->device_handle, string, sizeof(wchar_t) * (DWORD) MIN(maxlen, MAX_STRING_WCHARS)); + if (!res) { + register_error(dev, "HidD_GetSerialNumberString"); + return -1; + } + + return 0; +} + +int HID_API_EXPORT_CALL HID_API_CALL hid_get_indexed_string(hid_device *dev, int string_index, wchar_t *string, size_t maxlen) +{ + BOOL res; + + res = HidD_GetIndexedString(dev->device_handle, string_index, string, sizeof(wchar_t) * (DWORD) MIN(maxlen, MAX_STRING_WCHARS)); + if (!res) { + register_error(dev, "HidD_GetIndexedString"); + return -1; + } + + return 0; +} + + +HID_API_EXPORT const wchar_t * HID_API_CALL hid_error(hid_device *dev) +{ + if (dev) { + if (dev->last_error_str == NULL) + return L"Success"; + return (wchar_t*)dev->last_error_str; + } + + // Global error messages are not (yet) implemented on Windows. + return L"hid_error for global errors is not implemented yet"; +} + + +/*#define PICPGM*/ +/*#define S11*/ +#define P32 +#ifdef S11 + unsigned short VendorID = 0xa0a0; + unsigned short ProductID = 0x0001; +#endif + +#ifdef P32 + unsigned short VendorID = 0x04d8; + unsigned short ProductID = 0x3f; +#endif + + +#ifdef PICPGM + unsigned short VendorID = 0x04d8; + unsigned short ProductID = 0x0033; +#endif + + +#if 0 +int __cdecl main(int argc, char* argv[]) +{ + int res; + unsigned char buf[65]; + + UNREFERENCED_PARAMETER(argc); + UNREFERENCED_PARAMETER(argv); + + /* Set up the command buffer. */ + memset(buf,0x00,sizeof(buf)); + buf[0] = 0; + buf[1] = 0x81; + + + /* Open the device. */ + int handle = open(VendorID, ProductID, L"12345"); + if (handle < 0) + printf("unable to open device\n"); + + + /* Toggle LED (cmd 0x80) */ + buf[1] = 0x80; + res = write(handle, buf, 65); + if (res < 0) + printf("Unable to write()\n"); + + /* Request state (cmd 0x81) */ + buf[1] = 0x81; + write(handle, buf, 65); + if (res < 0) + printf("Unable to write() (2)\n"); + + /* Read requested state */ + read(handle, buf, 65); + if (res < 0) + printf("Unable to read()\n"); + + /* Print out the returned buffer. */ + for (int i = 0; i < 4; i++) + printf("buf[%d]: %d\n", i, buf[i]); + + return 0; +} +#endif + +#ifdef __cplusplus +} /* extern "C" */ +#endif diff --git a/external/hidapi/hidapi.h b/external/hidapi/hidapi.h new file mode 100644 index 00000000..efba3188 --- /dev/null +++ b/external/hidapi/hidapi.h @@ -0,0 +1,498 @@ +/******************************************************* + HIDAPI - Multi-Platform library for + communication with HID devices. + + Alan Ott + Signal 11 Software + + 8/22/2009 + + Copyright 2009, All Rights Reserved. + + At the discretion of the user of this library, + this software may be licensed under the terms of the + GNU General Public License v3, a BSD-Style license, or the + original HIDAPI license as outlined in the LICENSE.txt, + LICENSE-gpl3.txt, LICENSE-bsd.txt, and LICENSE-orig.txt + files located at the root of the source distribution. + These files may also be found in the public source + code repository located at: + https://github.com/libusb/hidapi . +********************************************************/ + +/** @file + * @defgroup API hidapi API + */ + +#ifndef HIDAPI_H__ +#define HIDAPI_H__ + +#include + +#ifdef _WIN32 + #define HID_API_EXPORT __declspec(dllexport) + #define HID_API_CALL +#else + #define HID_API_EXPORT /**< API export macro */ + #define HID_API_CALL /**< API call macro */ +#endif + +#define HID_API_EXPORT_CALL HID_API_EXPORT HID_API_CALL /**< API export and call macro*/ + +/** @brief Static/compile-time major version of the library. + + @ingroup API +*/ +#define HID_API_VERSION_MAJOR 0 +/** @brief Static/compile-time minor version of the library. + + @ingroup API +*/ +#define HID_API_VERSION_MINOR 10 +/** @brief Static/compile-time patch version of the library. + + @ingroup API +*/ +#define HID_API_VERSION_PATCH 1 + +/* Helper macros */ +#define HID_API_AS_STR_IMPL(x) #x +#define HID_API_AS_STR(x) HID_API_AS_STR_IMPL(x) +#define HID_API_TO_VERSION_STR(v1, v2, v3) HID_API_AS_STR(v1.v2.v3) + +/** @brief Static/compile-time string version of the library. + + @ingroup API +*/ +#define HID_API_VERSION_STR HID_API_TO_VERSION_STR(HID_API_VERSION_MAJOR, HID_API_VERSION_MINOR, HID_API_VERSION_PATCH) + +#ifdef __cplusplus +extern "C" { +#endif + struct hid_api_version { + int major; + int minor; + int patch; + }; + + struct hid_device_; + typedef struct hid_device_ hid_device; /**< opaque hidapi structure */ + + /** hidapi info structure */ + struct hid_device_info { + /** Platform-specific device path */ + char *path; + /** Device Vendor ID */ + unsigned short vendor_id; + /** Device Product ID */ + unsigned short product_id; + /** Serial Number */ + wchar_t *serial_number; + /** Device Release Number in binary-coded decimal, + also known as Device Version Number */ + unsigned short release_number; + /** Manufacturer String */ + wchar_t *manufacturer_string; + /** Product string */ + wchar_t *product_string; + /** Usage Page for this Device/Interface + (Windows/Mac/hidraw only) */ + unsigned short usage_page; + /** Usage for this Device/Interface + (Windows/Mac/hidraw only) */ + unsigned short usage; + /** The USB interface which this logical device + represents. + + * Valid on both Linux implementations in all cases. + * Valid on the Windows implementation only if the device + contains more than one interface. + * Valid on the Mac implementation if and only if the device + is a USB HID device. */ + int interface_number; + + /** Pointer to the next device */ + struct hid_device_info *next; + }; + + + /** @brief Initialize the HIDAPI library. + + This function initializes the HIDAPI library. Calling it is not + strictly necessary, as it will be called automatically by + hid_enumerate() and any of the hid_open_*() functions if it is + needed. This function should be called at the beginning of + execution however, if there is a chance of HIDAPI handles + being opened by different threads simultaneously. + + @ingroup API + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_init(void); + + /** @brief Finalize the HIDAPI library. + + This function frees all of the static data associated with + HIDAPI. It should be called at the end of execution to avoid + memory leaks. + + @ingroup API + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_exit(void); + + /** @brief Enumerate the HID Devices. + + This function returns a linked list of all the HID devices + attached to the system which match vendor_id and product_id. + If @p vendor_id is set to 0 then any vendor matches. + If @p product_id is set to 0 then any product matches. + If @p vendor_id and @p product_id are both set to 0, then + all HID devices will be returned. + + @ingroup API + @param vendor_id The Vendor ID (VID) of the types of device + to open. + @param product_id The Product ID (PID) of the types of + device to open. + + @returns + This function returns a pointer to a linked list of type + struct #hid_device_info, containing information about the HID devices + attached to the system, or NULL in the case of failure. Free + this linked list by calling hid_free_enumeration(). + */ + struct hid_device_info HID_API_EXPORT * HID_API_CALL hid_enumerate(unsigned short vendor_id, unsigned short product_id); + + /** @brief Free an enumeration Linked List + + This function frees a linked list created by hid_enumerate(). + + @ingroup API + @param devs Pointer to a list of struct_device returned from + hid_enumerate(). + */ + void HID_API_EXPORT HID_API_CALL hid_free_enumeration(struct hid_device_info *devs); + + /** @brief Open a HID device using a Vendor ID (VID), Product ID + (PID) and optionally a serial number. + + If @p serial_number is NULL, the first device with the + specified VID and PID is opened. + + This function sets the return value of hid_error(). + + @ingroup API + @param vendor_id The Vendor ID (VID) of the device to open. + @param product_id The Product ID (PID) of the device to open. + @param serial_number The Serial Number of the device to open + (Optionally NULL). + + @returns + This function returns a pointer to a #hid_device object on + success or NULL on failure. + */ + HID_API_EXPORT hid_device * HID_API_CALL hid_open(unsigned short vendor_id, unsigned short product_id, const wchar_t *serial_number); + + /** @brief Open a HID device by its path name. + + The path name be determined by calling hid_enumerate(), or a + platform-specific path name can be used (eg: /dev/hidraw0 on + Linux). + + This function sets the return value of hid_error(). + + @ingroup API + @param path The path name of the device to open + + @returns + This function returns a pointer to a #hid_device object on + success or NULL on failure. + */ + HID_API_EXPORT hid_device * HID_API_CALL hid_open_path(const char *path); + + /** @brief Write an Output report to a HID device. + + The first byte of @p data[] must contain the Report ID. For + devices which only support a single report, this must be set + to 0x0. The remaining bytes contain the report data. Since + the Report ID is mandatory, calls to hid_write() will always + contain one more byte than the report contains. For example, + if a hid report is 16 bytes long, 17 bytes must be passed to + hid_write(), the Report ID (or 0x0, for devices with a + single report), followed by the report data (16 bytes). In + this example, the length passed in would be 17. + + hid_write() will send the data on the first OUT endpoint, if + one exists. If it does not, it will send the data through + the Control Endpoint (Endpoint 0). + + This function sets the return value of hid_error(). + + @ingroup API + @param dev A device handle returned from hid_open(). + @param data The data to send, including the report number as + the first byte. + @param length The length in bytes of the data to send. + + @returns + This function returns the actual number of bytes written and + -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_write(hid_device *dev, const unsigned char *data, size_t length); + + /** @brief Read an Input report from a HID device with timeout. + + Input reports are returned + to the host through the INTERRUPT IN endpoint. The first byte will + contain the Report number if the device uses numbered reports. + + This function sets the return value of hid_error(). + + @ingroup API + @param dev A device handle returned from hid_open(). + @param data A buffer to put the read data into. + @param length The number of bytes to read. For devices with + multiple reports, make sure to read an extra byte for + the report number. + @param milliseconds timeout in milliseconds or -1 for blocking wait. + + @returns + This function returns the actual number of bytes read and + -1 on error. If no packet was available to be read within + the timeout period, this function returns 0. + */ + int HID_API_EXPORT HID_API_CALL hid_read_timeout(hid_device *dev, unsigned char *data, size_t length, int milliseconds); + + /** @brief Read an Input report from a HID device. + + Input reports are returned + to the host through the INTERRUPT IN endpoint. The first byte will + contain the Report number if the device uses numbered reports. + + This function sets the return value of hid_error(). + + @ingroup API + @param dev A device handle returned from hid_open(). + @param data A buffer to put the read data into. + @param length The number of bytes to read. For devices with + multiple reports, make sure to read an extra byte for + the report number. + + @returns + This function returns the actual number of bytes read and + -1 on error. If no packet was available to be read and + the handle is in non-blocking mode, this function returns 0. + */ + int HID_API_EXPORT HID_API_CALL hid_read(hid_device *dev, unsigned char *data, size_t length); + + /** @brief Set the device handle to be non-blocking. + + In non-blocking mode calls to hid_read() will return + immediately with a value of 0 if there is no data to be + read. In blocking mode, hid_read() will wait (block) until + there is data to read before returning. + + Nonblocking can be turned on and off at any time. + + @ingroup API + @param dev A device handle returned from hid_open(). + @param nonblock enable or not the nonblocking reads + - 1 to enable nonblocking + - 0 to disable nonblocking. + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_set_nonblocking(hid_device *dev, int nonblock); + + /** @brief Send a Feature report to the device. + + Feature reports are sent over the Control endpoint as a + Set_Report transfer. The first byte of @p data[] must + contain the Report ID. For devices which only support a + single report, this must be set to 0x0. The remaining bytes + contain the report data. Since the Report ID is mandatory, + calls to hid_send_feature_report() will always contain one + more byte than the report contains. For example, if a hid + report is 16 bytes long, 17 bytes must be passed to + hid_send_feature_report(): the Report ID (or 0x0, for + devices which do not use numbered reports), followed by the + report data (16 bytes). In this example, the length passed + in would be 17. + + This function sets the return value of hid_error(). + + @ingroup API + @param dev A device handle returned from hid_open(). + @param data The data to send, including the report number as + the first byte. + @param length The length in bytes of the data to send, including + the report number. + + @returns + This function returns the actual number of bytes written and + -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_send_feature_report(hid_device *dev, const unsigned char *data, size_t length); + + /** @brief Get a feature report from a HID device. + + Set the first byte of @p data[] to the Report ID of the + report to be read. Make sure to allow space for this + extra byte in @p data[]. Upon return, the first byte will + still contain the Report ID, and the report data will + start in data[1]. + + This function sets the return value of hid_error(). + + @ingroup API + @param dev A device handle returned from hid_open(). + @param data A buffer to put the read data into, including + the Report ID. Set the first byte of @p data[] to the + Report ID of the report to be read, or set it to zero + if your device does not use numbered reports. + @param length The number of bytes to read, including an + extra byte for the report ID. The buffer can be longer + than the actual report. + + @returns + This function returns the number of bytes read plus + one for the report ID (which is still in the first + byte), or -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_get_feature_report(hid_device *dev, unsigned char *data, size_t length); + + /** @brief Get a input report from a HID device. + + Set the first byte of @p data[] to the Report ID of the + report to be read. Make sure to allow space for this + extra byte in @p data[]. Upon return, the first byte will + still contain the Report ID, and the report data will + start in data[1]. + + @ingroup API + @param device A device handle returned from hid_open(). + @param data A buffer to put the read data into, including + the Report ID. Set the first byte of @p data[] to the + Report ID of the report to be read, or set it to zero + if your device does not use numbered reports. + @param length The number of bytes to read, including an + extra byte for the report ID. The buffer can be longer + than the actual report. + + @returns + This function returns the number of bytes read plus + one for the report ID (which is still in the first + byte), or -1 on error. + */ + int HID_API_EXPORT HID_API_CALL hid_get_input_report(hid_device *dev, unsigned char *data, size_t length); + + /** @brief Close a HID device. + + This function sets the return value of hid_error(). + + @ingroup API + @param dev A device handle returned from hid_open(). + */ + void HID_API_EXPORT HID_API_CALL hid_close(hid_device *dev); + + /** @brief Get The Manufacturer String from a HID device. + + @ingroup API + @param dev A device handle returned from hid_open(). + @param string A wide string buffer to put the data into. + @param maxlen The length of the buffer in multiples of wchar_t. + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT_CALL hid_get_manufacturer_string(hid_device *dev, wchar_t *string, size_t maxlen); + + /** @brief Get The Product String from a HID device. + + @ingroup API + @param dev A device handle returned from hid_open(). + @param string A wide string buffer to put the data into. + @param maxlen The length of the buffer in multiples of wchar_t. + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT_CALL hid_get_product_string(hid_device *dev, wchar_t *string, size_t maxlen); + + /** @brief Get The Serial Number String from a HID device. + + @ingroup API + @param dev A device handle returned from hid_open(). + @param string A wide string buffer to put the data into. + @param maxlen The length of the buffer in multiples of wchar_t. + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT_CALL hid_get_serial_number_string(hid_device *dev, wchar_t *string, size_t maxlen); + + /** @brief Get a string from a HID device, based on its string index. + + @ingroup API + @param dev A device handle returned from hid_open(). + @param string_index The index of the string to get. + @param string A wide string buffer to put the data into. + @param maxlen The length of the buffer in multiples of wchar_t. + + @returns + This function returns 0 on success and -1 on error. + */ + int HID_API_EXPORT_CALL hid_get_indexed_string(hid_device *dev, int string_index, wchar_t *string, size_t maxlen); + + /** @brief Get a string describing the last error which occurred. + + Whether a function sets the last error is noted in its + documentation. These functions will reset the last error + to NULL before their execution. + + Strings returned from hid_error() must not be freed by the user! + + This function is thread-safe, and error messages are thread-local. + + @ingroup API + @param dev A device handle returned from hid_open(), + or NULL to get the last non-device-specific error + (e.g. for errors in hid_open() itself). + + @returns + This function returns a string containing the last error + which occurred or NULL if none has occurred. + */ + HID_API_EXPORT const wchar_t* HID_API_CALL hid_error(hid_device *dev); + + /** @brief Get a runtime version of the library. + + @ingroup API + + @returns + Pointer to statically allocated struct, that contains version. + */ + HID_API_EXPORT const struct hid_api_version* HID_API_CALL hid_version(void); + + + /** @brief Get a runtime version string of the library. + + @ingroup API + + @returns + Pointer to statically allocated string, that contains version string. + */ + HID_API_EXPORT const char* HID_API_CALL hid_version_str(void); + +#ifdef __cplusplus +} +#endif + +#endif + diff --git a/external/misc/CMakeLists.txt b/external/misc/CMakeLists.txt new file mode 100644 index 00000000..685b89ad --- /dev/null +++ b/external/misc/CMakeLists.txt @@ -0,0 +1,73 @@ + +set(MISC_LIBRARIES misc CACHE INTERNAL "misc") + +include_directories( + ${CMAKE_SOURCE_DIR}/src + ) + +if(LINUX) +# Previously - +# list(APPEND misc_SOURCES +# # Provide our own copy of strlcpy and strlcat +# # because they are not included with Linux. +# ${CUSTOM_MISC_DIR}/strlcpy.c +# ${CUSTOM_MISC_DIR}/strlcat.c +# ) +# It seems that Alpine Linux and Void Linux have strlcpy and +# strlcat so we need to handle the situation more delicately. +# When doing it this way, there is probably no reason to +# distinguish between Linux and BSD-like systems here. +# If we kept going, the same thing could be done for each +# of the functions and no OS check would be needed. + + if (NOT HAVE_STRLCPY) + list(APPEND misc_SOURCES + ${CUSTOM_MISC_DIR}/strlcpy.c + ) + endif() + + if (NOT HAVE_STRLCAT) + list(APPEND misc_SOURCES + ${CUSTOM_MISC_DIR}/strlcat.c + ) + endif() + + # Add_library doesn't like to get an empty source file list. + # I tried several variations on this theme to test whether the list + # was not empty and was not successful in getting it to work + # on both Alpine and RPi. + #if("${misc_SOURCES}") + # This is less elegant and less maintainable but it works. + + if ((NOT HAVE_STRLCPY) OR (NOT HAVE_STRLCAT)) + add_library(misc STATIC + ${misc_SOURCES} + ) + else() + set(MISC_LIBRARIES "" CACHE INTERNAL "") + endif() + + + +elseif(WIN32 OR CYGWIN) # windows + + list(APPEND misc_SOURCES + # There are several string functions found in Linux + # but not on Windows. Need to provide our own copy. + ${CUSTOM_MISC_DIR}/strsep.c + ${CUSTOM_MISC_DIR}/strtok_r.c + ${CUSTOM_MISC_DIR}/strcasestr.c + ${CUSTOM_MISC_DIR}/strlcpy.c + ${CUSTOM_MISC_DIR}/strlcat.c + ) + + add_library(misc STATIC + ${misc_SOURCES} + ) + +else() + + # on macOS, OpenBSD and FreeBSD not misc is necessary + set(MISC_LIBRARIES "" CACHE INTERNAL "") + +endif() diff --git a/misc/README-dire-wolf.txt b/external/misc/README similarity index 52% rename from misc/README-dire-wolf.txt rename to external/misc/README index 140f9607..f69c05aa 100644 --- a/misc/README-dire-wolf.txt +++ b/external/misc/README @@ -4,31 +4,24 @@ Files in this directory fill in the gaps missing for some operating systems. -------------------------------------- -These are part of the standard C library for Linux and similar operating systems. -For the Windows version we need to include our own copy. +These are part of the standard C library for Linux, BSD Unix, and similar operating systems. +They are not present for MS Windows so we need to supply our own copy. -They were copied from Cygwin source. -/usr/src/cygwin-1.7.10-1/newlib/libc/string/... +From http://ftp.netbsd.org/pub/pkgsrc/current/pkgsrc/net/tnftp/files/libnetbsd/strsep.c +and other BSD locations. strsep.c - strtok_r.c - --------------------------------------- - -This was also missing on Windows but available everywhere else. - strcasestr.c + strtok_r.c -------------------------------------- -The are used for the Linux and Windows versions. +These are needed for the Linux and Windows versions. They should be part of the standard C library for OpenBSD, FreeBSD, Mac OS X. -These are from OpenBSD. + http://ftp.netbsd.org/pub/pkgsrc/current/pkgsrc/net/tnftp/files/libnetbsd/strlcpy.c http://ftp.netbsd.org/pub/pkgsrc/current/pkgsrc/net/tnftp/files/libnetbsd/strlcat.c - strlcpy.c - strlcat.c - \ No newline at end of file + strlcat.c \ No newline at end of file diff --git a/misc/strcasestr.c b/external/misc/strcasestr.c similarity index 100% rename from misc/strcasestr.c rename to external/misc/strcasestr.c diff --git a/misc/strlcat.c b/external/misc/strlcat.c similarity index 100% rename from misc/strlcat.c rename to external/misc/strlcat.c index c08c62d7..87f9c424 100644 --- a/misc/strlcat.c +++ b/external/misc/strlcat.c @@ -27,11 +27,11 @@ * *---------------------------------------------------------------*/ - +#include "direwolf.h" #include #include -#include "direwolf.h" + #include "textcolor.h" diff --git a/misc/strlcpy.c b/external/misc/strlcpy.c similarity index 99% rename from misc/strlcpy.c rename to external/misc/strlcpy.c index 64a18c1a..ff188001 100644 --- a/misc/strlcpy.c +++ b/external/misc/strlcpy.c @@ -49,10 +49,11 @@ * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ +#include "direwolf.h" #include #include -#include "direwolf.h" + #include "textcolor.h" /* diff --git a/external/misc/strsep.c b/external/misc/strsep.c new file mode 100644 index 00000000..a3338151 --- /dev/null +++ b/external/misc/strsep.c @@ -0,0 +1,72 @@ +/* $NetBSD: strsep.c,v 1.5 2014/10/31 18:59:32 spz Exp $ */ +/* from NetBSD: strsep.c,v 1.14 2003/08/07 16:43:52 agc Exp */ + +/*- + * Copyright (c) 1990, 1993 + * The Regents of the University of California. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * 3. Neither the name of the University nor the names of its contributors + * may be used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + */ + +//#include "tnftp.h" +#include + +/* + * Get next token from string *stringp, where tokens are possibly-empty + * strings separated by characters from delim. + * + * Writes NULs into the string at *stringp to end tokens. + * delim need not remain constant from call to call. + * On return, *stringp points past the last NUL written (if there might + * be further tokens), or is NULL (if there are definitely no more tokens). + * + * If *stringp is NULL, strsep returns NULL. + */ +char * +strsep(char **stringp, const char *delim) +{ + char *s; + const char *spanp; + int c, sc; + char *tok; + + if ((s = *stringp) == NULL) + return (NULL); + for (tok = s;;) { + c = *s++; + spanp = delim; + do { + if ((sc = *spanp++) == c) { + if (c == 0) + s = NULL; + else + s[-1] = 0; + *stringp = s; + return (tok); + } + } while (sc != 0); + } + /* NOTREACHED */ +} diff --git a/misc/strtok_r.c b/external/misc/strtok_r.c similarity index 100% rename from misc/strtok_r.c rename to external/misc/strtok_r.c diff --git a/external/regex/CMakeLists.txt b/external/regex/CMakeLists.txt new file mode 100644 index 00000000..76bbf9ba --- /dev/null +++ b/external/regex/CMakeLists.txt @@ -0,0 +1,24 @@ +set(REGEX_LIBRARIES "" CACHE INTERNAL "") + +if(WIN32 OR CYGWIN) # windows + + set(REGEX_LIBRARIES regex CACHE INTERNAL "regex") + + list(APPEND regex_SOURCES + # When building for Linux, we use regular expression + # functions supplied by the gnu C library. + # For the native WIN32 version, we need to use our own copy. + # These were copied from http://gnuwin32.sourceforge.net/packages/regex.htm + # Consider upgrading from https://www.gnu.org/software/libc/sources.html + ${CUSTOM_REGEX_DIR}/regex.c + ) + + add_library(regex STATIC + ${regex_SOURCES} + ) + + set_target_properties(regex + PROPERTIES COMPILE_FLAGS "-Dbool=int -Dtrue=1 -Dfalse=0 -DREGEX_STATIC" + ) + +endif() diff --git a/regex/COPYING b/external/regex/COPYING similarity index 100% rename from regex/COPYING rename to external/regex/COPYING diff --git a/regex/INSTALL b/external/regex/INSTALL similarity index 100% rename from regex/INSTALL rename to external/regex/INSTALL diff --git a/regex/LICENSES b/external/regex/LICENSES similarity index 100% rename from regex/LICENSES rename to external/regex/LICENSES diff --git a/regex/NEWS b/external/regex/NEWS similarity index 100% rename from regex/NEWS rename to external/regex/NEWS diff --git a/regex/README b/external/regex/README similarity index 100% rename from regex/README rename to external/regex/README diff --git a/regex/README-dire-wolf.txt b/external/regex/README-dire-wolf.txt similarity index 100% rename from regex/README-dire-wolf.txt rename to external/regex/README-dire-wolf.txt diff --git a/regex/re_comp.h b/external/regex/re_comp.h similarity index 100% rename from regex/re_comp.h rename to external/regex/re_comp.h diff --git a/regex/regcomp.c b/external/regex/regcomp.c similarity index 99% rename from regex/regcomp.c rename to external/regex/regcomp.c index 006fe5cf..eb6d7a4e 100644 --- a/regex/regcomp.c +++ b/external/regex/regcomp.c @@ -2510,7 +2510,7 @@ parse_dup_op (bin_tree_t *elem, re_string_t *regexp, re_dfa_t *dfa, old_tree = NULL; if (elem->token.type == SUBEXP) - postorder (elem, mark_opt_subexp, (void *) (long) elem->token.opr.idx); + postorder (elem, mark_opt_subexp, (void *) (ptrdiff_t) elem->token.opr.idx); tree = create_tree (dfa, elem, NULL, (end == -1 ? OP_DUP_ASTERISK : OP_ALT)); if (BE (tree == NULL, 0)) @@ -3725,7 +3725,7 @@ create_token_tree (re_dfa_t *dfa, bin_tree_t *left, bin_tree_t *right, static reg_errcode_t mark_opt_subexp (void *extra, bin_tree_t *node) { - int idx = (int) (long) extra; + int idx = (int) (ptrdiff_t) extra; if (node->token.type == SUBEXP && node->token.opr.idx == idx) node->token.opt_subexp = 1; diff --git a/regex/regex.c b/external/regex/regex.c similarity index 100% rename from regex/regex.c rename to external/regex/regex.c diff --git a/regex/regex.h b/external/regex/regex.h similarity index 98% rename from regex/regex.h rename to external/regex/regex.h index c2a9a4c3..a84f6a99 100644 --- a/regex/regex.h +++ b/external/regex/regex.h @@ -35,17 +35,23 @@ #if (defined __WIN32__) || (defined _WIN32) # ifdef BUILD_REGEX_DLL # define REGEX_DLL_IMPEXP __DLL_EXPORT__ +# define REGEX_VARIABLE_IMPEXP __DLL_EXPORT__ # elif defined(REGEX_STATIC) # define REGEX_DLL_IMPEXP +# define REGEX_VARIABLE_IMPEXP # elif defined (USE_REGEX_DLL) # define REGEX_DLL_IMPEXP __DLL_IMPORT__ +# define REGEX_VARIABLE_IMPEXP __DLL_IMPORT__ # elif defined (USE_REGEX_STATIC) # define REGEX_DLL_IMPEXP +# define REGEX_VARIABLE_IMPEXP extern # else /* assume USE_REGEX_DLL */ # define REGEX_DLL_IMPEXP __DLL_IMPORT__ +# define REGEX_VARIABLE_IMPEXP __DLL_IMPORT__ # endif #else /* __WIN32__ */ # define REGEX_DLL_IMPEXP +# define REGEX_VARIABLE_IMPEXP #endif /* Allow the use in C++ code. */ @@ -202,7 +208,8 @@ typedef unsigned long int reg_syntax_t; some interfaces). When a regexp is compiled, the syntax used is stored in the pattern buffer, so changing this does not affect already-compiled regexps. */ -REGEX_DLL_IMPEXP reg_syntax_t re_syntax_options; +//REGEX_VARIABLE_IMPEXP reg_syntax_t re_syntax_options; +extern reg_syntax_t re_syntax_options; /* Define combinations of the above bits for the standard possibilities. (The [[[ comments delimit what gets put into the Texinfo file, so diff --git a/regex/regex_internal.c b/external/regex/regex_internal.c similarity index 100% rename from regex/regex_internal.c rename to external/regex/regex_internal.c diff --git a/regex/regex_internal.h b/external/regex/regex_internal.h similarity index 100% rename from regex/regex_internal.h rename to external/regex/regex_internal.h diff --git a/regex/regexec.c b/external/regex/regexec.c similarity index 100% rename from regex/regexec.c rename to external/regex/regexec.c diff --git a/fsk_demod_agc.h b/fsk_demod_agc.h deleted file mode 100644 index 95c80794..00000000 --- a/fsk_demod_agc.h +++ /dev/null @@ -1,2 +0,0 @@ -#define TUNE_MS_FILTER_SIZE 140 -#define TUNE_PRE_BAUD 1.080 diff --git a/fsk_demod_state.h b/fsk_demod_state.h deleted file mode 100644 index a37b24d2..00000000 --- a/fsk_demod_state.h +++ /dev/null @@ -1,291 +0,0 @@ -/* fsk_demod_state.h */ - -#ifndef FSK_DEMOD_STATE_H - -#include "rpack.h" - -#include "audio.h" // for enum modem_t - -/* - * Demodulator state. - * The name of the file is from we only had FSK. Now we have other techniques. - * Different copy is required for each channel & subchannel being processed concurrently. - */ - -// TODO1.2: change prefix from BP_ to DSP_ - -typedef enum bp_window_e { BP_WINDOW_TRUNCATED, - BP_WINDOW_COSINE, - BP_WINDOW_HAMMING, - BP_WINDOW_BLACKMAN, - BP_WINDOW_FLATTOP } bp_window_t; - - -struct demodulator_state_s -{ -/* - * These are set once during initialization. - */ - enum modem_t modem_type; // MODEM_AFSK, MODEM_8PSK, etc. - - char profile; // 'A', 'B', etc. Upper case. - // Only needed to see if we are using 'F' to take fast path. - -#define TICKS_PER_PLL_CYCLE ( 256.0 * 256.0 * 256.0 * 256.0 ) - - int pll_step_per_sample; // PLL is advanced by this much each audio sample. - // Data is sampled when it overflows. - - - int ms_filter_size; /* Size of mark & space filters, in audio samples. */ - /* Started off as a guess of one bit length */ - /* but somewhat longer turned out to be better. */ - /* Currently using same size for any prefilter. */ - -#define MAX_FILTER_SIZE 320 /* 304 is needed for profile C, 300 baud & 44100. */ - -/* - * Filter length for Mark & Space in bit times. - * e.g. 1 means 1/1200 second for 1200 baud. - */ - float ms_filter_len_bits; - -/* - * Window type for the various filters. - */ - - bp_window_t pre_window; - bp_window_t ms_window; - bp_window_t lp_window; - - -/* - * Alternate Low pass filters. - * First is arbitrary number for quick IIR. - * Second is frequency as ratio to baud rate for FIR. - */ - int lpf_use_fir; /* 0 for IIR, 1 for FIR. */ - - float lpf_iir; /* Only if using IIR. */ - - float lpf_baud; /* Cutoff frequency as fraction of baud. */ - /* Intuitively we'd expect this to be somewhere */ - /* in the range of 0.5 to 1. */ - /* In practice, it turned out a little larger */ - /* for profiles B, C, D. */ - - float lp_filter_len_bits; /* Length in number of bit times. */ - - int lp_filter_size; /* Size of Low Pass filter, in audio samples. */ - /* Previously it was always the same as the M/S */ - /* filters but in version 1.2 it's now independent. */ - -/* - * Automatic gain control. Fast attack and slow decay factors. - */ - float agc_fast_attack; - float agc_slow_decay; - -/* - * Use a longer term view for reporting signal levels. - */ - float quick_attack; - float sluggish_decay; - -/* - * Hysteresis before final demodulator 0 / 1 decision. - */ - float hysteresis; - int num_slicers; /* >1 for multiple slicers. */ - -/* - * Phase Locked Loop (PLL) inertia. - * Larger number means less influence by signal transitions. - */ - float pll_locked_inertia; - float pll_searching_inertia; - - -/* - * Optional band pass pre-filter before mark/space detector. - */ - int use_prefilter; /* True to enable it. */ - - float prefilter_baud; /* Cutoff frequencies, as fraction of */ - /* baud rate, beyond tones used. */ - /* Example, if we used 1600/1800 tones at */ - /* 300 baud, and this was 0.5, the cutoff */ - /* frequencies would be: */ - /* lower = min(1600,1800) - 0.5 * 300 = 1450 */ - /* upper = max(1600,1800) + 0.5 * 300 = 1950 */ - - float pre_filter_len_bits; /* Length in number of bit times. */ - - int pre_filter_size; /* Size of pre filter, in audio samples. */ - - float pre_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); - -/* - * Kernel for the mark and space detection filters. - */ - - float m_sin_table[MAX_FILTER_SIZE] __attribute__((aligned(16))); - float m_cos_table[MAX_FILTER_SIZE] __attribute__((aligned(16))); - - float s_sin_table[MAX_FILTER_SIZE] __attribute__((aligned(16))); - float s_cos_table[MAX_FILTER_SIZE] __attribute__((aligned(16))); - -/* - * These are for PSK only. - * They are number of delay line taps into previous symbol. - * They are one symbol period and + or - 45 degrees of the carrier frequency. - */ - int boffs; /* symbol length based on sample rate and baud. */ - int coffs; /* to get cos component of previous symbol. */ - int soffs; /* to get sin component of previous symbol. */ - - unsigned int lo_step; /* How much to advance the local oscillator */ - /* phase for each audio sample. */ - - int psk_use_lo; /* Use local oscillator rather than self correlation. */ - - -/* - * The rest are continuously updated. - */ - - unsigned int lo_phase; /* Local oscillator for PSK. */ - - -/* - * Most recent raw audio samples, before/after prefiltering. - */ - float raw_cb[MAX_FILTER_SIZE] __attribute__((aligned(16))); - -/* - * Use half of the AGC code to get a measure of input audio amplitude. - * These use "quick" attack and "sluggish" decay while the - * AGC uses "fast" attack and "slow" decay. - */ - - float alevel_rec_peak; - float alevel_rec_valley; - float alevel_mark_peak; - float alevel_space_peak; - -/* - * Input to the mark/space detector. - * Could be prefiltered or raw audio. - */ - float ms_in_cb[MAX_FILTER_SIZE] __attribute__((aligned(16))); - -/* - * Outputs from the mark and space amplitude detection, - * used as inputs to the FIR lowpass filters. - * Kernel for the lowpass filters. - */ - - float m_amp_cb[MAX_FILTER_SIZE] __attribute__((aligned(16))); - float s_amp_cb[MAX_FILTER_SIZE] __attribute__((aligned(16))); - - float lp_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); - - - float m_peak, s_peak; - float m_valley, s_valley; - float m_amp_prev, s_amp_prev; - -/* - * For the PLL and data bit timing. - * starting in version 1.2 we can have multiple slicers for one demodulator. - * Each slicer has its own PLL and HDLC decoder. - */ - -/* - * Version 1.3: Clean up subchan vs. slicer. - * - * Originally some number of CHANNELS (originally 2, later 6) - * which can have multiple parallel demodulators called SUB-CHANNELS. - * This was originally for staggered frequencies for HF SSB. - * It can also be used for multiple demodulators with the same - * frequency but other differing parameters. - * Each subchannel has its own demodulator and HDLC decoder. - * - * In version 1.2 we added multiple SLICERS. - * The data structure, here, has multiple slicers per - * demodulator (subchannel). Due to fuzzy thinking or - * expediency, the multiple slicers got mapped into subchannels. - * This means we can't use both multiple decoders and - * multiple slicers at the same time. - * - * Clean this up in 1.3 and keep the concepts separate. - * This means adding a third variable many places - * we are passing around the origin. - * - */ - struct { - - signed int data_clock_pll; // PLL for data clock recovery. - // It is incremented by pll_step_per_sample - // for each audio sample. - - signed int prev_d_c_pll; // Previous value of above, before - // incrementing, to detect overflows. - - int prev_demod_data; // Previous data bit detected. - // Used to look for transitions. - float prev_demod_out_f; - - /* This is used only for "9600" baud data. */ - - int lfsr; // Descrambler shift register. - - } slicer [MAX_SLICERS]; // Actual number in use is num_slicers. - // Should be in range 1 .. MAX_SLICERS, - -/* - * Special for Rino decoder only. - * One for each possible signal polarity. - * The project showed promise but fell by the wayside. - */ - -#if 0 - - struct gr_state_s { - - signed int data_clock_pll; // PLL for data clock recovery. - // It is incremented by pll_step_per_sample - // for each audio sample. - - signed int prev_d_c_pll; // Previous value of above, before - // incrementing, to detect overflows. - - float gr_minus_peak; // For automatic gain control. - float gr_plus_peak; - - int gr_sync; // Is sync pulse present? - int gr_prev_sync; // Previous state to detect leading edge. - - int gr_first_sample; // Index of starting sample index for debugging. - - int gr_dcd; // Data carrier detect. i.e. are we - // currently decoding a message. - - float gr_early_sum; // For averaging bit values in two regions. - int gr_early_count; - float gr_late_sum; - int gr_late_count; - float gr_sync_sum; - int gr_sync_count; - - int gr_bit_count; // Bit index into message. - - struct rpack_s rpack; // Collection of bits. - - } gr_state[2]; -#endif - -}; - -#define FSK_DEMOD_STATE_H 1 -#endif \ No newline at end of file diff --git a/fx25.png b/fx25.png new file mode 100644 index 00000000..33d74a92 Binary files /dev/null and b/fx25.png differ diff --git a/generic.conf b/generic.conf deleted file mode 100644 index 223007f4..00000000 --- a/generic.conf +++ /dev/null @@ -1,573 +0,0 @@ -C############################################################# -C# # -C# Configuration file for Dire Wolf # -C# # -L# Linux version # -W# Windows version # -M# Macintosh version # -C# # -C############################################################# -R -R -R The sample config file was getting pretty messy -R with the Windows and Linux differences. -R It would be a maintenance burden to keep most of -R two different versions in sync. -R This common source is now used to generate the -R two different variations while having only a single -R copy of the common parts. -R -R The first column contains one of the following: -R -R R remark which is discarded. -R C common to both versions. -R W Windows version only. -R L Linux version only. -R M Macintosh version and possibly others (portaudio used). -R -C# -C# Consult the User Guide for more details on configuration options. -C# -C# -C# These are the most likely settings you might change: -C# -C# (1) MYCALL - call sign and SSID for your station. -C# -C# Look for lines starting with MYCALL and -C# change NOCALL to your own. -C# -C# (2) PBEACON - enable position beaconing. -C# -C# Look for lines starting with PBEACON and -C# modify for your call, location, etc. -C# -C# (3) DIGIPEATER - configure digipeating rules. -C# -C# Look for lines starting with DIGIPEATER. -C# Most people will probably use the given example. -C# Just remove the "#" from the start of the line -C# to enable it. -C# -C# (4) IGSERVER, IGLOGIN - IGate server and login -C# -C# Configure an IGate client to relay messages between -C# radio and internet servers. -C# -C# -C# The default location is "direwolf.conf" in the current working directory. -L# On Linux, the user's home directory will also be searched. -C# An alternate configuration file location can be specified with the "-c" command line option. -C# -C# As you probably guessed by now, # indicates a comment line. -C# -C# Remove the # at the beginning of a line if you want to use a sample -C# configuration that is currently commented out. -C# -C# Commands are a keyword followed by parameters. -C# -C# Command key words are case insensitive. i.e. upper and lower case are equivalent. -C# -C# Command parameters are generally case sensitive. i.e. upper and lower case are different. -C# -C -C -C############################################################# -C# # -C# FIRST AUDIO DEVICE PROPERTIES # -C# (Channel 0 + 1 if in stereo) # -C# # -C############################################################# -C -C# -C# Many people will simply use the default sound device. -C# Some might want to use an alternative device by chosing it here. -C# -W# When the Windows version starts up, it displays something like -W# this with the available sound devices and capabilities: -W# -W# Available audio input devices for receive (*=selected): -W# * 0: Microphone (C-Media USB Headpho (channel 2) -W# 1: Microphone (Bluetooth SCO Audio -W# 2: Microphone (Bluetooth AV Audio) -W# * 3: Microphone (Realtek High Defini (channels 0 & 1) -W# Available audio output devices for transmit (*=selected): -W# * 0: Speakers (C-Media USB Headphone (channel 2) -W# 1: Speakers (Bluetooth SCO Audio) -W# 2: Realtek Digital Output(Optical) -W# 3: Speakers (Bluetooth AV Audio) -W# * 4: Speakers (Realtek High Definiti (channels 0 & 1) -W# 5: Realtek Digital Output (Realtek -W# -W# Example: To use the microphone and speaker connections on the -W# system board, either of these forms can be used: -W -W#ADEVICE High -W#ADEVICE 3 4 -W -W -W# Example: To use the USB Audio, use a command like this with -W# the input and output device numbers. (Remove the # comment character.) -W#ADEVICE USB -W -W# The position in the list can change when devices (e.g. USB) are added and removed. -W# You can also specify devices by using part of the name. -W# Here is an example of specifying the USB Audio device. -W# This is case-sensitive. Upper and lower case are not treated the same. -W -W#ADEVICE USB -W -W -L# Linux ALSA is complicated. See User Guide for discussion. -L# To use something other than the default, generally use plughw -L# and a card number reported by "arecord -l" command. Example: -L -L# ADEVICE plughw:1,0 -L -L# Starting with version 1.0, you can also use "-" or "stdin" to -L# pipe stdout from some other application such as a software defined -L# radio. You can also specify "UDP:" and an optional port for input. -L# Something different must be specified for output. -L -M# Macintosh Operating System uses portaudio driver for audio -M# input/output. Default device selection not available. User/OP -M# must configure the sound input/output option. Note that -M# the device names can contain spaces. In this case, the names -M# must be enclosed by quotes. -M# -M# Examples: -M# -M# ADEVICE "USB Audio Codec:6" "USB Audio Codec:5" -M# -M# -W# ADEVICE - 0 -W# ADEVICE UDP:7355 0 -L# ADEVICE - plughw:1,0 -L# ADEVICE UDP:7355 default -M# ADEVICE UDP:7355 default -M# -L -L -C -C# -C# Number of audio channels for this souncard: 1 or 2. -C# -C -CACHANNELS 1 -C#ACHANNELS 2 -C -C -C############################################################# -C# # -C# SECOND AUDIO DEVICE PROPERTIES # -C# (Channel 2 + 3 if in stereo) # -C# # -C############################################################# -C -C#ADEVICE1 ... -C -C -C############################################################# -C# # -C# THIRD AUDIO DEVICE PROPERTIES # -C# (Channel 4 + 5 if in stereo) # -C# # -C############################################################# -C -C#ADEVICE2 ... -C -C -C############################################################# -C# # -C# CHANNEL 0 PROPERTIES # -C# # -C############################################################# -C -CCHANNEL 0 -C -C# -C# The following MYCALL, MODEM, PTT, etc. configuration items -C# apply to the most recent CHANNEL. -C# -C -C# -C# Station identifier for this channel. -C# Multiple channels can have the same or different names. -C# -C# It can be up to 6 letters and digits with an optional ssid. -C# The APRS specification requires that it be upper case. -C# -C# Example (don't use this unless you are me): MYCALL WB2OSZ-5 -C# -C -CMYCALL N0CALL -C -C# -C# Pick a suitable modem speed based on your situation. -C# 1200 Most common for VHF/UHF. Default if not specified. -C# 300 Low speed for HF SSB. -C# 9600 High speed - Can't use Microphone and Speaker connections. -C# -C# In the simplest form, just specify the speed. -C# -C -CMODEM 1200 -C#MODEM 300 -C#MODEM 9600 -C -C# -C# These are the defaults should be fine for most cases. In special situations, -C# you might want to specify different AFSK tones or the baseband mode which does -C# not use AFSK. -C# -C#MODEM 1200 1200:2200 -C#MODEM 300 1600:1800 -C#MODEM 9600 0:0 -C# -C# -C# On HF SSB, you might want to use multiple demodulators on slightly different -C# frequencies to compensate for stations off frequency. Here we have 7 different -C# demodulators at 30 Hz intervals. This takes a lot of CPU power so you will -C# probably need to reduce the audio sampling rate with the /n option. -C -C#MODEM 300 1600:1800 7@30 /4 -C -C -C# -C# Uncomment line below to enable the DTMF decoder for this channel. -C# -C -C#DTMF -C -C# -C# If not using a VOX circuit, the transmitter Push to Talk (PTT) -C# control is usually wired to a serial port with a suitable interface circuit. -C# DON'T connect it directly! -C# -C# For the PTT command, specify the device and either RTS or DTR. -C# RTS or DTR may be preceded by "-" to invert the signal. -C# Both can be used for interfaces that want them driven with opposite polarity. -C# -L# COM1 can be used instead of /dev/ttyS0, COM2 for /dev/ttyS1, and so on. -L# -C -C#PTT COM1 RTS -C#PTT COM1 RTS -DTR -L#PTT /dev/ttyUSB0 RTS -C -L# -L# On Linux, you can also use general purpose I/O pins if -L# your system is configured for user access to them. -L# This would apply mostly to microprocessor boards, not a regular PC. -L# See separate Raspberry Pi document for more details. -L# The number may be preceded by "-" to invert the signal. -L# -L -L#PTT GPIO 25 -L -C# The Data Carrier Detect (DCD) signal can be sent to the same places -C# as the PTT signal. This could be used to light up an LED like a normal TNC. -C -C#DCD COM1 -DTR -L#DCD GPIO 24 -C -C -C############################################################# -C# # -C# CHANNEL 1 PROPERTIES # -C# # -C############################################################# -C -C#CHANNEL 1 -C -C# -C# Specify MYCALL, MODEM, PTT, etc. configuration items for -C# CHANNEL 1. Repeat for any other channels. -C -C -C############################################################# -C# # -C# TEXT TO SPEECH COMMAND FILE # -C# # -C############################################################# -C -W#SPEECH dwespeak.bat -L#SPEECH dwespeak.sh -C -C -C############################################################# -C# # -C# VIRTUAL TNC SERVER PROPERTIES # -C# # -C############################################################# -C -C# -C# Dire Wolf acts as a virtual TNC and can communicate with -C# client applications by different protocols: -C# -C# - the "AGW TCPIP Socket Interface" - default port 8000 -C# - KISS protocol over TCP socket - default port 8001 -W# - KISS TNC via serial port -L# - KISS TNC via pseudo terminal (-p command line option) -C# -C -CAGWPORT 8000 -CKISSPORT 8001 -C -W# -W# Some applications are designed to operate with only a physical -W# TNC attached to a serial port. For these, we provide a virtual serial -W# port that appears to be connected to a TNC. -W# -W# Take a look at the User Guide for instructions to set up -W# two virtual serial ports named COM3 and COM4 connected by -W# a null modem. -W# -W# Using the configuration described, Dire Wolf will connect to -W# COM3 and the client application will use COM4. -W# -W# Uncomment following line to use this feature. -W -W#NULLMODEM COM3 -W -W -C# -C# It is sometimes possible to recover frames with a bad FCS. -C# This applies to all channels. -C# -C# 0 [NONE] - Don't try to repair. -C# 1 [SINGLE] - Attempt to fix single bit error. (default) -C# 2 [DOUBLE] - Also attempt to fix two adjacent bits. -C# ... see User Guide for more values and in-depth discussion. -C# -C -C#FIX_BITS 0 -C -C# -C############################################################# -C# # -C# BEACONING PROPERTIES # -C# # -C############################################################# -C -C -C# -C# Beaconing is configured with these two commands: -C# -C# PBEACON - for a position report (usually yourself) -C# OBEACON - for an object report (usually some other entity) -C# -C# Each has a series of keywords and values for options. -C# See User Guide for details. -C# -C# Example: -C# -C# This results in a broadcast once every 10 minutes. -C# Every half hour, it can travel via two digipeater hops. -C# The others are kept local. -C# -C -C#PBEACON delay=1 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" via=WIDE1-1,WIDE2-1 -C#PBEACON delay=11 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" -C#PBEACON delay=21 every=30 overlay=S symbol="digi" lat=42^37.14N long=071^20.83W power=50 height=20 gain=4 comment="Chelmsford MA" -C -C -C# With UTM coordinates instead of latitude and longitude. -C -C#PBEACON delay=1 every=10 overlay=S symbol="digi" zone=19T easting=307477 northing=4720178 -C -C -C# -C# When the destination field is set to "SPEECH" the information part is -C# converted to speech rather than transmitted as a data frame. -C# -C -C#CBEACON dest="SPEECH" info="Club meeting tonight at 7 pm." -C -C# Similar for Morse code. If SSID is specified, it is multiplied -C# by 2 to get speed in words per minute (WPM). -C -C#CBEACON dest="MORSE-6" info="de MYCALL" -C -C -C# -C# Modify for your particular situation before removing -C# the # comment character from the beginning of appropriate lines above. -C# -C -C -C############################################################# -C# # -C# DIGIPEATER PROPERTIES # -C# # -C############################################################# -C -C# -C# For most common situations, use something like this by removing -C# the "#" from the beginning of the line below. -C# -C -C#DIGIPEAT 0 0 ^WIDE[3-7]-[1-7]$|^TEST$ ^WIDE[12]-[12]$ TRACE -C -C# See User Guide for more explanation of what this means and how -C# it can be customized for your particular needs. -C -C# Filtering can be used to limit was is digipeated. -C# For example, only weather weather reports, received on channel 0, -C# will be retransmitted on channel 1. -C# -C -C#FILTER 0 1 t/wn -C -C -C############################################################# -C# # -C# INTERNET GATEWAY # -C# # -C############################################################# -C -C# First you need to specify the name of a Tier 2 server. -C# The current preferred way is to use one of these regional rotate addresses: -C -C# noam.aprs2.net - for North America -C# soam.aprs2.net - for South America -C# euro.aprs2.net - for Europe and Africa -C# asia.aprs2.net - for Asia -C# aunz.aprs2.net - for Oceania -C -C#IGSERVER noam.aprs2.net -C -C# You also need to specify your login name and passcode. -C# Contact the author if you can't figure out how to generate the passcode. -C -C#IGLOGIN WB2OSZ-5 123456 -C -C# That's all you need for a receive only IGate which relays -C# messages from the local radio channel to the global servers. -C -C# Some might want to send an IGate client position directly to a server -C# without sending it over the air and relying on someone else to -C# forward it to an IGate server. This is done by using sendto=IG rather -C# than a radio channel number. Overlay R for receive only, T for two way. -C -C#PBEACON sendto=IG delay=0:30 every=60:00 symbol="igate" overlay=R lat=42^37.14N long=071^20.83W -C#PBEACON sendto=IG delay=0:30 every=60:00 symbol="igate" overlay=T lat=42^37.14N long=071^20.83W -C -C -C# To relay messages from the Internet to radio, you need to add -C# one more option with the transmit channel number and a VIA path. -C -C#IGTXVIA 0 WIDE1-1 -C -C# You might want to apply a filter for what packets will be obtained from the server. -C# Read about filters here: http://www.aprs-is.net/javaprsfilter.aspx -C# Example, positions and objects within 50 km of my location: -C -C#IGFILTER m/50 -C -C# That is known as a server-side filter. It is processed by the IGate server. -C# You can also apply local filtering to limit what will be transmitted on the -C# RF side. For example, transmit only "messages" on channel 0 and weather -C# reports on channel 1. -C -C#FILTER IG 0 t/m -C#FILTER IG 1 t/wn -C -C# Finally, we don't want to flood the radio channel. -C# The IGate function will limit the number of packets transmitted -C# during 1 minute and 5 minute intervals. If a limit would -C# be exceeded, the packet is dropped and message is displayed in red. -C -CIGTXLIMIT 6 10 -C -C -C############################################################# -C# # -C# APRStt GATEWAY # -C# # -C############################################################# -C -C# -C# Dire Wolf can receive DTMF (commonly known as Touch Tone) -C# messages and convert them to packet objects. -C# -C# See separate "APRStt-Implementation-Notes" document for details. -C# -C -C# -C# Sample gateway configuration based on: -C# -C# http://www.aprs.org/aprstt/aprstt-coding24.txt -C# http://www.aprs.org/aprs-jamboree-2013.html -C# -C -C# Define specific points. -C -CTTPOINT B01 37^55.37N 81^7.86W -CTTPOINT B7495088 42.605237 -71.34456 -CTTPOINT B934 42.605237 -71.34456 -C -CTTPOINT B901 42.661279 -71.364452 -CTTPOINT B902 42.660411 -71.364419 -CTTPOINT B903 42.659046 -71.364452 -CTTPOINT B904 42.657578 -71.364602 -C -C -C# For location at given bearing and distance from starting point. -C -CTTVECTOR B5bbbddd 37^55.37N 81^7.86W 0.01 mi -C -C# For location specified by x, y coordinates. -C -CTTGRID Byyyxxx 37^50.00N 81^00.00W 37^59.99N 81^09.99W -C -C# UTM location for Lowell-Dracut-Tyngsborough State Forest. -C -CTTUTM B6xxxyyy 19T 10 300000 4720000 -C -C -C -C# Location for the corral. -C -CTTCORRAL 37^55.50N 81^7.00W 0^0.02N -C -C# Compact messages - Fixed locations xx and object yyy where -C# Object numbers 100 - 199 = bicycle -C# Object numbers 200 - 299 = fire truck -C# Others = dog -C -CTTMACRO xx1yy B9xx*AB166*AA2B4C5B3B0A1yy -CTTMACRO xx2yy B9xx*AB170*AA3C4C7C3B0A2yy -CTTMACRO xxyyy B9xx*AB180*AA3A6C4A0Ayyy -C -CTTMACRO z Cz -C -C# Receive on channel 0, Transmit object reports on channel 1 with optional via path. -C# You probably want to put in a transmit delay on the APRStt channel so it -C# it doesn't start sending a response before the user releases PTT. -C# This is in 10 ms units so 100 means 1000 ms = 1 second. -C -C#TTOBJ 0 1 WIDE1-1 -C#CHANNEL 0 -C#DWAIT 100 -C -C# Advertise gateway position with beacon. -C -C# OBEACON DELAY=0:15 EVERY=10:00 VIA=WIDE1-1 OBJNAME=WB2OSZ-tt SYMBOL=APRStt LAT=42^37.14N LONG=71^20.83W COMMENT="APRStt Gateway" -C -C -C# Sample speech responses. -C# Default is Morse code "R" for received OK and "?" for all errors. -C -C#TTERR OK SPEECH Message Received. -C#TTERR D_MSG SPEECH D not implemented. -C#TTERR INTERNAL SPEECH Internal error. -C#TTERR MACRO_NOMATCH SPEECH No definition for digit sequence. -C#TTERR BAD_CHECKSUM SPEECH Bad checksum on call. -C#TTERR INVALID_CALL SPEECH Invalid callsign. -C#TTERR INVALID_OBJNAME SPEECH Invalid object name. -C#TTERR INVALID_SYMBOL SPEECH Invalid symbol. -C#TTERR INVALID_LOC SPEECH Invalid location. -C#TTERR NO_CALL SPEECH No call or object name. -C#TTERR SATSQ SPEECH Satellite square must be 4 digits. -C#TTERR SUFFIX_NO_CALL SPEECH Send full call before using suffix. -C \ No newline at end of file diff --git a/hdlc_send.c b/hdlc_send.c deleted file mode 100644 index 4a09cc82..00000000 --- a/hdlc_send.c +++ /dev/null @@ -1,231 +0,0 @@ - -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2011, 2013, 2014 John Langner, WB2OSZ -// -// 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, see . -// - -#include "direwolf.h" - -#include - -#include "hdlc_send.h" -#include "audio.h" -#include "gen_tone.h" -#include "textcolor.h" -#include "fcs_calc.h" - -static void send_control (int, int); -static void send_data (int, int); -static void send_bit (int, int); - - - -static int number_of_bits_sent[MAX_CHANS]; // Count number of bits sent by "hdlc_send_frame" or "hdlc_send_flags" - - - - - -/*------------------------------------------------------------- - * - * Name: hdlc_send - * - * Purpose: Convert HDLC frames to a stream of bits. - * - * Inputs: chan - Audio channel number, 0 = first. - * - * fbuf - Frame buffer address. - * - * flen - Frame length, not including the FCS. - * - * bad_fcs - Append an invalid FCS for testing purposes. - * - * Outputs: Bits are shipped out by calling tone_gen_put_bit(). - * - * Returns: Number of bits sent including "flags" and the - * stuffing bits. - * The required time can be calculated by dividing this - * number by the transmit rate of bits/sec. - * - * Description: Convert to stream of bits including: - * start flag - * bit stuffed data - * calculated FCS - * end flag - * NRZI encoding - * - * - * Assumptions: It is assumed that the tone_gen module has been - * properly initialized so that bits sent with - * tone_gen_put_bit() are processed correctly. - * - *--------------------------------------------------------------*/ - -int hdlc_send_frame (int chan, unsigned char *fbuf, int flen, int bad_fcs) -{ - int j, fcs; - - - number_of_bits_sent[chan] = 0; - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("hdlc_send_frame ( chan = %d, fbuf = %p, flen = %d, bad_fcs = %d)\n", chan, fbuf, flen, bad_fcs); - fflush (stdout); -#endif - - - send_control (chan, 0x7e); /* Start frame */ - - for (j=0; j> 8) & 0xff); - } - else { - send_data (chan, fcs & 0xff); - send_data (chan, (fcs >> 8) & 0xff); - } - - send_control (chan, 0x7e); /* End frame */ - - return (number_of_bits_sent[chan]); -} - - -/*------------------------------------------------------------- - * - * Name: hdlc_send_flags - * - * Purpose: Send HDLC flags before and after the frame. - * - * Inputs: chan - Audio channel number, 0 = first. - * - * nflags - Number of flag patterns to send. - * - * finish - True for end of transmission. - * This causes the last audio buffer to be flushed. - * - * Outputs: Bits are shipped out by calling tone_gen_put_bit(). - * - * Returns: Number of bits sent. - * There is no bit-stuffing so we would expect this to - * be 8 * nflags. - * The required time can be calculated by dividing this - * number by the transmit rate of bits/sec. - * - * Assumptions: It is assumed that the tone_gen module has been - * properly initialized so that bits sent with - * tone_gen_put_bit() are processed correctly. - * - *--------------------------------------------------------------*/ - -int hdlc_send_flags (int chan, int nflags, int finish) -{ - int j; - - - number_of_bits_sent[chan] = 0; - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("hdlc_send_flags ( chan = %d, nflags = %d, finish = %d )\n", chan, nflags, finish); - fflush (stdout); -#endif - - /* The AX.25 spec states that when the transmitter is on but not sending data */ - /* it should send a continuous stream of "flags." */ - - for (j=0; j>= 1; - } - - stuff[chan] = 0; -} - -static void send_data (int chan, int x) -{ - int i; - - for (i=0; i<8; i++) { - send_bit (chan, x & 1); - if (x & 1) { - stuff[chan]++; - if (stuff[chan] == 5) { - send_bit (chan, 0); - stuff[chan] = 0; - } - } else { - stuff[chan] = 0; - } - x >>= 1; - } -} - -/* - * NRZI encoding. - * data 1 bit -> no change. - * data 0 bit -> invert signal. - */ - -static void send_bit (int chan, int b) -{ - static int output[MAX_CHANS]; - - if (b == 0) { - output[chan] = ! output[chan]; - } - - tone_gen_put_bit (chan, output[chan]); - - number_of_bits_sent[chan]++; -} - -/* end hdlc_send.c */ \ No newline at end of file diff --git a/hdlc_send.h b/hdlc_send.h deleted file mode 100644 index 10d200c1..00000000 --- a/hdlc_send.h +++ /dev/null @@ -1,10 +0,0 @@ - -/* hdlc_send.h */ - -int hdlc_send_frame (int chan, unsigned char *fbuf, int flen, int bad_fcs); - -int hdlc_send_flags (int chan, int flags, int finish); - -/* end hdlc_send.h */ - - diff --git a/kiss.c b/kiss.c deleted file mode 100644 index 19c258d5..00000000 --- a/kiss.c +++ /dev/null @@ -1,1013 +0,0 @@ -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2011, 2013, 2014, 2016 John Langner, WB2OSZ -// -// 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, see . -// - - - -/*------------------------------------------------------------------ - * - * Module: kiss.c - * - * Purpose: Act as a virtual KISS TNC for use by other packet radio applications. - * On Windows, it is a serial port. On Linux, a pseudo terminal. - * - * Input: - * - * Outputs: - * - * Description: It implements the KISS TNC protocol as described in: - * http://www.ka9q.net/papers/kiss.html - * - * Briefly, a frame is composed of - * - * * FEND (0xC0) - * * Contents - with special escape sequences so a 0xc0 - * byte in the data is not taken as end of frame. - * as part of the data. - * * FEND - * - * The first byte of the frame contains: - * - * * port number in upper nybble. - * * command in lower nybble. - * - * - * Commands from application recognized: - * - * 0 Data Frame AX.25 frame in raw format. - * - * 1 TXDELAY See explanation in xmit.c. - * - * 2 Persistence " " - * - * 3 SlotTime " " - * - * 4 TXtail " " - * Spec says it is obsolete but Xastir - * sends it and we respect it. - * - * 5 FullDuplex Ignored. Always full duplex. - * - * 6 SetHardware TNC specific. Ignored. - * - * FF Return Exit KISS mode. Ignored. - * - * - * Messages sent to client application: - * - * 0 Data Frame Received AX.25 frame in raw format. - * - * - * - * Platform differences: - * - * We can use a pseudo terminal for Linux or Cygwin applications. - * However, Microsoft Windows doesn't seem to have similar functionality. - * Native Windows applications expect to see a device named COM1, - * COM2, COM3, or COM4. Some might offer more flexibility but others - * might be limited to these four choices. - * - * The documentation instucts the user to install the com0com - * "Null-modem emulator" from http://sourceforge.net/projects/com0com/ - * and configure it for COM3 & COM4. - * - * By default Dire Wolf will use COM3 (/dev/ttyS2 or /dev/com3 - lower case!) - * and the client application will use COM4 (available as /dev/ttyS or - * /dev/com4 for Cygwin applications). - * - * - * This can get confusing. - * - * If __WIN32__ is defined, - * We use the Windows interface to the specfied serial port. - * This could be a real serial port or the nullmodem driver - * connected to another application. - * - * If __CYGWIN__ is defined, - * We connect to a serial port as in the previous case but - * use the Linux I/O interface. - * We also supply a pseudo terminal for any Cygwin applications - * such as Xastir so the null modem is not needed. - * - * For the Linux case, - * We supply a pseudo terminal for use by other applications. - * - * - * Reference: http://www.robbayer.com/files/serial-win.pdf - * - *---------------------------------------------------------------*/ - -#include "direwolf.h" - -#include -#include - -#if __WIN32__ -#include -#else -#include -#include -#include -#include -#include -#include -#include -#ifdef __OpenBSD__ -#include -#else -#include -#endif -#endif - -#include -#include - - -#include "tq.h" -#include "ax25_pad.h" -#include "textcolor.h" -#include "kiss.h" -#include "kiss_frame.h" -#include "xmit.h" - - -#if __WIN32__ -typedef HANDLE MYFDTYPE; -#define MYFDERROR INVALID_HANDLE_VALUE -#else -typedef int MYFDTYPE; -#define MYFDERROR (-1) -#endif - - -static kiss_frame_t kf; /* Accumulated KISS frame and state of decoder. */ - - -/* - * These are for a Linux/Cygwin pseudo terminal. - */ - -#if ! __WIN32__ - -static MYFDTYPE pt_master_fd = MYFDERROR; /* File descriptor for my end. */ - -static char pt_slave_name[32]; /* Pseudo terminal slave name */ - /* like /dev/pts/999 */ - - - -/* - * Symlink to pseudo terminal name which changes. - */ - -#define TMP_KISSTNC_SYMLINK "/tmp/kisstnc" - -#endif - -/* - * This is for native Windows applications and a virtual null modem. - */ - -#if __CYGWIN__ || __WIN32__ - -static MYFDTYPE nullmodem_fd = MYFDERROR; - -#endif - - -// TODO: define in one place, use everywhere. -#if __WIN32__ -#define THREAD_F unsigned __stdcall -#else -#define THREAD_F void * -#endif - -static THREAD_F kiss_listen_thread (void *arg); - - - -#if DEBUG9 -static FILE *log_fp; -#endif - - -static int kiss_debug = 0; /* Print information flowing from and to client. */ - -void kiss_serial_set_debug (int n) -{ - kiss_debug = n; -} - - -/* In server.c. Should probably move to some misc. function file. */ - -void hex_dump (unsigned char *p, int len); - - - - - -/*------------------------------------------------------------------- - * - * Name: kiss_init - * - * Purpose: Set up a pseudo terminal acting as a virtual KISS TNC. - * - * - * Inputs: mc->nullmodem - name of device for our end of nullmodem. - * - * Outputs: - * - * Description: (1) Create a pseudo terminal for the client to use. - * (2) Start a new thread to listen for commands from client app - * so the main application doesn't block while we wait. - * - * - *--------------------------------------------------------------------*/ - -#if __WIN32__ -static MYFDTYPE kiss_open_nullmodem (char *device); -#else -static MYFDTYPE kiss_open_pt (void); -#endif - - -void kiss_init (struct misc_config_s *mc) -{ - -#if __WIN32__ - HANDLE kiss_nullmodem_listen_th; -#else - pthread_t kiss_pterm_listen_tid; - //pthread_t kiss_nullmodem_listen_tid; - int e; -#endif - - memset (&kf, 0, sizeof(kf)); - -/* - * This reads messages from client. - */ - -#if ! __WIN32__ - -/* - * Pseudo terminal for Cygwin and Linux versions. - */ - pt_master_fd = MYFDERROR; - - if (mc->enable_kiss_pt) { - - pt_master_fd = kiss_open_pt (); - - if (pt_master_fd != MYFDERROR) { - e = pthread_create (&kiss_pterm_listen_tid, (pthread_attr_t*)NULL, kiss_listen_thread, NULL); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - perror("Could not create kiss listening thread for Linux pseudo terminal"); - } - } - } - else { - text_color_set(DW_COLOR_INFO); - dw_printf ("Use -p command line option to enable KISS pseudo terminal.\n"); - } -#endif - -#if __CYGWIN__ || __WIN32 - -/* - * Cygwin and native Windows versions have serial port connection. - */ - if (strlen(mc->nullmodem) > 0) { - -#if ! __WIN32__ - - /* Translate Windows device name into Linux name. */ - /* COM1 -> /dev/ttyS0, etc. */ - - if (strncasecmp(mc->nullmodem, "COM", 3) == 0) { - int n = atoi (mc->nullmodem + 3); - text_color_set(DW_COLOR_INFO); - dw_printf ("Converted nullmodem device '%s'", mc->nullmodem); - if (n < 1) n = 1; - snprintf (mc->nullmodem, sizeof(mc->nullmodem), "/dev/ttyS%d", n-1); - dw_printf (" to Linux equivalent '%s'\n", mc->nullmodem); - } -#endif - nullmodem_fd = kiss_open_nullmodem (mc->nullmodem); - - if (nullmodem_fd != MYFDERROR) { -#if __WIN32__ - kiss_nullmodem_listen_th = (HANDLE)_beginthreadex (NULL, 0, kiss_listen_thread, NULL, 0, NULL); - if (kiss_nullmodem_listen_th == NULL) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Could not create kiss nullmodem thread\n"); - return; - } -#else - e = pthread_create (&kiss_nullmodem_listen_tid, NULL, kiss_listen_thread, NULL); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - perror("Could not create kiss listening thread for Windows virtual COM port."); - - } -#endif - } - } -#endif - - -#if DEBUG - text_color_set (DW_COLOR_DEBUG); -#if ! __WIN32__ - dw_printf ("end of kiss_init: pt_master_fd = %d\n", pt_master_fd); -#endif -#if __CYGWIN__ || __WIN32__ - dw_printf ("end of kiss_init: nullmodem_fd = %d\n", nullmodem_fd); -#endif - -#endif -} - - -/* - * Returns fd for master side of pseudo terminal or MYFDERROR for error. - */ - -#if ! __WIN32__ - -static MYFDTYPE kiss_open_pt (void) -{ - int fd; - char *pts; - struct termios ts; - int e; - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kiss_open_pt ( )\n"); -#endif - - - fd = posix_openpt(O_RDWR|O_NOCTTY); - - if (fd == MYFDERROR - || grantpt (fd) == MYFDERROR - || unlockpt (fd) == MYFDERROR - || (pts = ptsname (fd)) == NULL) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("ERROR - Could not create pseudo terminal for KISS TNC.\n"); - return (MYFDERROR); - } - - strlcpy (pt_slave_name, pts, sizeof(pt_slave_name)); - - e = tcgetattr (fd, &ts); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Can't get pseudo terminal attributes, err=%d\n", e); - perror ("pt tcgetattr"); - } - - cfmakeraw (&ts); - - ts.c_cc[VMIN] = 1; /* wait for at least one character */ - ts.c_cc[VTIME] = 0; /* no fancy timing. */ - - - e = tcsetattr (fd, TCSANOW, &ts); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Can't set pseudo terminal attributes, err=%d\n", e); - perror ("pt tcsetattr"); - } - -/* - * We had a problem here since the beginning. - * If no one was reading from the other end of the pseudo - * terminal, the buffer space would eventually fill up, - * the write here would block, and the receive decode - * thread would get stuck. - * - * March 2016 - A "select" was put before the read to - * solve a different problem. With that in place, we can - * now use non-blocking I/O and detect the buffer full - * condition here. - */ - - // text_color_set(DW_COLOR_DEBUG); - // dw_printf("Debug: Try using non-blocking mode for pseudo terminal.\n"); - - int flags = fcntl(fd, F_GETFL, 0); - e = fcntl (fd, F_SETFL, flags | O_NONBLOCK); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Can't set pseudo terminal to nonblocking, fcntl returns %d, errno = %d\n", e, errno); - perror ("pt fcntl"); - } - - text_color_set(DW_COLOR_INFO); - dw_printf("Virtual KISS TNC is available on %s\n", pt_slave_name); - - -#if 1 - // Sample code shows this. Why would we open it here? - // On Ubuntu, the slave side disappears after a few - // seconds if no one opens it. Same on Raspian which - // is also based on Debian. - // Need to revisit this. - - MYFDTYPE pt_slave_fd; - - pt_slave_fd = open(pt_slave_name, O_RDWR|O_NOCTTY); - - if (pt_slave_fd < 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Can't open %s\n", pt_slave_name); - perror (""); - return MYFDERROR; - } -#endif - -/* - * The device name is not the same every time. - * This is inconvenient for the application because it might - * be necessary to change the device name in the configuration. - * Create a symlink, /tmp/kisstnc, so the application configuration - * does not need to change when the pseudo terminal name changes. - */ - - unlink (TMP_KISSTNC_SYMLINK); - - -// TODO: Is this removed when application exits? - - if (symlink (pt_slave_name, TMP_KISSTNC_SYMLINK) == 0) { - dw_printf ("Created symlink %s -> %s\n", TMP_KISSTNC_SYMLINK, pt_slave_name); - } - else { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Failed to create symlink %s\n", TMP_KISSTNC_SYMLINK); - perror (""); - } - - return (fd); -} - -#endif - -/* - * Returns fd for our side of null modem or MYFDERROR for error. - */ - - -#if __CYGWIN__ || __WIN32__ - -static MYFDTYPE kiss_open_nullmodem (char *devicename) -{ - -#if __WIN32__ - - MYFDTYPE fd; - DCB dcb; - int ok; - char bettername[50]; - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kiss_open_nullmodem ( '%s' )\n", devicename); -#endif - -#if DEBUG9 - log_fp = fopen ("kiss-debug.txt", "w"); -#endif - -// Need to use FILE_FLAG_OVERLAPPED for full duplex operation. -// Without it, write blocks when waiting on read. - -// Read http://support.microsoft.com/kb/156932 - -// Bug fix in release 1.1 - Need to munge name for COM10 and up. -// http://support.microsoft.com/kb/115831 - - strlcpy (bettername, devicename, sizeof(bettername)); - if (strncasecmp(devicename, "COM", 3) == 0) { - int n; - n = atoi(devicename+3); - if (n >= 10) { - strlcpy (bettername, "\\\\.\\", sizeof(bettername)); - strlcat (bettername, devicename, sizeof(bettername)); - } - } - - fd = CreateFile(bettername, GENERIC_READ | GENERIC_WRITE, - 0, NULL, OPEN_EXISTING, FILE_FLAG_OVERLAPPED, NULL); - - if (fd == MYFDERROR) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("ERROR - Could not connect to %s side of null modem for Windows KISS TNC.\n", devicename); - return (MYFDERROR); - } - - /* Reference: http://msdn.microsoft.com/en-us/library/windows/desktop/aa363201(v=vs.85).aspx */ - - memset (&dcb, 0, sizeof(dcb)); - dcb.DCBlength = sizeof(DCB); - - ok = GetCommState (fd, &dcb); - if (! ok) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("kiss_open_nullmodem: GetCommState failed.\n"); - } - - /* http://msdn.microsoft.com/en-us/library/windows/desktop/aa363214(v=vs.85).aspx */ - - dcb.DCBlength = sizeof(DCB); - dcb.BaudRate = CBR_9600; // shouldn't matter - dcb.fBinary = 1; - dcb.fParity = 0; - dcb.fOutxCtsFlow = 0; - dcb.fOutxDsrFlow = 0; - dcb.fDtrControl = 0; - dcb.fDsrSensitivity = 0; - dcb.fOutX = 0; - dcb.fInX = 0; - dcb.fErrorChar = 0; - dcb.fNull = 0; /* Don't drop nul characters! */ - dcb.fRtsControl = 0; - dcb.ByteSize = 8; - dcb.Parity = NOPARITY; - dcb.StopBits = ONESTOPBIT; - - ok = SetCommState (fd, &dcb); - if (! ok) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("kiss_open_nullmodem: SetCommState failed.\n"); - } - - text_color_set(DW_COLOR_INFO); - dw_printf("Virtual KISS TNC is connected to %s side of null modem.\n", devicename); - -#else - -/* Cygwin version. */ - - int fd; - struct termios ts; - int e; - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kiss_open_nullmodem ( '%s' )\n", devicename); -#endif - - fd = open (devicename, O_RDWR); - - if (fd == MYFDERROR) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("ERROR - Could not connect to %s side of null modem for Windows KISS TNC.\n", devicename); - return (MYFDERROR); - } - - e = tcgetattr (fd, &ts); - if (e != 0) { perror ("nm tcgetattr"); } - - cfmakeraw (&ts); - - ts.c_cc[VMIN] = 1; /* wait for at least one character */ - ts.c_cc[VTIME] = 0; /* no fancy timing. */ - - e = tcsetattr (fd, TCSANOW, &ts); - if (e != 0) { perror ("nm tcsetattr"); } - - text_color_set(DW_COLOR_INFO); - dw_printf("Virtual KISS TNC is connected to %s side of null modem.\n", devicename); - -#endif - - return (fd); -} - -#endif - - - - -/*------------------------------------------------------------------- - * - * Name: kiss_send_rec_packet - * - * Purpose: Send a received packet or text string to the client app. - * - * Inputs: chan - Channel number where packet was received. - * 0 = first, 1 = second if any. - * - * pp - Identifier for packet object. - * - * fbuf - Address of raw received frame buffer - * or a text string. - * - * flen - Length of raw received frame not including the FCS - * or -1 for a text string. - * - * Description: Send message to client. - * We really don't care if anyone is listening or not. - * I don't even know if we can find out. - * - *--------------------------------------------------------------------*/ - - -void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen) -{ - unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN + 2]; - int kiss_len; - int err; - -#if ! __WIN32__ - if (pt_master_fd == MYFDERROR) { - return; - } -#endif - -#if __CYGWIN__ || __WIN32__ - - if (nullmodem_fd == MYFDERROR) { - return; - } -#endif - - if (flen < 0) { - flen = strlen((char*)fbuf); - if (kiss_debug) { - kiss_debug_print (TO_CLIENT, "Fake command prompt", fbuf, flen); - } - strlcpy ((char *)kiss_buff, (char *)fbuf, sizeof(kiss_buff)); - kiss_len = strlen((char *)kiss_buff); - } - else { - - - unsigned char stemp[AX25_MAX_PACKET_LEN + 1]; - - assert (flen < (int)(sizeof(stemp))); - - stemp[0] = (chan << 4) + 0; - memcpy (stemp+1, fbuf, flen); - - if (kiss_debug >= 2) { - /* AX.25 frame with the CRC removed. */ - text_color_set(DW_COLOR_DEBUG); - dw_printf ("\n"); - dw_printf ("Packet content before adding KISS framing and any escapes:\n"); - hex_dump (fbuf, flen); - } - - kiss_len = kiss_encapsulate (stemp, flen+1, kiss_buff); - - /* This has KISS framing and escapes for sending to client app. */ - - if (kiss_debug) { - kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len); - } - - } - -#if ! __WIN32__ - -/* Pseudo terminal for Cygwin and Linux. */ - - err = write (pt_master_fd, kiss_buff, (size_t)kiss_len); - - if (err == -1 && errno == EWOULDBLOCK) { - text_color_set (DW_COLOR_INFO); - dw_printf ("KISS SEND - Discarding message because no one is listening.\n"); - dw_printf ("This happens when you use the -p option and don't read from the pseudo terminal.\n"); - } - else if (err != kiss_len) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError sending KISS message to client application on pseudo terminal. fd=%d, len=%d, write returned %d, errno = %d\n\n", - pt_master_fd, kiss_len, err, errno); - perror ("pt write"); - } - -#endif - -#if __CYGWIN__ || __WIN32__ - - -/* - * This write can block if nothing is connected to the other end. - * The solution is found in the com0com ReadMe file: - * - * Q. My application hangs during its startup when it sends anything to one paired - * COM port. The only way to unhang it is to start HyperTerminal, which is connected - * to the other paired COM port. I didn't have this problem with physical serial - * ports. - * A. Your application can hang because receive buffer overrun is disabled by - * default. You can fix the problem by enabling receive buffer overrun for the - * receiving port. Also, to prevent some flow control issues you need to enable - * baud rate emulation for the sending port. So, if your application use port CNCA0 - * and other paired port is CNCB0, then: - * - * 1. Launch the Setup Command Prompt shortcut. - * 2. Enter the change commands, for example: - * - * command> change CNCB0 EmuOverrun=yes - * command> change CNCA0 EmuBR=yes - */ - -#if __WIN32__ - - DWORD nwritten; - - /* Without this, write blocks while we are waiting on a read. */ - static OVERLAPPED ov_wr; - memset (&ov_wr, 0, sizeof(ov_wr)); - - if ( ! WriteFile (nullmodem_fd, kiss_buff, kiss_len, &nwritten, &ov_wr)) - { - err = GetLastError(); - if (err != ERROR_IO_PENDING) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError sending KISS message to client application thru null modem. Error %d.\n\n", (int)GetLastError()); - //CloseHandle (nullmodem_fd); - //nullmodem_fd = MYFDERROR; - } - } - else if ((int)nwritten != kiss_len) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError sending KISS message to client application thru null modem. Only %d of %d written.\n\n", (int)nwritten, kiss_len); - //CloseHandle (nullmodem_fd); - //nullmodem_fd = MYFDERROR; - } - -#else - err = write (nullmodem_fd, kiss_buf, (size_t)kiss_len); - if (err != len) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError sending KISS message to client application thru null modem. err=%d\n\n", err); - //close (nullmodem_fd); - //nullmodem_fd = MYFDERROR; - } -#endif - -#endif - -} /* kiss_send_rec_packet */ - - - - -/*------------------------------------------------------------------- - * - * Name: kiss_get - * - * Purpose: Read one byte from the KISS client app. - * - * Global In: nullmodem_fd (Windows) or pt_master_fd (Linux) - * - * Returns: one byte (value 0 - 255) or terminate thread on error. - * - * Description: There is room for improvment here. Reading one byte - * at a time is inefficient. We could read a large block - * into a local buffer and return a byte from that most of the time. - * Is it worth the effort? I don't know. With GHz processors and - * the low data rate here it might not make a noticable difference. - * - *--------------------------------------------------------------------*/ - - -static int kiss_get (/* MYFDTYPE fd*/ void ) -{ - unsigned char ch; - -#if __WIN32__ /* Native Windows version. */ - - DWORD n; - static OVERLAPPED ov_rd; - - memset (&ov_rd, 0, sizeof(ov_rd)); - ov_rd.hEvent = CreateEvent (NULL, TRUE, FALSE, NULL); - - - /* Overlapped I/O makes reading rather complicated. */ - /* See: http://msdn.microsoft.com/en-us/library/ms810467.aspx */ - - /* It seems that the read completes OK with a count */ - /* of 0 every time we send a message to the serial port. */ - - n = 0; /* Number of characters read. */ - - while (n == 0) { - - if ( ! ReadFile (nullmodem_fd, &ch, 1, &n, &ov_rd)) - { - int err1 = GetLastError(); - - if (err1 == ERROR_IO_PENDING) - { - /* Wait for completion. */ - - if (WaitForSingleObject (ov_rd.hEvent, INFINITE) == WAIT_OBJECT_0) - { - if ( ! GetOverlappedResult (nullmodem_fd, &ov_rd, &n, 1)) - { - int err3 = GetLastError(); - - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nKISS GetOverlappedResult error %d.\n\n", err3); - } - else - { - /* Success! n should be 1 */ - } - } - } - else - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nKISS ReadFile error %d. Closing connection.\n\n", err1); - CloseHandle (nullmodem_fd); - nullmodem_fd = MYFDERROR; - //pthread_exit (NULL); - } - } - - } /* end while n==0 */ - - CloseHandle(ov_rd.hEvent); - - if (n != 1) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nKISS failed to get one byte. n=%d.\n\n", (int)n); - -#if DEBUG9 - fprintf (log_fp, "n=%d\n", n); -#endif - } - - -#else /* Linux/Cygwin version */ - - int n = 0; - fd_set fd_in, fd_ex; - int rc; - - while ( n == 0 ) { - -/* - * Since the beginning we've always had a couple annoying problems with - * the pseudo terminal KISS interface. - * When using "kissattach" we would sometimes get the error message: - * - * kissattach: Error setting line discipline: TIOCSETD: Device or resource busy - * Are you sure you have enabled MKISS support in the kernel - * or, if you made it a module, that the module is loaded? - * - * martinhpedersen came up with the interesting idea of putting in a "select" - * before the "read" and explained it like this: - * - * "Reading from master fd of the pty before the client has connected leads - * to trouble with kissattach. Use select to check if the slave has sent - * any data before trying to read from it." - * - * "This fix resolves the issue by not reading from the pty's master fd, until - * kissattach has opened and configured the slave. This is implemented using - * select() to wait for data before reading from the master fd." - * - * The submitted code looked like this: - * - * FD_ZERO(&fd_in); - * rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_in, NULL); - * - * That doesn't look right to me for a couple reasons. - * First, I would expect to use FD_SET for the fd. - * Second, using the same bit mask for two arguments doesn't seem - * like a good idea because select modifies them. - * When I tried running it, we don't get the failure message - * anymore but the select never returns so we can't read data from - * the KISS client app. - * - * I think this is what we want. - * - * Tested on Raspian (ARM) and Ubuntu (x86_64). - * We don't get the error from kissattach anymore. - */ - - FD_ZERO(&fd_in); - FD_SET(pt_master_fd, &fd_in); - - FD_ZERO(&fd_ex); - FD_SET(pt_master_fd, &fd_ex); - - rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_ex, NULL); - -#if 0 - text_color_set(DW_COLOR_DEBUG); - dw_printf ("select returns %d, errno=%d, fd=%d, fd_in=%08x, fd_ex=%08x\n", rc, errno, pt_master_fd, *((int*)(&fd_in)), *((int*)(&fd_in))); -#endif - - if (rc == 0) - { - continue; // When could we get a 0? - } - - if (rc == MYFDERROR - || (n = read(pt_master_fd, &ch, (size_t)1)) != 1) - { - - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError receiving KISS message from client application. Closing %s.\n\n", pt_slave_name); - perror (""); - - close (pt_master_fd); - - pt_master_fd = MYFDERROR; - unlink (TMP_KISSTNC_SYMLINK); - pthread_exit (NULL); - } - } - -#endif - -#if DEBUGx - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kiss_get(%d) returns 0x%02x\n", fd, ch); -#endif - -#if DEBUG9 - fprintf (log_fp, "%02x %c %c", ch, - isprint(ch) ? ch : '.' , - (isupper(ch>>1) || isdigit(ch>>1) || (ch>>1) == ' ') ? (ch>>1) : '.'); - if (ch == FEND) fprintf (log_fp, " FEND"); - if (ch == FESC) fprintf (log_fp, " FESC"); - if (ch == TFEND) fprintf (log_fp, " TFEND"); - if (ch == TFESC) fprintf (log_fp, " TFESC"); - if (ch == '\r') fprintf (log_fp, " CR"); - if (ch == '\n') fprintf (log_fp, " LF"); - fprintf (log_fp, "\n"); - if (ch == FEND) fflush (log_fp); -#endif - return (ch); -} - - -/*------------------------------------------------------------------- - * - * Name: kiss_listen_thread - * - * Purpose: Read messages from serial port KISS client application. - * - * Global In: nullmodem_fd (Windows) or pt_master_fd (Linux) - * - * Description: Reads bytes from the KISS client app and - * sends them to kiss_rec_byte for processing. - * - *--------------------------------------------------------------------*/ - - -static THREAD_F kiss_listen_thread (void *arg) -{ - unsigned char ch; - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kiss_listen_thread ( %d )\n", fd); -#endif - - - while (1) { - ch = kiss_get(); - kiss_rec_byte (&kf, ch, kiss_debug, kiss_send_rec_packet); - } - -#if __WIN32__ - return(0); -#else - return (THREAD_F) 0; /* Unreachable but avoids compiler warning. */ -#endif -} - -/* end kiss.c */ diff --git a/kiss.h b/kiss.h deleted file mode 100644 index 4c037fbb..00000000 --- a/kiss.h +++ /dev/null @@ -1,21 +0,0 @@ - -/* - * Name: kiss.h - */ - - -#include "ax25_pad.h" /* for packet_t */ - -#include "config.h" - - - - -void kiss_init (struct misc_config_s *misc_config); - -void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen); - -void kiss_serial_set_debug (int n); - - -/* end kiss.h */ diff --git a/kiss_frame.c b/kiss_frame.c deleted file mode 100644 index 3e9c45a8..00000000 --- a/kiss_frame.c +++ /dev/null @@ -1,681 +0,0 @@ -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2013, 2014 John Langner, WB2OSZ -// -// 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, see . -// - - - -/*------------------------------------------------------------------ - * - * Module: kiss_frame.c - * - * Purpose: Common code used by Serial port and network versions of KISS protocol. - * - * Description: The KISS TNS protocol is described in http://www.ka9q.net/papers/kiss.html - * - * Briefly, a frame is composed of - * - * * FEND (0xC0) - * * Contents - with special escape sequences so a 0xc0 - * byte in the data is not taken as end of frame. - * as part of the data. - * * FEND - * - * The first byte of the frame contains: - * - * * port number in upper nybble. - * * command in lower nybble. - * - * - * Commands from application recognized: - * - * 0 Data Frame AX.25 frame in raw format. - * - * 1 TXDELAY See explanation in xmit.c. - * - * 2 Persistence " " - * - * 3 SlotTime " " - * - * 4 TXtail " " - * Spec says it is obsolete but Xastir - * sends it and we respect it. - * - * 5 FullDuplex Ignored. Always full duplex. - * - * 6 SetHardware TNC specific. Ignored. - * - * FF Return Exit KISS mode. Ignored. - * - * - * Messages sent to client application: - * - * 0 Data Frame Received AX.25 frame in raw format. - * - *---------------------------------------------------------------*/ - -#include "direwolf.h" - -#include -#include -#include -#include -#include -#include - -#include "ax25_pad.h" -#include "textcolor.h" -#include "kiss_frame.h" -#include "tq.h" -#include "xmit.h" - -/* In server.c. Should probably move to some misc. function file. */ -void hex_dump (unsigned char *p, int len); - - - - - -#if KISSTEST - -#define dw_printf printf - -void text_color_set (dw_color_t c) -{ - return; -} - -#else - -static void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug); - -#endif - - - -/*------------------------------------------------------------------- - * - * Name: kiss_frame_init - * - * Purpose: Save information about valid channels for later error checking. - * - * Inputs: pa - Address of structure of type audio_s. - * - *-----------------------------------------------------------------*/ - -static struct audio_s *save_audio_config_p; - -void kiss_frame_init (struct audio_s *pa) -{ - save_audio_config_p = pa; -} - - -/*------------------------------------------------------------------- - * - * Name: kiss_encapsulate - * - * Purpose: Ecapsulate a frame into KISS format. - * - * Inputs: in - Address of input block. - * First byte is the "type indicator" with type and - * channel but we don't care about that here. - * - * This seems cumbersome and confusing to have this - * one byte offset when encapsulating an AX.25 frame. - * Maybe the type/channel byte should be passed in - * as a separate argument. - * - * Note that this is "binary" data and can contain - * nul (0x00) values. Don't treat it like a text string! - * - * ilen - Number of bytes in input block. - * - * Outputs: out - Address where to place the KISS encoded representation. - * The sequence is: - * FEND - Magic frame separator. - * data - with certain byte values replaced so - * FEND will never occur here. - * FEND - Magic frame separator. - * - * Returns: Number of bytes in the output. - * Absolute max length will be twice input plus 2. - * - *-----------------------------------------------------------------*/ - -int kiss_encapsulate (unsigned char *in, int ilen, unsigned char *out) -{ - int olen; - int j; - - olen = 0; - out[olen++] = FEND; - for (j=0; j - * <0x0d> - * XFLOW OFF<0x0d> - * FULLDUP OFF<0x0d> - * KISS ON<0x0d> - * RESTART<0x0d> - * <0x03><0x03><0x03> - * TC 1<0x0d> - * TN 2,0<0x0d><0x0d><0x0d> - * XFLOW OFF<0x0d> - * FULLDUP OFF<0x0d> - * KISS ON<0x0d> - * RESTART<0x0d> - * - * This keeps repeating over and over and over and over again if - * it doesn't get any sort of response. - * - * Let's try to keep it happy by sending back a command prompt. - */ - - - - -void kiss_rec_byte (kiss_frame_t *kf, unsigned char ch, int debug, void (*sendfun)(int,unsigned char*,int)) -{ - - //dw_printf ("kiss_frame ( %c %02x ) \n", ch, ch); - - switch (kf->state) { - - case KS_SEARCHING: /* Searching for starting FEND. */ - default: - - if (ch == FEND) { - - /* Start of frame. But first print any collected noise for debugging. */ - - if (kf->noise_len > 0) { - if (debug) { - kiss_debug_print (FROM_CLIENT, "Rejected Noise", kf->noise, kf->noise_len); - } - kf->noise_len = 0; - } - - kf->kiss_len = 0; - kf->kiss_msg[kf->kiss_len++] = ch; - kf->state = KS_COLLECTING; - return; - } - - /* Noise to be rejected. */ - - if (kf->noise_len < MAX_NOISE_LEN) { - kf->noise[kf->noise_len++] = ch; - } - if (ch == '\r') { - if (debug) { - kiss_debug_print (FROM_CLIENT, "Rejected Noise", kf->noise, kf->noise_len); - kf->noise[kf->noise_len] = '\0'; - } - - /* Try to appease client app by sending something back. */ - if (strcasecmp("restart\r", (char*)(kf->noise)) == 0 || - strcasecmp("reset\r", (char*)(kf->noise)) == 0) { - (*sendfun) (0, (unsigned char *)"\xc0\xc0", -1); - } - else { - (*sendfun) (0, (unsigned char *)"\r\ncmd:", -1); - } - kf->noise_len = 0; - } - return; - break; - - case KS_COLLECTING: /* Frame collection in progress. */ - - - if (ch == FEND) { - - unsigned char unwrapped[AX25_MAX_PACKET_LEN]; - int ulen; - - /* End of frame. */ - - if (kf->kiss_len == 0) { - /* Empty frame. Starting a new one. */ - kf->kiss_msg[kf->kiss_len++] = ch; - return; - } - if (kf->kiss_len == 1 && kf->kiss_msg[0] == FEND) { - /* Empty frame. Just go on collecting. */ - return; - } - - kf->kiss_msg[kf->kiss_len++] = ch; - if (debug) { - /* As received over the wire from client app. */ - kiss_debug_print (FROM_CLIENT, NULL, kf->kiss_msg, kf->kiss_len); - } - - ulen = kiss_unwrap (kf->kiss_msg, kf->kiss_len, unwrapped); - - if (debug >= 2) { - /* Append CRC to this and it goes out over the radio. */ - text_color_set(DW_COLOR_DEBUG); - dw_printf ("\n"); - dw_printf ("Packet content after removing KISS framing and any escapes:\n"); - /* Don't include the "type" indicator. */ - /* It contains the radio channel and type should always be 0 here. */ - hex_dump (unwrapped+1, ulen-1); - } - - kiss_process_msg (unwrapped, ulen, debug); - - kf->state = KS_SEARCHING; - return; - } - - if (kf->kiss_len < MAX_KISS_LEN) { - kf->kiss_msg[kf->kiss_len++] = ch; - } - else { - text_color_set(DW_COLOR_ERROR); - dw_printf ("KISS message exceeded maximum length.\n"); - } - return; - break; - } - - return; /* unreachable but suppress compiler warning. */ - -} /* end kiss_rec_byte */ - - - - -/*------------------------------------------------------------------- - * - * Name: kiss_process_msg - * - * Purpose: Process a message from the KISS client. - * - * Inputs: kiss_msg - Kiss frame with FEND and escapes removed. - * The first byte contains channel and command. - * - * kiss_len - Number of bytes including the command. - * - * debug - Debug option is selected. - * - *-----------------------------------------------------------------*/ - -static void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug) -{ - int port; - int cmd; - packet_t pp; - alevel_t alevel; - - port = (kiss_msg[0] >> 4) & 0xf; - cmd = kiss_msg[0] & 0xf; - - switch (cmd) - { - case 0: /* Data Frame */ - - /* Special hack - Discard apparently bad data from Linux AX25. */ - - if ((port == 2 || port == 8) && - kiss_msg[1] == 'Q' << 1 && - kiss_msg[2] == 'S' << 1 && - kiss_msg[3] == 'T' << 1 && - kiss_msg[4] == ' ' << 1 && - kiss_msg[15] == 3 && - kiss_msg[16] == 0xcd) { - - if (debug) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Special case - Drop packets which appear to be in error.\n"); - } - return; - } - - /* Verify that the port (channel) number is valid. */ - - if (port < 0 || port >= MAX_CHANS || ! save_audio_config_p->achan[port].valid) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Invalid transmit channel %d from KISS client app.\n", port); - text_color_set(DW_COLOR_DEBUG); - kiss_debug_print (FROM_CLIENT, NULL, kiss_msg, kiss_len); - return; - } - - memset (&alevel, 0xff, sizeof(alevel)); - pp = ax25_from_frame (kiss_msg+1, kiss_len-1, alevel); - if (pp == NULL) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("ERROR - Invalid KISS data frame from client app.\n"); - } - else { - - /* How can we determine if it is an original or repeated message? */ - /* If there is at least one digipeater in the frame, AND */ - /* that digipeater has been used, it should go out quickly thru */ - /* the high priority queue. */ - /* Otherwise, it is an original for the low priority queue. */ - - if (ax25_get_num_repeaters(pp) >= 1 && - ax25_get_h(pp,AX25_REPEATER_1)) { - tq_append (port, TQ_PRIO_0_HI, pp); - } - else { - tq_append (port, TQ_PRIO_1_LO, pp); - } - } - break; - - case 1: /* TXDELAY */ - - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol set TXDELAY = %d (*10mS units = %d mS), port %d\n", kiss_msg[1], kiss_msg[1] * 10, port); - xmit_set_txdelay (port, kiss_msg[1]); - break; - - case 2: /* Persistence */ - - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol set Persistence = %d, port %d\n", kiss_msg[1], port); - xmit_set_persist (port, kiss_msg[1]); - break; - - case 3: /* SlotTime */ - - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol set SlotTime = %d (*10mS units = %d mS), port %d\n", kiss_msg[1], kiss_msg[1] * 10, port); - xmit_set_slottime (port, kiss_msg[1]); - break; - - case 4: /* TXtail */ - - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol set TXtail = %d (*10mS units = %d mS), port %d\n", kiss_msg[1], kiss_msg[1] * 10, port); - xmit_set_txtail (port, kiss_msg[1]); - break; - - case 5: /* FullDuplex */ - - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol set FullDuplex = %d, port %d\n", kiss_msg[1], port); - break; - - case 6: /* TNC specific */ - - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol set hardware - ignored.\n"); - break; - - case 15: /* End KISS mode, port should be 15. */ - /* Ignore it. */ - text_color_set(DW_COLOR_INFO); - dw_printf ("KISS protocol end KISS mode\n"); - break; - - default: - text_color_set(DW_COLOR_ERROR); - dw_printf ("KISS Invalid command %d\n", cmd); - kiss_debug_print (FROM_CLIENT, NULL, kiss_msg, kiss_len); - - text_color_set(DW_COLOR_INFO); - dw_printf ("Troubleshooting tip:\n"); - dw_printf ("Use \"-d kn\" option on direwolf command line to observe\n"); - dw_printf ("all communication with the client application.\n"); - - break; - } - -} /* end kiss_process_msg */ - - -/*------------------------------------------------------------------- - * - * Name: kiss_debug_print - * - * Purpose: Print message to/from client for debugging. - * - * Inputs: fromto - Direction of message. - * special - Comment if not a KISS frame. - * pmsg - Address of the message block. - * msg_len - Length of the message. - * - *--------------------------------------------------------------------*/ - - -void kiss_debug_print (fromto_t fromto, char *special, unsigned char *pmsg, int msg_len) -{ - const char *direction [2] = { "from", "to" }; - const char *prefix [2] = { "<<<", ">>>" }; - const char *function[16] = { - "Data frame", "TXDELAY", "P", "SlotTime", - "TXtail", "FullDuplex", "SetHardware", "Invalid 7", - "Invalid 8", "Invalid 9", "Invalid 10", "Invalid 11", - "Invalid 12", "Invalid 13", "Invalid 14", "Return" }; - - - text_color_set(DW_COLOR_DEBUG); - dw_printf ("\n"); - - if (special == NULL) { - unsigned char *p; /* to skip over FEND if present. */ - - p = pmsg; - if (*p == FEND) p++; - - dw_printf ("%s %s %s KISS client application, port %d, total length = %d\n", - prefix[(int)fromto], function[p[0] & 0xf], direction[(int)fromto], - (p[0] >> 4) & 0xf, msg_len); - } - else { - dw_printf ("%s %s %s KISS client application, total length = %d\n", - prefix[(int)fromto], special, direction[(int)fromto], - msg_len); - } - hex_dump (pmsg, msg_len); - -} /* end kiss_debug_print */ - - -#endif - - -/* Quick unit test for encapsulate & unwrap */ - -// $ gcc -DKISSTEST kiss_frame.c ; ./a -// Quick KISS test passed OK. - - -#if KISSTEST - - -int main () -{ - unsigned char din[512]; - unsigned char kissed[520]; - unsigned char dout[520]; - int klen; - int dlen; - int k; - - for (k = 0; k < 512; k++) { - if (k < 256) { - din[k] = k; - } - else { - din[k] = 511 - k; - } - } - - klen = kiss_encapsulate (din, 512, kissed); - assert (klen == 512 + 6); - - dlen = kiss_unwrap (kissed, klen, dout); - assert (dlen == 512); - assert (memcmp(din, dout, 512) == 0); - - dlen = kiss_unwrap (kissed+1, klen-1, dout); - assert (dlen == 512); - assert (memcmp(din, dout, 512) == 0); - - dw_printf ("Quick KISS test passed OK.\n"); - exit (EXIT_SUCCESS); -} - -#endif - -#endif /* WALK96 */ - -/* end kiss_frame.c */ diff --git a/kiss_frame.h b/kiss_frame.h deleted file mode 100644 index ac84b820..00000000 --- a/kiss_frame.h +++ /dev/null @@ -1,54 +0,0 @@ - -/* kiss_frame.h */ - -#include "audio.h" /* for struct audio_s */ - - -/* - * Special characters used by SLIP protocol. - */ - -#define FEND 0xC0 -#define FESC 0xDB -#define TFEND 0xDC -#define TFESC 0xDD - - -enum kiss_state_e { - KS_SEARCHING, /* Looking for FEND to start KISS frame. */ - KS_COLLECTING}; /* In process of collecting KISS frame. */ - - -#define MAX_KISS_LEN 2048 /* Spec calls for at least 1024. */ - /* Might want to make it longer to accomodate */ - /* maximum packet length. */ - -#define MAX_NOISE_LEN 100 - -typedef struct kiss_frame_s { - - enum kiss_state_e state; - - unsigned char kiss_msg[MAX_KISS_LEN]; - /* Leading FEND is optional. */ - /* Contains escapes and ending FEND. */ - int kiss_len; - - unsigned char noise[MAX_NOISE_LEN]; - int noise_len; - -} kiss_frame_t; - - -void kiss_frame_init (struct audio_s *pa); - -int kiss_encapsulate (unsigned char *in, int ilen, unsigned char *out); - -void kiss_rec_byte (kiss_frame_t *kf, unsigned char ch, int debug, void (*sendfun)(int,unsigned char*,int)); - - -typedef enum fromto_e { FROM_CLIENT=0, TO_CLIENT=1 } fromto_t; - -void kiss_debug_print (fromto_t fromto, char *special, unsigned char *pmsg, int msg_len); - -/* end kiss_frame.h */ \ No newline at end of file diff --git a/kissnet.c b/kissnet.c deleted file mode 100644 index ccdbd0a3..00000000 --- a/kissnet.c +++ /dev/null @@ -1,705 +0,0 @@ -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2011-2014, 2015 John Langner, WB2OSZ -// -// 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, see . -// - - -/*------------------------------------------------------------------ - * - * Module: kissnet.c - * - * Purpose: Provide service to other applications via KISS protocol via TCP socket. - * - * Input: - * - * Outputs: - * - * Description: This provides a TCP socket for communication with a client application. - * - * It implements the KISS TNS protocol as described in: - * http://www.ka9q.net/papers/kiss.html - * - * Briefly, a frame is composed of - * - * * FEND (0xC0) - * * Contents - with special escape sequences so a 0xc0 - * byte in the data is not taken as end of frame. - * as part of the data. - * * FEND - * - * The first byte of the frame contains: - * - * * port number in upper nybble. - * * command in lower nybble. - * - * - * Commands from application recognized: - * - * 0 Data Frame AX.25 frame in raw format. - * - * 1 TXDELAY See explanation in xmit.c. - * - * 2 Persistence " " - * - * 3 SlotTime " " - * - * 4 TXtail " " - * Spec says it is obsolete but Xastir - * sends it and we respect it. - * - * 5 FullDuplex Ignored. Always full duplex. - * - * 6 SetHardware TNC specific. Ignored. - * - * FF Return Exit KISS mode. Ignored. - * - * - * Messages sent to client application: - * - * 0 Data Frame Received AX.25 frame in raw format. - * - * - * - * - * References: Getting Started with Winsock - * http://msdn.microsoft.com/en-us/library/windows/desktop/bb530742(v=vs.85).aspx - * - * Future: Originally we had: - * KISS over serial port. - * AGW over socket. - * This is the two of them munged together and we end up with duplicate code. - * It would have been better to separate out the transport and application layers. - * Maybe someday. - * - *---------------------------------------------------------------*/ - - -/* - * Native Windows: Use the Winsock interface. - * Linux: Use the BSD socket interface. - * Cygwin: Can use either one. - */ - - -#include "direwolf.h" // Sets _WIN32_WINNT for XP API level needed by ws2tcpip.h - - -#if __WIN32__ -#include -#include // _WIN32_WINNT must be set to 0x0501 before including this -#else -#include -#include -#include -#include -#include -#ifdef __OpenBSD__ -#include -#else -#include -#endif -#endif - -#include -#include -#include -#include - - -#include "tq.h" -#include "ax25_pad.h" -#include "textcolor.h" -#include "audio.h" -#include "kissnet.h" -#include "kiss_frame.h" -#include "xmit.h" - -void hex_dump (unsigned char *p, int len); // This should be in a .h file. - - -static kiss_frame_t kf; /* Accumulated KISS frame and state of decoder. */ - // TODO: multiple instances if multiple KISS network clients! - - -static int client_sock; /* File descriptor for socket for */ - /* communication with client application. */ - /* Set to -1 if not connected. */ - /* (Don't use SOCKET type because it is unsigned.) */ - - -// TODO: define in one place, use everywhere. -#if __WIN32__ -#define THREAD_F unsigned __stdcall -#else -#define THREAD_F void * -#endif - -static THREAD_F connect_listen_thread (void *arg); -static THREAD_F kissnet_listen_thread (void *arg); - - - -static int kiss_debug = 0; /* Print information flowing from and to client. */ - -void kiss_net_set_debug (int n) -{ - kiss_debug = n; -} - - - -/*------------------------------------------------------------------- - * - * Name: kissnet_init - * - * Purpose: Set up a server to listen for connection requests from - * an application such as Xastir or APRSIS32. - * - * Inputs: mc->kiss_port - TCP port for server. - * Main program has default of 8000 but allows - * an alternative to be specified on the command line - * - * 0 means disable. New in version 1.2. - * - * Outputs: - * - * Description: This starts two threads: - * * to listen for a connection from client app. - * * to listen for commands from client app. - * so the main application doesn't block while we wait for these. - * - *--------------------------------------------------------------------*/ - - -void kissnet_init (struct misc_config_s *mc) -{ -#if __WIN32__ - HANDLE connect_listen_th; - HANDLE cmd_listen_th; -#else - pthread_t connect_listen_tid; - pthread_t cmd_listen_tid; - int e; -#endif - int kiss_port = mc->kiss_port; - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kissnet_init ( %d )\n", kiss_port); -#endif - - memset (&kf, 0, sizeof(kf)); - - client_sock = -1; - - if (kiss_port == 0) { - text_color_set(DW_COLOR_INFO); - dw_printf ("Disabled KISS network client port.\n"); - return; - } - -/* - * This waits for a client to connect and sets client_sock. - */ -#if __WIN32__ - connect_listen_th = (HANDLE)_beginthreadex (NULL, 0, connect_listen_thread, (void *)kiss_port, 0, NULL); - if (connect_listen_th == NULL) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Could not create KISS socket connect listening thread\n"); - return; - } -#else - e = pthread_create (&connect_listen_tid, NULL, connect_listen_thread, (void *)(long)kiss_port); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - perror("Could not create KISS socket connect listening thread"); - return; - } -#endif - -/* - * This reads messages from client when client_sock is valid. - */ -#if __WIN32__ - cmd_listen_th = (HANDLE)_beginthreadex (NULL, 0, kissnet_listen_thread, NULL, 0, NULL); - if (cmd_listen_th == NULL) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Could not create KISS socket command listening thread\n"); - return; - } -#else - e = pthread_create (&cmd_listen_tid, NULL, kissnet_listen_thread, NULL); - if (e != 0) { - text_color_set(DW_COLOR_ERROR); - perror("Could not create KISS socket command listening thread"); - return; - } -#endif -} - - -/*------------------------------------------------------------------- - * - * Name: connect_listen_thread - * - * Purpose: Wait for a connection request from an application. - * - * Inputs: arg - TCP port for server. - * Main program has default of 8001 but allows - * an alternative to be specified on the command line - * - * Outputs: client_sock - File descriptor for communicating with client app. - * - * Description: Wait for connection request from client and establish - * communication. - * Note that the client can go away and come back again and - * re-establish communication without restarting this application. - * - *--------------------------------------------------------------------*/ - -static THREAD_F connect_listen_thread (void *arg) -{ -#if __WIN32__ - - struct addrinfo hints; - struct addrinfo *ai = NULL; - int err; - char kiss_port_str[12]; - - SOCKET listen_sock; - WSADATA wsadata; - - snprintf (kiss_port_str, sizeof(kiss_port_str), "%d", (int)(long)arg); -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("DEBUG: kissnet port = %d = '%s'\n", (int)(long)arg, kiss_port_str); -#endif - err = WSAStartup (MAKEWORD(2,2), &wsadata); - if (err != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf("WSAStartup failed: %d\n", err); - return (0); - } - - if (LOBYTE(wsadata.wVersion) != 2 || HIBYTE(wsadata.wVersion) != 2) { - text_color_set(DW_COLOR_ERROR); - dw_printf("Could not find a usable version of Winsock.dll\n"); - WSACleanup(); - //sleep (1); - return (0); - } - - memset (&hints, 0, sizeof(hints)); - hints.ai_family = AF_INET; - hints.ai_socktype = SOCK_STREAM; - hints.ai_protocol = IPPROTO_TCP; - hints.ai_flags = AI_PASSIVE; - - err = getaddrinfo(NULL, kiss_port_str, &hints, &ai); - if (err != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf("getaddrinfo failed: %d\n", err); - //sleep (1); - WSACleanup(); - return (0); - } - - listen_sock= socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol); - if (listen_sock == INVALID_SOCKET) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("connect_listen_thread: Socket creation failed, err=%d", WSAGetLastError()); - return (0); - } - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf("Binding to port %s ... \n", kiss_port_str); -#endif - - err = bind( listen_sock, ai->ai_addr, (int)ai->ai_addrlen); - if (err == SOCKET_ERROR) { - text_color_set(DW_COLOR_ERROR); - dw_printf("Bind failed with error: %d\n", WSAGetLastError()); // TODO: provide corresponding text. - dw_printf("Some other application is probably already using port %s.\n", kiss_port_str); - dw_printf("Try using a different port number with KISSPORT in the configuration file.\n"); - freeaddrinfo(ai); - closesocket(listen_sock); - WSACleanup(); - return (0); - } - - freeaddrinfo(ai); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf("opened KISS socket as fd (%d) on port (%s) for stream i/o\n", listen_sock, kiss_port_str ); -#endif - - while (1) { - - while (client_sock > 0) { - SLEEP_SEC(1); /* Already connected. Try again later. */ - } - -#define QUEUE_SIZE 5 - - if(listen(listen_sock,QUEUE_SIZE) == SOCKET_ERROR) - { - text_color_set(DW_COLOR_ERROR); - dw_printf("Listen failed with error: %d\n", WSAGetLastError()); - return (0); - } - - text_color_set(DW_COLOR_INFO); - dw_printf("Ready to accept KISS client application on port %s ...\n", kiss_port_str); - - client_sock = accept(listen_sock, NULL, NULL); - - if (client_sock == -1) { - text_color_set(DW_COLOR_ERROR); - dw_printf("Accept failed with error: %d\n", WSAGetLastError()); - closesocket(listen_sock); - WSACleanup(); - return (0); - } - - text_color_set(DW_COLOR_INFO); - dw_printf("\nConnected to KISS client application ...\n\n"); - - } - - -#else /* End of Windows case, now Linux. */ - - - struct sockaddr_in sockaddr; /* Internet socket address stuct */ - socklen_t sockaddr_size = sizeof(struct sockaddr_in); - int kiss_port = (int)(long)arg; - int listen_sock; - int bcopt = 1; - - listen_sock= socket(AF_INET,SOCK_STREAM,0); - if (listen_sock == -1) { - text_color_set(DW_COLOR_ERROR); - perror ("connect_listen_thread: Socket creation failed"); - return (NULL); - } - - /* Version 1.3 - as suggested by G8BPQ. */ - /* Without this, if you kill the application then try to run it */ - /* again quickly the port number is unavailable for a while. */ - /* Don't try doing the same thing On Windows; It has a different meaning. */ - /* http://stackoverflow.com/questions/14388706/socket-options-so-reuseaddr-and-so-reuseport-how-do-they-differ-do-they-mean-t */ - - setsockopt (listen_sock, SOL_SOCKET, SO_REUSEADDR, (const char *)&bcopt, 4); - - sockaddr.sin_addr.s_addr = INADDR_ANY; - sockaddr.sin_port = htons(kiss_port); - sockaddr.sin_family = AF_INET; - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf("Binding to port %d ... \n", kiss_port); -#endif - - if (bind(listen_sock,(struct sockaddr*)&sockaddr,sizeof(sockaddr)) == -1) { - text_color_set(DW_COLOR_ERROR); - dw_printf("Bind failed with error: %d\n", errno); - dw_printf("%s\n", strerror(errno)); - dw_printf("Some other application is probably already using port %d.\n", kiss_port); - dw_printf("Try using a different port number with KISSPORT in the configuration file.\n"); - return (NULL); - } - - getsockname( listen_sock, (struct sockaddr *)(&sockaddr), &sockaddr_size); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf("opened KISS socket as fd (%d) on port (%d) for stream i/o\n", listen_sock, ntohs(sockaddr.sin_port) ); -#endif - - while (1) { - - while (client_sock > 0) { - SLEEP_SEC(1); /* Already connected. Try again later. */ - } - -#define QUEUE_SIZE 5 - - if(listen(listen_sock,QUEUE_SIZE) == -1) - { - text_color_set(DW_COLOR_ERROR); - perror ("connect_listen_thread: Listen failed"); - return (NULL); - } - - text_color_set(DW_COLOR_INFO); - dw_printf("Ready to accept KISS client application on port %d ...\n", kiss_port); - - client_sock = accept(listen_sock, (struct sockaddr*)(&sockaddr),&sockaddr_size); - - text_color_set(DW_COLOR_INFO); - dw_printf("\nConnected to KISS client application ...\n\n"); - - } -#endif -} - - - - - -/*------------------------------------------------------------------- - * - * Name: kissnet_send_rec_packet - * - * Purpose: Send a received packet to the client app. - * - * Inputs: chan - Channel number where packet was received. - * 0 = first, 1 = second if any. - * - * fbuf - Address of raw received frame buffer - * or a text string. - * - * flen - Number of bytes for AX.25 frame. - * or -1 for a text string. - * - * - * Description: Send message to client if connected. - * Disconnect from client, and notify user, if any error. - * - *--------------------------------------------------------------------*/ - - -void kissnet_send_rec_packet (int chan, unsigned char *fbuf, int flen) -{ - unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN]; - int kiss_len; - int err; - - - if (client_sock == -1) { - return; - } - if (flen < 0) { - flen = strlen((char*)fbuf); - if (kiss_debug) { - kiss_debug_print (TO_CLIENT, "Fake command prompt", fbuf, flen); - } - strlcpy ((char *)kiss_buff, (char *)fbuf, sizeof(kiss_buff)); - kiss_len = strlen((char *)kiss_buff); - } - else { - - - unsigned char stemp[AX25_MAX_PACKET_LEN + 1]; - - assert (flen < (int)(sizeof(stemp))); - - stemp[0] = (chan << 4) + 0; - memcpy (stemp+1, fbuf, flen); - - if (kiss_debug >= 2) { - /* AX.25 frame with the CRC removed. */ - text_color_set(DW_COLOR_DEBUG); - dw_printf ("\n"); - dw_printf ("Packet content before adding KISS framing and any escapes:\n"); - hex_dump (fbuf, flen); - } - - kiss_len = kiss_encapsulate (stemp, flen+1, kiss_buff); - - /* This has the escapes and the surrounding FENDs. */ - - if (kiss_debug) { - kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len); - } - } - -#if __WIN32__ - err = send (client_sock, (char*)kiss_buff, kiss_len, 0); - if (err == SOCKET_ERROR) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError %d sending message to KISS client application. Closing connection.\n\n", WSAGetLastError()); - closesocket (client_sock); - client_sock = -1; - WSACleanup(); - } -#else - err = write (client_sock, kiss_buff, kiss_len); - if (err <= 0) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError sending message to KISS client application. Closing connection.\n\n"); - close (client_sock); - client_sock = -1; - } -#endif - -} /* end kissnet_send_rec_packet */ - - - -/*------------------------------------------------------------------- - * - * Name: read_from_socket - * - * Purpose: Read from socket until we have desired number of bytes. - * - * Inputs: fd - file descriptor. - * ptr - address where data should be placed. - * len - desired number of bytes. - * - * Description: Just a wrapper for the "read" system call but it should - * never return fewer than the desired number of bytes. - * - * Not really needed for KISS because we are dealing with - * a stream of bytes rather than message blocks. - * - *--------------------------------------------------------------------*/ - -static int read_from_socket (int fd, char *ptr, int len) -{ - int got_bytes = 0; - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("read_from_socket (%d, %p, %d)\n", fd, ptr, len); -#endif - while (got_bytes < len) { - int n; - -#if __WIN32__ - -//TODO: any flags for send/recv? -//TODO: Would be useful to have more detailed explanation from the error code. - - n = recv (fd, ptr + got_bytes, len - got_bytes, 0); -#else - n = read (fd, ptr + got_bytes, len - got_bytes); -#endif - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("read_from_socket: n = %d\n", n); -#endif - if (n <= 0) { - return (n); - } - - got_bytes += n; - } - assert (got_bytes >= 0 && got_bytes <= len); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("read_from_socket: return %d\n", got_bytes); -#endif - return (got_bytes); -} - - -/*------------------------------------------------------------------- - * - * Name: kissnet_listen_thread - * - * Purpose: Wait for KISS messages from an application. - * - * Inputs: arg - Not used. - * - * Outputs: client_sock - File descriptor for communicating with client app. - * - * Description: Process messages from the client application. - * Note that the client can go away and come back again and - * re-establish communication without restarting this application. - * - *--------------------------------------------------------------------*/ - - -/* Return one byte (value 0 - 255) */ - - -static int kiss_get (void) -{ - unsigned char ch; - int n; - - while (1) { - - while (client_sock <= 0) { - SLEEP_SEC(1); /* Not connected. Try again later. */ - } - - /* Just get one byte at a time. */ - - n = read_from_socket (client_sock, (char *)(&ch), 1); - - if (n == 1) { -#if DEBUG9 - dw_printf (log_fp, "%02x %c %c", ch, - isprint(ch) ? ch : '.' , - (isupper(ch>>1) || isdigit(ch>>1) || (ch>>1) == ' ') ? (ch>>1) : '.'); - if (ch == FEND) fprintf (log_fp, " FEND"); - if (ch == FESC) fprintf (log_fp, " FESC"); - if (ch == TFEND) fprintf (log_fp, " TFEND"); - if (ch == TFESC) fprintf (log_fp, " TFESC"); - if (ch == '\r') fprintf (log_fp, " CR"); - if (ch == '\n') fprintf (log_fp, " LF"); - fprintf (log_fp, "\n"); - if (ch == FEND) fflush (log_fp); -#endif - return(ch); - } - - text_color_set(DW_COLOR_ERROR); - dw_printf ("\nError reading KISS byte from client application. Closing connection.\n\n"); -#if __WIN32__ - closesocket (client_sock); -#else - close (client_sock); -#endif - client_sock = -1; - } -} - - - -static THREAD_F kissnet_listen_thread (void *arg) -{ - unsigned char ch; - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("kissnet_listen_thread ( socket = %d )\n", client_sock); -#endif - - while (1) { - ch = kiss_get(); - kiss_rec_byte (&kf, ch, kiss_debug, kissnet_send_rec_packet); - } - -#if __WIN32__ - return(0); -#else - return (THREAD_F) 0; /* Unreachable but avoids compiler warning. */ -#endif - -} /* end kissnet_listen_thread */ - -/* end kissnet.c */ diff --git a/kissnet.h b/kissnet.h deleted file mode 100644 index 361f435f..00000000 --- a/kissnet.h +++ /dev/null @@ -1,21 +0,0 @@ - -/* - * Name: kissnet.h - */ - - -#include "ax25_pad.h" /* for packet_t */ - -#include "config.h" - - - - -void kissnet_init (struct misc_config_s *misc_config); - -void kissnet_send_rec_packet (int chan, unsigned char *fbuf, int flen); - -void kiss_net_set_debug (int n); - - -/* end kissnet.h */ diff --git a/man/CMakeLists.txt b/man/CMakeLists.txt new file mode 100644 index 00000000..071db62a --- /dev/null +++ b/man/CMakeLists.txt @@ -0,0 +1,13 @@ +if(NOT (WIN32 OR CYGWIN)) + install(FILES "${CUSTOM_MAN_DIR}/aclients.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/atest.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/decode_aprs.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/direwolf.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/gen_packets.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/kissutil.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/ll2utm.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/log2gpx.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/text2tt.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/tt2text.1" DESTINATION ${INSTALL_MAN_DIR}) + install(FILES "${CUSTOM_MAN_DIR}/utm2ll.1" DESTINATION ${INSTALL_MAN_DIR}) +endif(NOT (WIN32 OR CYGWIN)) diff --git a/man1/aclients.1 b/man/aclients.1 similarity index 94% rename from man1/aclients.1 rename to man/aclients.1 index 578c1f3e..71dab11c 100644 --- a/man1/aclients.1 +++ b/man/aclients.1 @@ -48,5 +48,5 @@ of how to set up tests and the results. .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/atest.1 b/man/atest.1 similarity index 51% rename from man1/atest.1 rename to man/atest.1 index 83fba0fc..58c90f64 100644 --- a/man1/atest.1 +++ b/man/atest.1 @@ -21,17 +21,50 @@ atest \- Decode AX.25 frames from an audio file. .SH OPTIONS + .TP -.BI "-B " "n" -Bits / second for data. Proper modem selected for 300, 1200, 9600. -300 baud uses 1600/1800 Hz AFSK. -1200 (default) baud uses 1200/2200 Hz AFSK. -9600 baud uses K9NG/G2RUH standard. +.BI "-B " "n" +Data rate in bits/sec. Standard values are 300, 1200, 2400, 4800, 9600. +.PD 0 +.RS +.RS +300 bps defaults to AFSK tones of 1600 & 1800. +.P +1200 bps uses AFSK tones of 1200 & 2200. +.P +2400 bps uses QPSK based on V.26 standard. +.P +4800 bps uses 8PSK based on V.27 standard. +.P +9600 bps and up uses K9NG/G3RUH standard. +.P +AIS for ship Automatic Identification System. +.P +EAS for Emergency Alert System (EAS) Specific Area Message Encoding (SAME). +.RE +.RE +.PD + +.TP +.BI "-g " +Force G3RUH modem regardless of data rate. + +.TP +.BI "-j " +2400 bps QPSK compatible with Dire Wolf <= 1.5. + +.TP +.BI "-J " +2400 bps QPSK compatible with MFJ-2400. .TP .BI "-D " "n" Divide audio sample rate by n. +.TP +.BI "-h " +Print frame contents as hexadecimal bytes. + .TP .BI "-F " "n" Amount of effort to try fixing frames with an invalid CRC. @@ -39,9 +72,17 @@ Amount of effort to try fixing frames with an invalid CRC. 1 = Try to fix only a single bit. more = Try modifying more bits to get a good CRC. +.TP +.BI "-L " +Error if Less than this number decoded. + +.TP +.BI "-G " +Error if Greater than this number decoded. + .TP .BI "-P " "m" -Select the demodulator type such as A, B, C, D (default for 300 baud), E (default for 1200 baud), F, A+, B+, C+, D+, E+, F+. +Select the demodulator type such as D (default for 300 bps), E+ (default for 1200 bps), PQRS for 2400 bps, etc. @@ -72,20 +113,20 @@ This generates and decodes 3 test files with 1200, 300, and 9600 bits per second .PD 0 .B atest 02_Track_2.wav .P -.B atest -P C+ 02_Track_2.wav +.B atest -P E- 02_Track_2.wav .P .B atest -F 1 02_Track_2.wav .P -.B atest -P C+ -F 1 02_Track_2.wav +.B atest -P E- -F 1 02_Track_2.wav .PD .P .RS -Try different combinations of options to find the best decoding performance. +Try different combinations of options to compare decoding performance. .RE .P .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, cm108, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/decode_aprs.1 b/man/decode_aprs.1 similarity index 76% rename from man1/decode_aprs.1 rename to man/decode_aprs.1 index b7b24d9f..634fd669 100644 --- a/man1/decode_aprs.1 +++ b/man/decode_aprs.1 @@ -9,7 +9,9 @@ decode_aprs \- Convert APRS raw data to human readable form. [ \fItext-file\fR ] .RS .P -\fItext-file\fR should contain AX.25 packets in the standard monitoring format. +\fItext-file\fR should contain AX.25 packets in the standard monitoring format or +as a series two digit hexadecimal numbers. +If the first number is 00 or c0, it will be treated as a KISS frame. If no file specified, data will be read from stdin. .P .RE @@ -43,12 +45,16 @@ Pipe it into decode_aprs to find out. http://www.findu.com/cgi-bin/errors.cgi has a never-ending collection of packets with errors. Sometimes it's not obvious what is wrong with them. Dire Wolf will usually tell you what is wrong. First, -cut-n-paste the bad packets into a text file. Here a couple examples: +cut-n-paste the bad packets into a text file. Here a few examples: .P .RS +.nf n2cma>APRS,TCPIP*,qAC,SEVENTH:@212127z43.2333n/77.1w_338/002g001t025P000h65b10208.wview_5_19_0 .P K0YTH-10>APNU3B,NULL,qAR,K0DMF-10:!4601.5NS09255.52W#PHG6360/W2,MNn 444.575 +.P +00 82 a0 ae ae 62 60 e0 82 96 68 84 40 40 60 9c 68 b0 ae 86 40 e0 40 ae 92 88 8a 64 63 03 f0 3e 45 4d 36 34 6e 65 2f 23 20 45 63 68 6f 6c 69 6e 6b 20 31 34 35 2e 33 31 30 2f 31 30 30 68 7a 20 54 6f 6e 65 ++.fi .RE .P If you simply fed this into decode_aprs, it would complain about the @@ -67,7 +73,7 @@ Address has lower case letters. "n2cma" must be all upper case. .P After changing the source address to upper case, there are other issues. Identifying them is left as an exercise for the reader. .P -And in the second example, +In the second example, .P .RS .PD 0 @@ -77,6 +83,17 @@ Invalid character in longitude. Found '9' when expecting 0 or 1 for hundreds of .PD .RE +.P +In the third example, +.P +.RS +.PD 0 +Warning: Lower case letter in Maidenhead locator. Specification requires upper case. +.P +Digi2 Address, " WIDE2-1" contains character other than letter or digit in character position 1. +.PD +.RE + .SH SEE ALSO diff --git a/man1/direwolf.1 b/man/direwolf.1 similarity index 58% rename from man1/direwolf.1 rename to man/direwolf.1 index 36beb648..93f786dc 100644 --- a/man1/direwolf.1 +++ b/man/direwolf.1 @@ -27,8 +27,12 @@ RMS Express, and many others. Read configuration file from specified location rather than the default locations. .TP -.BI "-l " "dir" -Generate log files in specified directory. Use "." for current directory. +.BI "-l " "logdir" +Generate daily log files in specified directory. Use "." for current directory. + +.TP +.BI "-L " "logfile" +Generate single log file with fixed name. .TP .BI "-r " "n" @@ -44,23 +48,55 @@ Audio sample size for first channel. 8 or 16. Default 16. .TP .BI "-B " "n" -Data rate in bits/sec for first channel. Standard values are 300, 1200, 9600. +Data rate in bits/sec for first channel. Standard values are 300, 1200, 2400, 4800, 9600. .PD 0 -.RS .RS -If < 600, tones are set to 1600 & 1800. +.RS +300 bps defaults to AFSK tones of 1600 & 1800. +.P +1200 bps uses AFSK tones of 1200 & 2200. +.P +2400 bps uses QPSK based on V.26 standard. .P -If > 2400, K9NG/G3RUH scrambling is used. +4800 bps uses 8PSK based on V.27 standard. .P -Otherwise, AFSK tones are set to 1200 & 2200. +9600 bps and up uses K9NG/G3RUH standard. +.P +AIS for ship Automatic Identification System. +.P +EAS for Emergency Alert System (EAS) Specific Area Message Encoding (SAME). .RE .RE .PD +.TP +.BI "-g " +Force G3RUH modem regardless of data rate. + +.TP +.BI "-j " +2400 bps QPSK compatible with Dire Wolf <= 1.5. + +.TP +.BI "-J " +2400 bps QPSK compatible with MFJ-2400. + .TP .BI "-D " "n" Divide audio sample by n for first channel. +.TP +.BI "-X " "n" +1 to enable FX.25 transmit. 16, 32, 64 for specific number of check bytes. + +.TP +.BI "-I " "n" +Enable IL2P transmit. n=1 is recommended. 0 uses weaker FEC. + +.TP +.BI "-i " "n" +Enable IL2P transmit, inverted polarity. n=1 is recommended. 0 uses weaker FEC. + .TP .BI "-d " "x" Debug options. Specify one or more of the following in place of x. @@ -88,6 +124,14 @@ o = Output controls such as PTT and DCD. i = IGate .P h = Hamlib verbose level. Repeat for more. +.P +m = Monitor heard station list. +.P +f = Packet filtering. +.P +x = FX.25 increase verbose level. +.P +d = APRStt (DTMF to APRS object conversion). .RE .RE .PD @@ -101,24 +145,42 @@ Quiet (suppress output). Specify one or more of the following in place of x. h = Heard line with the audio level. .P d = Decoding of APRS packets. +.P +x = Silence FX.25 information. .RE .RE .PD .TP .BI "-t " "n" -Text colors. 1=normal, 0=disabled. +Text colors. 0=disabled. 1=default. 2,3,4,... alternatives. Use 9 to test compatibility with your terminal. + .TP .B "-p " Enable pseudo terminal for KISS protocol. .TP -.B "-x " +.BI "-x " Send Xmit level calibration tones. +.PD 0 +.RS +.RS +a = Alternating mark/space tones. +.P +m = steady Mark tone (e.g. 1200 Hz) +.P +s = steady Space tone (e.g. 2200 Hz) +.P +p = selence (set Ptt only). +.P +Optionally add a number to specify radio channel. +.RE +.RE +.PD .TP -.B "-U " +.B "-u " Print UTF-8 test string and exit. .TP @@ -129,24 +191,31 @@ Print Symbol tables and exit. .BI "-a " "n" Report audio device statistics each n seconds. +.TP +.BI "-T " "fmt" +Time stamp format for sent and received frames. + +.TP +.BI "-e " "ber" +Receive Bit Error Rate (BER), e.g. 1e-5 .SH EXAMPLES gqrx (2.3 and later) has the ability to send streaming audio through a UDP socket to another application for further processing. direwolf can listen over a UDP port with options like this: .RS .P -direwolf -n 1 -r 48000 -b 16 udp:7355 +direwolf \-n 1 \-r 48000 \-b 16 udp:7355 .RE .P Other SDR applications might produce audio on stdout so it is convenient to pipe into the next application. In this example, the final "-" means read from stdin. .RS .P -rtl_fm -f 144.39M -o 4 - | direwolf -n 1 -r 24000 -b 16 - +rtl_fm \-f 144.39M \-o 4 \- | direwolf \-n 1 \-r 24000 \-b 16 \- .RE .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, cm108, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/gen_packets.1 b/man/gen_packets.1 similarity index 51% rename from man1/gen_packets.1 rename to man/gen_packets.1 index f1782aac..740d4db4 100644 --- a/man1/gen_packets.1 +++ b/man/gen_packets.1 @@ -5,12 +5,12 @@ gen_packets \- Generate audio file for AX.25 frames. .SH SYNOPSIS -.B gen_packets -o +.B gen_packets \-o .I wav-file-out -[ \fIoptions\fR ] [ \fItext-file\fR | - ] +[ \fIoptions\fR ] [ \fItext-file\fR | \- ] .RS .P -\fIwav-file-out\fR is the result. The -o option is required. +\fIwav-file-out\fR is the result. The \-o option is required. .P \fItext-file\fR may contain AX.25 packets in the standard monitoring format. Use "-" to read from stdin. If not specified, a default builtin message will be used. .RE @@ -25,19 +25,57 @@ It is very flexible allowing a wide range of audio sample rates, data speeds, an .TP .BI "-a " "n" -Signal amplitude in range of 0 - 200%. Default 50. Note that 100% is corresponds to signal peaks of +/- 16383 so we have plenty of headroom to avoid saturation. +Signal amplitude in range of 0-200%. Default 50. Note that 100% is corresponds to signal peaks of +/- 16383 so we have plenty of headroom to avoid saturation. .TP .BI "-b " "n" Bits / second for data. Default is 1200. .TP -.BI "-B " "n" -Bits / second for data. Proper modem selected for 300, 1200, 9600. +.BI "-B " "n" +Data rate in bits/sec for first channel. Standard values are 300, 1200, 2400, 4800, 9600. +.PD 0 +.RS +.RS +300 bps defaults to AFSK tones of 1600 & 1800. +.P +1200 bps uses AFSK tones of 1200 & 2200. +.P +2400 bps uses QPSK based on V.26 standard. +.P +4800 bps uses 8PSK based on V.27 standard. +.P +9600 bps and up uses K9NG/G3RUH standard. +.P +EAS for Emergency Alert System (EAS) Specific Area Message Encoding (SAME). +.RE +.RE +.PD + +.TP +.BI "-g " +Force G3RUH modem regardless of data rate. + +.TP +.BI "-j " +2400 bps QPSK compatible with Dire Wolf <= 1.5. + +.TP +.BI "-J " +2400 bps QPSK compatible with MFJ-2400. + +.TP +.BI "-X " "n" +1 to enable FX.25 transmit. 16, 32, 64 for specific number of check bytes. + +.TP +.BI "-I " "n" +Enable IL2P transmit. n=1 is recommended. 0 uses weaker FEC. .TP -.BI "-g" -Scrambled baseband rather than AFSK. +.BI "-i " "n" +Enable IL2P transmit, inverted polarity. n=1 is recommended. 0 uses weaker FEC. + .TP .BI "-m " "n" @@ -64,13 +102,17 @@ Send output to .wav file. 8 bit audio rather than 16. .TP -.B "-2" +.BI "-2" 2 channels of audio rather than 1. +.TP +.BI "-v" "max[,incr]" +Variable speed with specified maximum error and optional increment. + .SH EXAMPLES .P -.B gen_packets -o x.wav +.B gen_packets \-o x.wav .P .RS With all defaults, a built-in test message is generated @@ -78,27 +120,27 @@ with standard Bell 202 tones used for packet radio on ordinary VHF FM transceivers. .RE .P -.B gen_packets -o x.wav -g -b 9600 +.B gen_packets \-o x.wav \-g \-b 9600 .PD 0 .P .PD -.B gen_packets -o x.wav -B 9600 +.B gen_packets \-o x.wav \-B 9600 .P .RS Both of these are equivalent. "-B 9600" automatically selects scrambled baseband rather than AFSK. .RE .P -.B gen_packets -o x.wav -m 1600 -s 1800 -b 300 +.B gen_packets \-o x.wav \-m 1600 \-s 1800 \-b 300 .PD 0 .P .PD -.B gen_packets -o x.wav -B 300 +.B gen_packets \-o x.wav \-B 300 .P .RS Both of these generate 200 Hz shift, 300 baud, suitable for HF SSB transceiver. .RE .P -.B echo -n 'WB2OSZ>WORLD:Hello, world!' | gen_packets -a 25 -o x.wav - +.B echo \-n 'WB2OSZ>WORLD:Hello, world!' | gen_packets \-a 25 \-o x.wav \- .PD 0 .P .PD @@ -112,5 +154,5 @@ Read message from stdin and put quarter volume sound into the file x.wav. Decod .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, cm108, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man/kissutil.1 b/man/kissutil.1 new file mode 100644 index 00000000..a7968f97 --- /dev/null +++ b/man/kissutil.1 @@ -0,0 +1,60 @@ +.TH KISSUTIL 1 + +.SH NAME +kissutil \- KISS TNC troubleshooting and Application Interface. + + +.SH SYNOPSIS +.B kissutil +[ \fIoptions\fR ] + + + +.SH DESCRIPTION +\fBkissutil\fR can be used interactively for troubleshooting a KISS TNC. +It is usable with direwolf and other generic KISS TNCs connected to a serial port. +It can also be used as an application interface where each side places files in a +directory for the other to process. +See User Guide for more details. + + +.SH OPTIONS +.TP +.BI "-h " "host" +Hostname or IP address for a TCP KISS TNC. Default is localhost. + +.TP +.BI "-p " "port" +A number may be specified for a TCP port other than the default 8001. +If not a number, it is considered to be a serial port name such as /dev/ttyS0 or COM3. + +.TP +.BI "-s " "speed" +Speed for serial port. e.g. 9600. + +.TP +.BI "-o " "rec-directory" +For each received frame, a new file is created here. +It is expected that some other application will process files in this directory then delete them. + +.TP +.BI "-T " "format" +Each received frame will be preceded by a timestamp in the specified format. +See strftime documentation for a description of the format string. +Example: %H:%M:%S for current time in hours, minutes, seconds. + +.TP +.BI "-f " "xmit-directory" +Files in this directory are transmitted and deleted. +Another application places a file here when it wants something to be transmitted. + +.TP +.BI "-v " +Verbose - Display the KISS frames going to and from the TNC. + + +.SH SEE ALSO +More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. + +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll + diff --git a/man1/ll2utm.1 b/man/ll2utm.1 similarity index 90% rename from man1/ll2utm.1 rename to man/ll2utm.1 index e29e0b6d..10f37ebc 100644 --- a/man1/ll2utm.1 +++ b/man/ll2utm.1 @@ -30,5 +30,5 @@ zone = 19T, easting = 307504, northing = 4721177 .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/log2gpx.1 b/man/log2gpx.1 similarity index 92% rename from man1/log2gpx.1 rename to man/log2gpx.1 index f94b56b9..094aa036 100644 --- a/man1/log2gpx.1 +++ b/man/log2gpx.1 @@ -36,5 +36,5 @@ None. .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/text2tt.1 b/man/text2tt.1 similarity index 95% rename from man1/text2tt.1 rename to man/text2tt.1 index 43a9a58a..2551d5ca 100644 --- a/man1/text2tt.1 +++ b/man/text2tt.1 @@ -60,5 +60,5 @@ Push buttons for two-key method: .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/tt2text.1 b/man/tt2text.1 similarity index 90% rename from man1/tt2text.1 rename to man/tt2text.1 index 84159c52..c6214c27 100644 --- a/man1/tt2text.1 +++ b/man/tt2text.1 @@ -11,7 +11,7 @@ tt2text \- Convert Touch Tone sequence to text .SH DESCRIPTION -\fBtt2text\fR converts a Touch Tone squence to text. There are two types +\fBtt2text\fR converts a Touch Tone sequence to text. There are two types of encoding: .RS .HP @@ -62,5 +62,5 @@ Decoded text from two-key method: .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/man1/utm2ll.1 b/man/utm2ll.1 similarity index 91% rename from man1/utm2ll.1 rename to man/utm2ll.1 index 6d420bfd..9bc3ef86 100644 --- a/man1/utm2ll.1 +++ b/man/utm2ll.1 @@ -36,5 +36,5 @@ latitude = 42.662139, longitude = -71.365553 .SH SEE ALSO More detailed information is in the pdf files in /usr/local/share/doc/direwolf, or possibly /usr/share/doc/direwolf, depending on installation location. -Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, ll2utm, log2gpx, text2tt, tt2text, utm2ll +Applications in this package: aclients, atest, decode_aprs, direwolf, gen_packets, kissutil, ll2utm, log2gpx, text2tt, tt2text, utm2ll diff --git a/misc/strsep.c b/misc/strsep.c deleted file mode 100644 index 7d764d00..00000000 --- a/misc/strsep.c +++ /dev/null @@ -1,22 +0,0 @@ -/* BSD strsep function */ - -/* Copyright 2002, Red Hat Inc. */ - -/* undef STRICT_ANSI so that strsep prototype will be defined */ -#undef __STRICT_ANSI__ -#include -//#include <_ansi.h> -//#include - -#define _DEFUN(name,arglist,args) name(args) -#define _AND , - -extern char *__strtok_r (char *, const char *, char **, int); - -char * -_DEFUN (strsep, (source_ptr, delim), - register char **source_ptr _AND - register const char *delim) -{ - return __strtok_r (*source_ptr, delim, source_ptr, 0); -} diff --git a/rdq.c b/rdq.c deleted file mode 100644 index 6a47cb83..00000000 --- a/rdq.c +++ /dev/null @@ -1,372 +0,0 @@ -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2011, 2012, 2015 John Langner, WB2OSZ -// -// 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, see . -// - - - -/*------------------------------------------------------------------ - * - * Module: rdq.c - * - * Purpose: Retry later decode queue for frames with bad FCS. - * - * Description: - * - *---------------------------------------------------------------*/ - -#include "direwolf.h" - -#include -#include -#include -#include -#include - -#include "ax25_pad.h" -#include "textcolor.h" -#include "audio.h" -#include "rdq.h" -#include "dedupe.h" - - - -static rrbb_t queue_head = NULL; /* Head of linked list for queue. */ -static int rdq_len = 0; -#define RDQ_UNDERRUN_THRESHOLD 30 /* A warning will be emitted if there are still this number of packets to decode in the queue and we try to add another one */ - - - -static dw_mutex_t rdq_mutex; /* Critical section for updating queues. */ - - -#if __WIN32__ - -static HANDLE wake_up_event; /* Notify try decode again thread when queue not empty. */ - -#else - -static pthread_cond_t wake_up_cond; /* Notify try decode again thread when queue not empty. */ - -static dw_mutex_t wake_up_mutex; /* Required by cond_wait. */ - -#endif - - -/*------------------------------------------------------------------- - * - * Name: rdq_init - * - * Purpose: Initialize the receive decode again queue. - * - * Inputs: None. Only single queue for all channels. - * - * Outputs: - * - * Description: Initialize the queue to be empty and set up other - * mechanisms for sharing it between different threads. - * - *--------------------------------------------------------------------*/ - - -void rdq_init (void) -{ - //int c, p; -#if __WIN32__ -#else - int err; -#endif - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_init ( )\n"); - dw_printf ("rdq_init: pthread_mutex_init...\n"); -#endif - - dw_mutex_init (&rdq_mutex); - -#if __WIN32__ -#else - dw_mutex_init (&wake_up_mutex); -#endif - - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_init: pthread_cond_init...\n"); -#endif - -#if __WIN32__ - - wake_up_event = CreateEvent (NULL, 0, 0, NULL); - - if (wake_up_event == NULL) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("rdq_init: pthread_cond_init: can't create decode wake up event"); - exit (1); - } - -#else - err = pthread_cond_init (&wake_up_cond, NULL); - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_init: pthread_cond_init returns %d\n", err); -#endif - - - if (err != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("rdq_init: pthread_cond_init err=%d", err); - perror (""); - exit (1); - } -#endif - - -} /* end rdq_init */ - - -/*------------------------------------------------------------------- - * - * Name: rdq_append - * - * Purpose: Add a packet to the end of the queue. - * - * Inputs: pp - Address of raw received bit buffer. - * Caller should NOT make any references to - * it after this point because it could - * be deleted at any time. - * - * Outputs: - * - * Description: Add buffer to end of linked list. - * Signal the decode thread if the queue was formerly empty. - * - *--------------------------------------------------------------------*/ - -void rdq_append (rrbb_t rrbb) -{ - //int was_empty; - rrbb_t plast; - rrbb_t pnext; -#ifndef __WIN32__ - int err; -#endif - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_append (rrbb=%p)\n", rrbb); - dw_printf ("rdq_append: enter critical section\n"); -#endif - - - dw_mutex_lock (&rdq_mutex); - - //was_empty = 1; - //if (queue_head != NULL) { - //was_empty = 0; - //} - if (queue_head == NULL) { - queue_head = rrbb; - } - else { - plast = queue_head; - while ((pnext = rrbb_get_nextp(plast)) != NULL) { - plast = pnext; - } - rrbb_set_nextp (plast, rrbb); - } - rdq_len++; - if (rdq_len > RDQ_UNDERRUN_THRESHOLD) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Too many packets to decode (%d) in the queue, decrease the FIX_BITS value\n", rdq_len); - } - - dw_mutex_unlock (&rdq_mutex); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_append: left critical section\n"); - dw_printf ("rdq_append (): about to wake up retry decode thread.\n"); -#endif - -#if __WIN32__ - SetEvent (wake_up_event); -#else - dw_mutex_lock (&wake_up_mutex); - - err = pthread_cond_signal (&wake_up_cond); - if (err != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("rdq_append: pthread_cond_signal err=%d", err); - perror (""); - exit (1); - } - - dw_mutex_unlock (&wake_up_mutex); -#endif - -} - - -/*------------------------------------------------------------------- - * - * Name: rdq_wait_while_empty - * - * Purpose: Sleep while the queue is empty rather than - * polling periodically. - * - * Inputs: None. - * - *--------------------------------------------------------------------*/ - - -void rdq_wait_while_empty (void) -{ - int is_empty; -#ifndef __WIN32__ - int err; -#endif - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty () : enter critical section\n"); -#endif - - dw_mutex_lock (&rdq_mutex); - -#if DEBUG - //text_color_set(DW_COLOR_DEBUG); - //dw_printf ("rdq_wait_while_empty (): after pthread_mutex_lock\n"); -#endif - is_empty = 1; - if (queue_head != NULL) - is_empty = 0; - - dw_mutex_unlock (&rdq_mutex); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty () : left critical section\n"); -#endif - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty (): is_empty = %d\n", is_empty); -#endif - - if (is_empty) { -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty (): SLEEP - about to call cond wait\n"); -#endif - - -#if __WIN32__ - WaitForSingleObject (wake_up_event, INFINITE); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty (): returned from wait\n"); -#endif - -#else - dw_mutex_lock (&wake_up_mutex); - - err = pthread_cond_wait (&wake_up_cond, &wake_up_mutex); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty (): WOKE UP - returned from cond wait, err = %d\n", err); -#endif - - if (err != 0) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("rdq_wait_while_empty: pthread_cond_wait err=%d", err); - perror (""); - exit (1); - } - - dw_mutex_unlock (&wake_up_mutex); - -#endif - } - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_wait_while_empty () returns (%d buffers remaining)\n", rdq_len); -#endif - -} - - -/*------------------------------------------------------------------- - * - * Name: rdq_remove - * - * Purpose: Remove raw bit buffer from the head of the queue. - * - * Inputs: none - * - * Returns: Pointer to rrbb object. - * Caller should destroy it with rrbb_delete when finished with it. - * - *--------------------------------------------------------------------*/ - -rrbb_t rdq_remove (void) -{ - - rrbb_t result_p; - - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_remove() enter critical section\n"); -#endif - - dw_mutex_lock (&rdq_mutex); - - rdq_len--; -#if DEBUG - dw_printf ("-rdq_len: %d\n", rdq_len); -#endif - if (queue_head == NULL) { - result_p = NULL; - } - else { - - result_p = queue_head; - queue_head = rrbb_get_nextp(result_p); - rrbb_set_nextp (result_p, NULL); - } - - dw_mutex_unlock (&rdq_mutex); - -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf ("rdq_remove() leave critical section, returns %p\n", result_p); -#endif - return (result_p); -} - -/* end rdq.c */ diff --git a/rdq.h b/rdq.h deleted file mode 100644 index b0e430c0..00000000 --- a/rdq.h +++ /dev/null @@ -1,28 +0,0 @@ - -/*------------------------------------------------------------------ - * - * Module: rdq.h - * - * Purpose: Retry decode queue - Hold raw received frames with errors - * for retrying the decoding later. - * - *---------------------------------------------------------------*/ - -#ifndef RDQ_H -#define RDQ_H 1 - -#include "rrbb.h" -//#include "audio.h" - -void rdq_init (void); - -void rdq_append (rrbb_t rrbb); - -void rdq_wait_while_empty (void); - -rrbb_t rdq_remove (void); - - -#endif - -/* end rdq.h */ diff --git a/rpm/direwolf.spec b/rpm/direwolf.spec new file mode 100644 index 00000000..d37bf5c1 --- /dev/null +++ b/rpm/direwolf.spec @@ -0,0 +1,175 @@ +%global shorttag 0d2c175c +Name: direwolf +Version: 1.6 +Release: 0.4.20200419git%{shorttag}%{?dist} +Summary: Sound Card-based AX.25 TNC + +License: GPLv2+ +URL: https://github.com/wb2osz/direwolf/ +Source0: https://github.com/wb2osz/direwolf/archive/%{version}/%{name}-%{version}.tar.gz +#Source0: https://github.com/wb2osz/direwolf/archive/%{version}/%{name}-%{shorttag}.tar.gz + +BuildRequires: gcc gcc-c++ +BuildRequires: cmake +BuildRequires: glibc-devel +BuildRequires: alsa-lib-devel +BuildRequires: gpsd-devel +BuildRequires: hamlib-devel +BuildRequires: systemd systemd-devel +Requires: ax25-tools ax25-apps +Requires(pre): shadow-utils + + +%description +Dire Wolf is a modern software replacement for the old 1980's style +TNC built with special hardware. Without any additional software, it +can perform as an APRS GPS Tracker, Digipeater, Internet Gateway +(IGate), APRStt gateway. It can also be used as a virtual TNC for +other applications such as APRSIS32, UI-View32, Xastir, APRS-TW, YAAC, +UISS, Linux AX25, SARTrack, Winlink Express, BPQ32, Outpost PM, and many +others. + + +%prep +%autosetup -n %{name}-%{version} + + +%build +%cmake -DUNITTEST=1 -DENABLE_GENERIC=1 . + + +%check +ctest -V %{?_smp_mflags} + + +%install +%make_install + +# Install service file +mkdir -p ${RPM_BUILD_ROOT}%{_unitdir} +cp %{_builddir}/%{buildsubdir}/systemd/%{name}.service ${RPM_BUILD_ROOT}%{_unitdir}/%{name}.service + +# Install service config file +mkdir -p ${RPM_BUILD_ROOT}%{_sysconfdir}/sysconfig +cp %{_builddir}/%{buildsubdir}/systemd/%{name}.sysconfig ${RPM_BUILD_ROOT}%{_sysconfdir}/sysconfig/%{name} + +# Install logrotate config file +mkdir -p ${RPM_BUILD_ROOT}%{_sysconfdir}/logrotate.d +cp %{_builddir}/%{buildsubdir}/systemd/%{name}.logrotate ${RPM_BUILD_ROOT}%{_sysconfdir}/logrotate.d/%{name} + +# copy config file +cp ${RPM_BUILD_ROOT}%{_pkgdocdir}/conf/%{name}.conf ${RPM_BUILD_ROOT}/%{_sysconfdir}/%{name}.conf + +# Make log directory +mkdir -m 0755 -p ${RPM_BUILD_ROOT}/var/log/%{name} + +# Move udev rules to system dir +mkdir -p ${RPM_BUILD_ROOT}%{_udevrulesdir} +mv ${RPM_BUILD_ROOT}%{_sysconfdir}/udev/rules.d/99-direwolf-cmedia.rules ${RPM_BUILD_ROOT}%{_udevrulesdir}/99-direwolf-cmedia.rules + +# Copy doc pngs +cp direwolf-block-diagram.png ${RPM_BUILD_ROOT}%{_pkgdocdir}/direwolf-block-diagram.png +cp tnc-test-cd-results.png ${RPM_BUILD_ROOT}%{_pkgdocdir}/tnc-test-cd-results.png + +# remove extraneous files +# This is not a desktop application, per the guidelines. Running it in a terminal +# does not make it a desktop application. +rm ${RPM_BUILD_ROOT}/usr/share/applications/direwolf.desktop +rm ${RPM_BUILD_ROOT}%{_datadir}/pixmaps/direwolf_icon.png +rm ${RPM_BUILD_ROOT}%{_pkgdocdir}/CHANGES.md +rm ${RPM_BUILD_ROOT}%{_pkgdocdir}/LICENSE +rm ${RPM_BUILD_ROOT}%{_pkgdocdir}/README.md + +# remove Windows external library directories +rm -r ${RPM_BUILD_ROOT}%{_pkgdocdir}/external + +# Move Telemetry Toolkit sample scripts into docs +mkdir -p ${RPM_BUILD_ROOT}%{_pkgdocdir}/telem/ +mv ${RPM_BUILD_ROOT}%{_bindir}/telem* ${RPM_BUILD_ROOT}%{_pkgdocdir}/telem/ +chmod 0644 ${RPM_BUILD_ROOT}%{_pkgdocdir}/telem/* + + +%package -n %{name}-doc +Summary: Documentation for Dire Wolf +BuildArch: noarch +Requires: %{name} = %{version}-%{release} + +%description -n %{name}-doc +Dire Wolf is a modern software replacement for the old 1980's style +TNC built with special hardware. Without any additional software, it +can perform as an APRS GPS Tracker, Digipeater, Internet Gateway +(IGate), APRStt gateway. It can also be used as a virtual TNC for +other applications such as APRSIS32, UI-View32, Xastir, APRS-TW, YAAC, +UISS, Linux AX25, SARTrack, RMS Express, BPQ32, Outpost PM, and many +others. + + +%files +%license LICENSE +%{_udevrulesdir}/99-direwolf-cmedia.rules +%{_bindir}/* +%{_mandir}/man1/* +%{_datadir}/%{name}/* +%dir %{_pkgdocdir} +%{_pkgdocdir}/conf/* +%{_pkgdocdir}/scripts/* +%{_pkgdocdir}/telem/* +%{_unitdir}/%{name}.service +%config(noreplace) %attr(0644,root,root) %{_sysconfdir}/sysconfig/%{name} +%config(noreplace) %attr(0644,root,root) %{_sysconfdir}/%{name}.conf +%config(noreplace) %attr(0644,root,root) %{_sysconfdir}/logrotate.d/%{name} +%dir %attr(0755, %{name}, %{name}) /var/log/%{name} + +%files -n %{name}-doc +%{_pkgdocdir}/*.pdf +%{_pkgdocdir}/*.png + +# At install, create a user in group audio (so can open sound card device files) +# and in group dialout (so can open serial device files) +%pre +getent group direwolf >/dev/null || groupadd -r direwolf +getent passwd direwolf >/dev/null || \ + useradd -r -g audio -G audio,dialout -d %{_datadir}/%{name} -s /sbin/nologin \ + -c "Direwolf Sound Card-based AX.25 TNC" direwolf +exit 0 + + +%changelog +* Mon Apr 20 2020 Matt Domsch - 1.6-0.3 +- drop unneeded BR libax25-devel + +* Mon Apr 20 2020 Matt Domsch - 1.6-0.2 +- write stdout/err to /var/log/direwolf, logrotate 30 days. +- run ctest +- remove CPU instruction tests, leave architecture choice up to the distro + +* Sun Apr 19 2020 Matt Domsch - 1.6-0.1 +- upstream 1.6 prerelease +- drop obsolete patches, use cmake +- add systemd startup, direwolf user + +* Tue Mar 31 2020 Richard Shaw - 1.5-6 +- Rebuild for hamlib 4. + +* Thu Feb 20 2020 Matt Domsch - 1.5-5 +- Remove unneeded dependency on python2-devel (#1805225) + +* Tue Jan 28 2020 Fedora Release Engineering - 1.5-4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_32_Mass_Rebuild + +* Wed Jul 24 2019 Fedora Release Engineering - 1.5-3 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_31_Mass_Rebuild + +* Wed Jul 03 2019 BjÃļrn Esser - 1.5-2 +- Rebuild (gpsd) + +* Sun Feb 17 2019 Matt Domsch - 1.5-1 +- Upgrade to released version 1.5 +- Apply upstream patch for newer gpsd API + +* Thu Jan 31 2019 Fedora Release Engineering - 1.5-0.2.beta4 +- Rebuilt for https://fedoraproject.org/wiki/Fedora_30_Mass_Rebuild + +* Mon Aug 27 2018 Matt Domsch - 1.5-0.1.beta4 +- Fedora Packaging Guidelines, based on spec by David Ranch + Moved Telemetry Toolkit examples into examples/ docs. diff --git a/scripts/CMakeLists.txt b/scripts/CMakeLists.txt new file mode 100644 index 00000000..886e5b17 --- /dev/null +++ b/scripts/CMakeLists.txt @@ -0,0 +1,6 @@ + +if(NOT (WIN32 OR CYGWIN)) + install(PROGRAMS "${CUSTOM_SCRIPTS_DIR}/dwespeak.sh" DESTINATION ${INSTALL_BIN_DIR}) + install(PROGRAMS "${CUSTOM_SCRIPTS_DIR}/dw-start.sh" DESTINATION ${INSTALL_SCRIPTS_DIR}) + add_subdirectory(telemetry-toolkit) +endif() diff --git a/dw-start.sh b/scripts/dw-start.sh similarity index 91% rename from dw-start.sh rename to scripts/dw-start.sh index 360b9c73..e0b06cda 100755 --- a/dw-start.sh +++ b/scripts/dw-start.sh @@ -1,4 +1,11 @@ -#!/bin/bash +#!/usr/bin/env bash + +# Why not simply "#!/bin/bash" ? + +# For OpenBSD, the bash location is /usr/local/bin/bash. +# By using env here, bash is found based on the user's $PATH. +# I hope this does not break some other operating system. + # Run this from crontab periodically to start up # Dire Wolf automatically. @@ -37,7 +44,8 @@ RUNMODE=AUTO DIREWOLF="direwolf" -#Direwolf start up command :: two examples where example one is enabled + +#Direwolf start up command :: Uncomment only one of the examples. # # 1. For normal operation as TNC, digipeater, IGate, etc. # Print audio statistics each 100 seconds for troubleshooting. @@ -45,9 +53,14 @@ DIREWOLF="direwolf" DWCMD="$DIREWOLF -a 100" +# 2. FX.25 Forward Error Correction (FEC) will allow your signal to +# go farther under poor radio conditions. Add "-X 1" to the command line. + +#DWCMD="$DIREWOLF -a 100 -X 1" + #--------------------------------------------------------------- # -# 2. Alternative for running with SDR receiver. +# 3. Alternative for running with SDR receiver. # Piping one application into another makes it a little more complicated. # We need to use bash for the | to be recognized. diff --git a/dwespeak.bat b/scripts/dwespeak.bat similarity index 100% rename from dwespeak.bat rename to scripts/dwespeak.bat diff --git a/dwespeak.sh b/scripts/dwespeak.sh old mode 100644 new mode 100755 similarity index 100% rename from dwespeak.sh rename to scripts/dwespeak.sh diff --git a/scripts/telemetry-toolkit/CMakeLists.txt b/scripts/telemetry-toolkit/CMakeLists.txt new file mode 100644 index 00000000..46f8e61c --- /dev/null +++ b/scripts/telemetry-toolkit/CMakeLists.txt @@ -0,0 +1,13 @@ +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-balloon.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-bits.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-data.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-data91.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-eqns.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-parm.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-seq.sh" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-unit.pl" DESTINATION ${INSTALL_BIN_DIR}) +install(PROGRAMS "${CUSTOM_TELEMETRY_DIR}/telem-volts.py" DESTINATION ${INSTALL_BIN_DIR}) + +install(FILES "${CUSTOM_TELEMETRY_DIR}/telem-m0xer-3.txt" DESTINATION ${INSTALL_CONF_DIR}) +install(FILES "${CUSTOM_TELEMETRY_DIR}/telem-balloon.conf" DESTINATION ${INSTALL_CONF_DIR}) +install(FILES "${CUSTOM_TELEMETRY_DIR}/telem-volts.conf" DESTINATION ${INSTALL_CONF_DIR}) diff --git a/telemetry-toolkit/telem-balloon.conf b/scripts/telemetry-toolkit/telem-balloon.conf similarity index 100% rename from telemetry-toolkit/telem-balloon.conf rename to scripts/telemetry-toolkit/telem-balloon.conf diff --git a/telemetry-toolkit/telem-balloon.pl b/scripts/telemetry-toolkit/telem-balloon.pl similarity index 100% rename from telemetry-toolkit/telem-balloon.pl rename to scripts/telemetry-toolkit/telem-balloon.pl diff --git a/telemetry-toolkit/telem-bits.pl b/scripts/telemetry-toolkit/telem-bits.pl similarity index 100% rename from telemetry-toolkit/telem-bits.pl rename to scripts/telemetry-toolkit/telem-bits.pl diff --git a/telemetry-toolkit/telem-data.pl b/scripts/telemetry-toolkit/telem-data.pl similarity index 100% rename from telemetry-toolkit/telem-data.pl rename to scripts/telemetry-toolkit/telem-data.pl diff --git a/telemetry-toolkit/telem-data91.pl b/scripts/telemetry-toolkit/telem-data91.pl similarity index 100% rename from telemetry-toolkit/telem-data91.pl rename to scripts/telemetry-toolkit/telem-data91.pl diff --git a/telemetry-toolkit/telem-eqns.pl b/scripts/telemetry-toolkit/telem-eqns.pl similarity index 100% rename from telemetry-toolkit/telem-eqns.pl rename to scripts/telemetry-toolkit/telem-eqns.pl diff --git a/telemetry-toolkit/telem-m0xer-3.txt b/scripts/telemetry-toolkit/telem-m0xer-3.txt similarity index 100% rename from telemetry-toolkit/telem-m0xer-3.txt rename to scripts/telemetry-toolkit/telem-m0xer-3.txt diff --git a/telemetry-toolkit/telem-parm.pl b/scripts/telemetry-toolkit/telem-parm.pl similarity index 100% rename from telemetry-toolkit/telem-parm.pl rename to scripts/telemetry-toolkit/telem-parm.pl diff --git a/telemetry-toolkit/telem-seq.sh b/scripts/telemetry-toolkit/telem-seq.sh similarity index 100% rename from telemetry-toolkit/telem-seq.sh rename to scripts/telemetry-toolkit/telem-seq.sh diff --git a/telemetry-toolkit/telem-unit.pl b/scripts/telemetry-toolkit/telem-unit.pl similarity index 100% rename from telemetry-toolkit/telem-unit.pl rename to scripts/telemetry-toolkit/telem-unit.pl diff --git a/telemetry-toolkit/telem-volts.conf b/scripts/telemetry-toolkit/telem-volts.conf similarity index 100% rename from telemetry-toolkit/telem-volts.conf rename to scripts/telemetry-toolkit/telem-volts.conf diff --git a/telemetry-toolkit/telem-volts.py b/scripts/telemetry-toolkit/telem-volts.py similarity index 95% rename from telemetry-toolkit/telem-volts.py rename to scripts/telemetry-toolkit/telem-volts.py index f5249683..34c59c4d 100644 --- a/telemetry-toolkit/telem-volts.py +++ b/scripts/telemetry-toolkit/telem-volts.py @@ -1,4 +1,4 @@ -#!/usr/bin/python +#!/usr/bin/python3 # Part of Dire Wolf APRS Telemetry Toolkit, WB2OSZ, 2015 @@ -33,4 +33,4 @@ # (multiply by expected value, divide by uncalibrated result.) #volts = volts * 4.98 / 4.889 -print "%.3f" % (volts) +print("%.3f" % (volts)) diff --git a/search_sdks.sh b/search_sdks.sh deleted file mode 100644 index 3ddac3af..00000000 --- a/search_sdks.sh +++ /dev/null @@ -1,109 +0,0 @@ -#!/bin/bash -# -# This file is part of Dire Wolf, an amateur radio packet TNC. -# -# Bash script to search for SDKs on various MacOSX versions. -# -# Copyright (C) 2015 Robert Stiles -# -# 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, see . -# - -FILENAME="./use_this_sdk" -selected_sdk="" -valid_flag=0 -system_sdk="" - -if [ -f $FILENAME ]; then - selected_sdk=`cat $FILENAME` - if [ -d $selected_sdk ]; then - valid_flag=1 - fi -fi - -if [ $valid_flag -eq "0" ]; then - echo " " >&2 - echo " " >&2 - echo "Searching for SDKs.... (Wait for results)" >&2 - echo " " >&2 - echo "Enter the number and press Enter/Return Key" >&2 - echo " " >&2 - echo " " >&2 - - prompt="Select SDK to use:" - - loc1=( $(find /Applications/XCode.app -type d -name "MacOSX10.*.sdk") ) - loc2=( $(find /Developer/SDKs -maxdepth 1 -type d -name "MacOSX10.*.sdk") ) - - options=("${loc1[@]}" "${loc2[@]}") - - if [ "${#options[@]}" -lt "2" ]; then - echo "$options" - fi - - PS3="$prompt " - select opt in "${options[@]}" "Do not use any SDK" ; do - if (( REPLY == 1 + ${#options[@]} )) ; then - echo " " - break - elif (( REPLY > 0 && REPLY <= ${#options[@]} )) ; then - selected_sdk="$opt" - break - fi - done - - if [ ! -z "$selected_sdk" ]; then - echo "$selected_sdk" > $FILENAME - else - echo " " > $FILENAME - fi -fi - -if [ ! -z "$selected_sdk" ]; then - temp_str="$selected_sdk" - min_str="" - flag=true - - # Search for the last MacOSX in the string. - while [ "${#temp_str}" -gt 4 ]; do - temp_str="${temp_str#*MacOSX}" - temp_str="${temp_str%%.sdk}" - min_str="$temp_str" - temp_str="${temp_str:1}" - done - - # Remove the "u" if 10.4u Universal SDK is used. - min_str="${min_str%%u}" - - system_sdk="-isystem ${selected_sdk} -mmacosx-version-min=${min_str}" -else - system_sdk=" " -fi - -echo " " >&2 -echo "*******************************************************************" >&2 - -if [ -z "${system_sdk}" ]; then - echo "SDK Selected: None" >&2 -else - echo "SDK Selected: ${system_sdk}" >&2 -fi - -echo "To change SDK version execute 'make clean' followed by 'make'." >&2 -echo "*******************************************************************" >&2 -echo " " >&2 - -echo ${system_sdk} - - diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt new file mode 100644 index 00000000..a2c3963d --- /dev/null +++ b/src/CMakeLists.txt @@ -0,0 +1,573 @@ + +# global includes +# not ideal but not so slow +# otherwise use target_include_directories +include_directories( + ${GPSD_INCLUDE_DIRS} + ${HAMLIB_INCLUDE_DIRS} + ${ALSA_INCLUDE_DIRS} + ${UDEV_INCLUDE_DIRS} + ${PORTAUDIO_INCLUDE_DIRS} + ${SNDIO_INCLUDE_DIRS} + ${CUSTOM_GEOTRANZ_DIR} + ${CUSTOM_HIDAPI_DIR} + ) + +if(WIN32 OR CYGWIN) + include_directories( + ${CUSTOM_REGEX_DIR} + ) +endif() + + +# direwolf +list(APPEND direwolf_SOURCES + direwolf.c + ais.c + aprs_tt.c + audio_stats.c + ax25_link.c + ax25_pad.c + ax25_pad2.c + beacon.c + config.c + decode_aprs.c + dedupe.c + demod_9600.c + demod_afsk.c + demod_psk.c + demod.c + digipeater.c + cdigipeater.c + dlq.c + dsp.c + dtime_now.c + dtmf.c + dwgps.c + dwsock.c + encode_aprs.c + encode_aprs.c + fcs_calc.c + fcs_calc.c + fx25_encode.c + fx25_extract.c + fx25_init.c + fx25_rec.c + fx25_send.c + fx25_auto.c + gen_tone.c + hdlc_rec.c + hdlc_rec2.c + hdlc_send.c + igate.c + il2p_codec.c + il2p_scramble.c + il2p_rec.c + il2p_payload.c + il2p_init.c + il2p_header.c + il2p_send.c + kiss_frame.c + kiss.c + kissserial.c + kissnet.c + latlong.c + latlong.c + log.c + morse.c + multi_modem.c + waypoint.c + serial_port.c + pfilter.c + ptt.c + recv.c + rrbb.c + server.c + symbols.c + telemetry.c + textcolor.c + tq.c + tt_text.c + tt_user.c + xid.c + xmit.c + dwgps.c + dwgpsnmea.c + dwgpsd.c + mheard.c + ) + +if(LINUX) + list(APPEND direwolf_SOURCES + audio.c + ) + if(UDEV_FOUND) + list(APPEND direwolf_SOURCES + cm108.c + ) + endif() + if(AVAHI_CLIENT_FOUND) + list(APPEND direwolf_SOURCES + dns_sd_common.c + dns_sd_avahi.c + ) + endif() + elseif(WIN32 OR CYGWIN) # windows + list(APPEND direwolf_SOURCES + audio_win.c + cm108.c + + # icon + # require plain gcc binary or link + #${CMAKE_SOURCE_DIR}/cmake/cpack/direwolf.rc + ) + list(REMOVE_ITEM direwolf_SOURCES + dwgpsd.c + ) + elseif(HAVE_SNDIO) + list(APPEND direwolf_SOURCES + audio.c + ) + else() # macOS freebsd + list(APPEND direwolf_SOURCES + audio_portaudio.c + ) + if(USE_MACOS_DNSSD) + list(APPEND direwolf_SOURCES + dns_sd_common.c + dns_sd_macos.c + ) + endif() +endif() + +add_executable(direwolf + ${direwolf_SOURCES} + ) + +target_link_libraries(direwolf + ${GEOTRANZ_LIBRARIES} + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + ${HIDAPI_LIBRARIES} + Threads::Threads + ${GPSD_LIBRARIES} + ${HAMLIB_LIBRARIES} + ${ALSA_LIBRARIES} + ${UDEV_LIBRARIES} + ${PORTAUDIO_LIBRARIES} + ${SNDIO_LIBRARIES} + ${AVAHI_LIBRARIES} + ) + +if(WIN32 OR CYGWIN) + set_target_properties(direwolf + PROPERTIES COMPILE_FLAGS "-DUSE_REGEX_STATIC" + ) + target_link_libraries(direwolf winmm ws2_32 setupapi) +endif() + +# decode_aprs +list(APPEND decode_aprs_SOURCES + decode_aprs.c + ais.c + kiss_frame.c + ax25_pad.c + dwgpsnmea.c + dwgps.c + dwgpsd.c + serial_port.c + symbols.c + textcolor.c + fcs_calc.c + latlong.c + log.c + telemetry.c + tt_text.c + ) + +if(WIN32 OR CYGWIN) + list(REMOVE_ITEM decode_aprs_SOURCES + dwgpsd.c + ) +endif() + +add_executable(decode_aprs + ${decode_aprs_SOURCES} + ) + +set_target_properties(decode_aprs + PROPERTIES COMPILE_FLAGS "-DDECAMAIN -DUSE_REGEX_STATIC" + ) + +target_link_libraries(decode_aprs + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + Threads::Threads + ${GPSD_LIBRARIES} + ) + + +# Convert between text and touch tone representation. +# text2tt +list(APPEND text2tt_SOURCES + tt_text.c + ) + +add_executable(text2tt + ${text2tt_SOURCES} + ) + +set_target_properties(text2tt + PROPERTIES COMPILE_FLAGS "-DENC_MAIN" + ) + +target_link_libraries(text2tt + ${MISC_LIBRARIES} + ) + +# tt2text +list(APPEND tt2text_SOURCES + tt_text.c + ) + +add_executable(tt2text + ${tt2text_SOURCES} + ) + +set_target_properties(tt2text + PROPERTIES COMPILE_FLAGS "-DDEC_MAIN" + ) + +target_link_libraries(tt2text + ${MISC_LIBRARIES} + ) + + +# Convert between Latitude/Longitude and UTM coordinates. +# ll2utm +list(APPEND ll2utm_SOURCES + ll2utm.c + textcolor.c + ) + +add_executable(ll2utm + ${ll2utm_SOURCES} + ) + +target_link_libraries(ll2utm + ${GEOTRANZ_LIBRARIES} + ${MISC_LIBRARIES} + ) + +# utm2ll +list(APPEND utm2ll_SOURCES + utm2ll.c + textcolor.c + ) + +add_executable(utm2ll + ${utm2ll_SOURCES} + ) + +target_link_libraries(utm2ll + ${GEOTRANZ_LIBRARIES} + ${MISC_LIBRARIES} + ) + + +# Convert from log file to GPX. +# log2gpx +list(APPEND log2gpx_SOURCES + log2gpx.c + textcolor.c + ) + +add_executable(log2gpx + ${log2gpx_SOURCES} + ) + +target_link_libraries(log2gpx + ${MISC_LIBRARIES} + ) + + +# Test application to generate sound. +# gen_packets +list(APPEND gen_packets_SOURCES + gen_packets.c + ax25_pad.c + ax25_pad2.c + fx25_encode.c + fx25_extract.c + fx25_init.c + fx25_send.c + hdlc_send.c + fcs_calc.c + gen_tone.c + il2p_codec.c + il2p_scramble.c + il2p_payload.c + il2p_init.c + il2p_header.c + il2p_send.c + morse.c + dtmf.c + textcolor.c + dsp.c + ) + +add_executable(gen_packets + ${gen_packets_SOURCES} + ) + +target_link_libraries(gen_packets + ${MISC_LIBRARIES} + ) + + +# Unit test for AFSK demodulator +# atest +list(APPEND atest_SOURCES + atest.c + ais.c + demod.c + demod_afsk.c + demod_psk.c + demod_9600.c + dsp.c + fx25_extract.c + fx25_encode.c + fx25_init.c + fx25_rec.c + hdlc_rec.c + hdlc_rec2.c + il2p_codec.c + il2p_scramble.c + il2p_rec.c + il2p_payload.c + il2p_init.c + il2p_header.c + multi_modem.c + rrbb.c + fcs_calc.c + ax25_pad.c + ax25_pad2.c + decode_aprs.c + dwgpsnmea.c + dwgps.c + dwgpsd.c + serial_port.c + telemetry.c + dtime_now.c + latlong.c + symbols.c + tt_text.c + textcolor.c + ) + +if(WIN32 OR CYGWIN) + list(REMOVE_ITEM atest_SOURCES + dwgpsd.c + ) +endif() + +add_executable(atest + ${atest_SOURCES} + ) + +target_link_libraries(atest + ${MISC_LIBRARIES} + ${GPSD_LIBRARIES} + ${REGEX_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + set_target_properties(atest + PROPERTIES COMPILE_FLAGS "-DUSE_REGEX_STATIC" + ) +endif() + + +# Multiple AGWPE network or serial port clients to test TNCs side by side. +# aclients +list(APPEND aclients_SOURCES + aclients.c + ax25_pad.c + fcs_calc.c + textcolor.c + ) + +add_executable(aclients + ${aclients_SOURCES} + ) + +target_link_libraries(aclients + ${MISC_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(aclients ws2_32) +endif() + + +# Talk to a KISS TNC. +# Note: kiss_frame.c has conditional compilation on KISSUTIL. +# kissutil +list(APPEND kissutil_SOURCES + kissutil.c + kiss_frame.c + ax25_pad.c + fcs_calc.c + textcolor.c + serial_port.c + dtime_now.c + dwsock.c + ) + +add_executable(kissutil + ${kissutil_SOURCES} + ) + +set_target_properties(kissutil + PROPERTIES COMPILE_FLAGS "-DKISSUTIL" + ) + +target_link_libraries(kissutil + ${MISC_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(kissutil ws2_32) +endif() + + +# TNC interoperability testing for AX.25 connected mode. +# tnctest +list(APPEND tnctest_SOURCES + tnctest.c + textcolor.c + dtime_now.c + serial_port.c + ) + +add_executable(tnctest + ${tnctest_SOURCES} + ) + +target_link_libraries(tnctest + ${MISC_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(tnctest ws2_32) +endif() + + +# List USB audio adapters than can use GPIO for PTT. +# Originally for Linux only (using udev). +# Version 1.7 adds it for Windows. Needs hidapi library. + +# cm108 +if(UDEV_FOUND OR WIN32 OR CYGWIN) + list(APPEND cm108_SOURCES + cm108.c + textcolor.c + ) + + add_executable(cm108 + ${cm108_SOURCES} + ) + + set_target_properties(cm108 + PROPERTIES COMPILE_FLAGS "-DCM108_MAIN" + ) + + target_link_libraries(cm108 + ${MISC_LIBRARIES} + ) + + if (LINUX) + target_link_libraries(cm108 + ${UDEV_LIBRARIES} + ) + endif() + + if (WIN32 OR CYGWIN) + target_link_libraries(cm108 + ${HIDAPI_LIBRARIES} + ws2_32 + setupapi + ) + endif() +endif() + + +# Touch Tone to Speech sample application. +# ttcalc +list(APPEND ttcalc_SOURCES + ttcalc.c + ax25_pad.c + fcs_calc.c + textcolor.c + ) + +add_executable(ttcalc + ${ttcalc_SOURCES} + ) + +target_link_libraries(ttcalc + ${MISC_LIBRARIES} + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(ttcalc ws2_32) +endif() + + +# Sample for packet radio server application. +# appserver +list(APPEND appserver_SOURCES + appserver.c + agwlib.c + dwsock.c + dtime_now.c + ax25_pad.c + fcs_calc.c + textcolor.c + ) + +add_executable(appserver + ${appserver_SOURCES} + ) + +target_link_libraries(appserver + ${MISC_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(appserver ws2_32) +endif() + + +install(TARGETS direwolf DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS decode_aprs DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS text2tt DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS tt2text DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS ll2utm DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS utm2ll DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS aclients DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS log2gpx DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS gen_packets DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS atest DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS ttcalc DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS kissutil DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS tnctest DESTINATION ${INSTALL_BIN_DIR}) +install(TARGETS appserver DESTINATION ${INSTALL_BIN_DIR}) +if(UDEV_FOUND OR WIN32 OR CYGWIN) + install(TARGETS cm108 DESTINATION ${INSTALL_BIN_DIR}) +endif() diff --git a/aclients.c b/src/aclients.c similarity index 96% rename from aclients.c rename to src/aclients.c index 28b7cf32..0d0c3e73 100644 --- a/aclients.c +++ b/src/aclients.c @@ -69,13 +69,14 @@ #include #include #include -#include +#include #endif #include #include #include #include +#include #include #include @@ -283,10 +284,10 @@ int main (int argc, char *argv[]) #if __WIN32__ if (isdigit(port[j][0])) { - client_th[j] = (HANDLE)_beginthreadex (NULL, 0, client_thread_net, (void *)j, 0, NULL); + client_th[j] = (HANDLE)_beginthreadex (NULL, 0, client_thread_net, (void *)(ptrdiff_t)j, 0, NULL); } else { - client_th[j] = (HANDLE)_beginthreadex (NULL, 0, client_thread_serial, (void *)j, 0, NULL); + client_th[j] = (HANDLE)_beginthreadex (NULL, 0, client_thread_serial, (void *)(ptrdiff_t)j, 0, NULL); } if (client_th[j] == NULL) { printf ("Internal error: Could not create client thread %d.\n", j); @@ -294,10 +295,10 @@ int main (int argc, char *argv[]) } #else if (isdigit(port[j][0])) { - e = pthread_create (&client_tid[j], NULL, client_thread_net, (void *)(long)j); + e = pthread_create (&client_tid[j], NULL, client_thread_net, (void *)(ptrdiff_t)j); } else { - e = pthread_create (&client_tid[j], NULL, client_thread_serial, (void *)(long)j); + e = pthread_create (&client_tid[j], NULL, client_thread_serial, (void *)(ptrdiff_t)j); } if (e != 0) { perror("Internal error: Could not create client thread."); @@ -394,7 +395,7 @@ static void * client_thread_net (void *arg) int use_chan = -1; - my_index = (int)(long)arg; + my_index = (int)(ptrdiff_t)arg; #if DEBUGx printf ("DEBUG: client_thread_net %d start, port = '%s'\n", my_index, port[my_index]); @@ -549,12 +550,7 @@ static void * client_thread_net (void *arg) mon_cmd.kind_lo = 'k'; -#if __WIN32__ - send (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd), 0); -#else - err = write (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); -#endif - + SOCK_SEND (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); /* * Print all of the monitored packets. @@ -563,14 +559,10 @@ static void * client_thread_net (void *arg) while (1) { int n; -#if __WIN32__ - n = recv (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd), 0); -#else - n = read (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); -#endif + n = SOCK_RECV (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); if (n != sizeof(mon_cmd)) { - printf ("Read error, client %d received %d command bytes.\n", my_index, n); + printf ("Read error, client %d received %d command bytes. Terminating.\n", my_index, n); exit (1); } @@ -581,11 +573,7 @@ static void * client_thread_net (void *arg) assert (mon_cmd.data_len >= 0 && mon_cmd.data_len < (int)(sizeof(data))); if (mon_cmd.data_len > 0) { -#if __WIN32__ - n = recv (server_sock, data, mon_cmd.data_len, 0); -#else - n = read (server_sock, data, mon_cmd.data_len); -#endif + n = SOCK_RECV (server_sock, data, mon_cmd.data_len); if (n != mon_cmd.data_len) { printf ("Read error, client %d received %d data bytes.\n", my_index, n); @@ -677,7 +665,7 @@ static unsigned __stdcall client_thread_serial (void *arg) static void * client_thread_serial (void *arg) #endif { - int my_index = (int)(long)arg; + int my_index = (int)(ptrdiff_t)arg; #if __WIN32__ diff --git a/src/agwlib.c b/src/agwlib.c new file mode 100644 index 00000000..2c03adaa --- /dev/null +++ b/src/agwlib.c @@ -0,0 +1,749 @@ + +// ****** PRELIMINARY - needs work ****** + +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// + + +/*------------------------------------------------------------------ + * + * Module: agwlib.c + * + * Purpose: Sample application Program Interface (API) to use network TNC with AGW protocol. + * + * Input: + * + * Outputs: + * + * Description: This file contains functions to attach to a TNC over a TCP socket and send + * commands to it. The current list includes some of the following: + * + * 'C' Connect, Start an AX.25 Connection + * 'v' Connect VIA, Start an AX.25 circuit thru digipeaters + * 'c' Connection with non-standard PID + * 'D' Send Connected Data + * 'd' Disconnect, Terminate an AX.25 Connection + * 'X' Register CallSign + * 'x' Unregister CallSign + * 'R' Request for version number. + * 'G' Ask about radio ports. + * 'g' Capabilities of a port. + * 'k' Ask to start receiving RAW AX25 frames. + * 'm' Ask to start receiving Monitor AX25 frames. + * 'V' Transmit UI data frame. + * 'H' Report recently heard stations. Not implemented yet in direwolf. + * 'K' Transmit raw AX.25 frame. + * 'y' Ask Outstanding frames waiting on a Port + * 'Y' How many frames waiting for transmit for a particular station + * + * + * The user supplied application must supply functions to handle or ignore + * messages that come from the TNC. Common examples: + * + * 'C' AX.25 Connection Received + * 'D' Connected AX.25 Data + * 'd' Disconnected + * 'R' Reply to Request for version number. + * 'G' Reply to Ask about radio ports. + * 'g' Reply to capabilities of a port. + * 'K' Received AX.25 frame in raw format. (Enabled with 'k' command.) + * 'U' Received AX.25 frame in monitor format. (Enabled with 'm' command.) + * 'y' Outstanding frames waiting on a Port + * 'Y' How many frames waiting for transmit for a particular station + * 'C' AX.25 Connection Received + * 'D' Connected AX.25 Data + * 'd' Disconnected + * + * + * + * References: AGWPE TCP/IP API Tutorial + * http://uz7ho.org.ua/includes/agwpeapi.htm + * + * Usage: See appclient.c and appserver.c for examples of how to use this. + * + *---------------------------------------------------------------*/ + + +#include "direwolf.h" // Sets _WIN32_WINNT for XP API level needed by ws2tcpip.h + +#if __WIN32__ +#include +#include // _WIN32_WINNT must be set to 0x0501 before including this +#else +#include +#include +#include +#include +#include +#include +#endif + +#include +#include +#include +#include +#include +#include + +#include "textcolor.h" +#include "dwsock.h" // socket helper functions. +#include "ax25_pad.h" // forAX25_MAX_PACKET_LEN +#include "agwlib.h" + + + + +/* + * Message header for AGW protocol. + * Multibyte numeric values require rearranging for big endian cpu. + */ + +/* + * With MinGW version 4.6, obviously x86. + * or Linux gcc version 4.9, Linux ARM. + * + * $ gcc -E -dM - < /dev/null | grep END + * #define __ORDER_LITTLE_ENDIAN__ 1234 + * #define __FLOAT_WORD_ORDER__ __ORDER_LITTLE_ENDIAN__ + * #define __ORDER_PDP_ENDIAN__ 3412 + * #define __ORDER_BIG_ENDIAN__ 4321 + * #define __BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__ + * + * This is for standard OpenWRT on MIPS. + * + * #define __ORDER_LITTLE_ENDIAN__ 1234 + * #define __FLOAT_WORD_ORDER__ __ORDER_BIG_ENDIAN__ + * #define __ORDER_PDP_ENDIAN__ 3412 + * #define __ORDER_BIG_ENDIAN__ 4321 + * #define __BYTE_ORDER__ __ORDER_BIG_ENDIAN__ + * + * This was reported for an old Mac with PowerPC processor. + * (Newer versions have x86.) + * + * $ gcc -E -dM - < /dev/null | grep END + * #define __BIG_ENDIAN__ 1 + * #define _BIG_ENDIAN 1 + */ + + +#if defined(__BIG_ENDIAN__) || (defined(__BYTE_ORDER__) && defined(__ORDER_BIG_ENDIAN__) && __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__) + +// gcc >= 4.2 has __builtin_swap32() but might not be compatible with older versions or other compilers. + +#define host2netle(x) ( (((x)>>24)&0x000000ff) | (((x)>>8)&0x0000ff00) | (((x)<<8)&0x00ff0000) | (((x)<<24)&0xff000000) ) +#define netle2host(x) ( (((x)>>24)&0x000000ff) | (((x)>>8)&0x0000ff00) | (((x)<<8)&0x00ff0000) | (((x)<<24)&0xff000000) ) + +#else + +#define host2netle(x) (x) +#define netle2host(x) (x) + +#endif + + +struct agw_hdr_s { /* Command header. */ + + unsigned char portx; /* 0 for first, 1 for second, etc. */ + /* Dire Wolf uses the term "channel" to avoid confusion with TCP ports */ + /* or other places port might be used. */ + unsigned char reserved1; + unsigned char reserved2; + unsigned char reserved3; + + unsigned char datakind; /* Message type, usually written as a letter. */ + unsigned char reserved4; + unsigned char pid; + unsigned char reserved5; + + char call_from[10]; + + char call_to[10]; + + int data_len_NETLE; /* Number of data bytes following. */ + /* _NETLE suffix is reminder to convert for network byte order. */ + + int user_reserved_NETLE; +}; + + +struct agw_cmd_s { /* Complete command with header and data. */ + + struct agw_hdr_s hdr; /* Command header. */ + char data[AX25_MAX_PACKET_LEN]; /* Possible variable length data. */ +}; + + + +/*------------------------------------------------------------------- + * + * Name: agwlib_init + * + * Purpose: Attach to TNC over TCP. + * + * Inputs: host - Host name or address. Often "localhost". + * + * port - TCP port number as text. Usually "8000". + * + * init_func - Call this function after establishing communication + * with the TNC. We put it here, so that it can be done + * again automatically if the TNC disappears and we + * reattach to it. + * It must return 0 for success. + * Can be NULL if not needed. + * + * Returns: 0 for success, -1 for failure. + * + * Description: This starts up a thread which listens to the socket and + * dispatches the messages to the corresponding callback functions. + * It will also attempt to re-establish communication with the + * TNC if it goes away. + * + *--------------------------------------------------------------------*/ + + +static char s_tnc_host[80]; +static char s_tnc_port[8]; +static int s_tnc_sock; // Socket handle or file descriptor. +static int (*s_tnc_init_func)(void); // Call after establishing socket. + + +// TODO: define macros somewhere to hide platform specifics. + +#if __WIN32__ +#define THREAD_F unsigned __stdcall +#else +#define THREAD_F void * +#endif + +#if __WIN32__ +static HANDLE tnc_listen_th; +static THREAD_F tnc_listen_thread (void *arg); +#else +static pthread_t tnc_listen_tid; +static THREAD_F tnc_listen_thread (void *arg); +#endif + + +int agwlib_init (char *host, char *port, int (*init_func)(void)) +{ + char tncaddr[DWSOCK_IPADDR_LEN]; + int e; + + strlcpy (s_tnc_host, host, sizeof(s_tnc_host)); + strlcpy (s_tnc_port, port, sizeof(s_tnc_port)); + s_tnc_sock = -1; + s_tnc_init_func = init_func; + + dwsock_init(); + + s_tnc_sock = dwsock_connect (host, port, "TNC", 0, 0, tncaddr); + + if (s_tnc_sock == -1) { + return (-1); + } + + +/* + * Incoming messages are dispatched to application-supplied callback functions. + * If the TNC disappears, try to reestablish communication. + */ + + +#if __WIN32__ + tnc_listen_th = (HANDLE)_beginthreadex (NULL, 0, tnc_listen_thread, (void *)NULL, 0, NULL); + if (tnc_listen_th == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Internal error: Could not create TNC listening thread\n"); + return (-1); + } +#else + e = pthread_create (&tnc_listen_tid, NULL, tnc_listen_thread, (void *)NULL); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + perror("Internal error: Could not create TNC listening thread"); + return (-1); + } +#endif + +// TNC initialization if specified. + + if (s_tnc_init_func != NULL) { + e = (*s_tnc_init_func)(); + return (e); + } + + return (0); +} + + +/*------------------------------------------------------------------- + * + * Name: tnc_listen_thread + * + * Purpose: Listen for anything from TNC and process it. + * Reconnect if something goes wrong and we got disconnected. + * + * Inputs: s_tnc_host + * s_tnc_port + * + * Outputs: s_tnc_sock - File descriptor for communicating with TNC. + * Will be -1 if not connected. + * + *--------------------------------------------------------------------*/ + +static void process_from_tnc (struct agw_cmd_s *cmd); + + +#if __WIN32__ +static unsigned __stdcall tnc_listen_thread (void *arg) +#else +static void * tnc_listen_thread (void *arg) +#endif +{ + char tncaddr[DWSOCK_IPADDR_LEN]; + + struct agw_cmd_s cmd; + + while (1) { + +/* + * Connect to TNC if not currently connected. + */ + + if (s_tnc_sock == -1) { + + text_color_set(DW_COLOR_ERROR); + // I'm using the term "attach" here, in an attempt to + // avoid confusion with the AX.25 connect. + dw_printf ("Attempting to reattach to network TNC...\n"); + + s_tnc_sock = dwsock_connect (s_tnc_host, s_tnc_port, "TNC", 0, 0, tncaddr); + + if (s_tnc_sock != -1) { + dw_printf ("Successfully reattached to network TNC.\n"); + + // Might need to run TNC initialization again. + // For example, a server would register its callsigns. + + if (s_tnc_init_func != NULL) { + int e = (*s_tnc_init_func)(); + (void) e; + } + + } + SLEEP_SEC(5); + } + else { + int n = SOCK_RECV (s_tnc_sock, (char *)(&cmd.hdr), sizeof(cmd.hdr)); + + if (n == -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Lost communication with network TNC. Will try to reattach.\n"); + dwsock_close (s_tnc_sock); + s_tnc_sock = -1; + continue; + } + else if (n != sizeof(cmd.hdr)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Error reading message header from network TNC.\n"); + dw_printf ("Tried to read %d bytes but got only %d.\n", (int)sizeof(cmd.hdr), n); + dw_printf ("Closing socket to TNC. Will try to reattach.\n"); + dwsock_close (s_tnc_sock); + s_tnc_sock = -1; + continue; + } + +/* + * Take some precautions to guard against bad data which could cause problems later. + */ + if (cmd.hdr.portx < 0 || cmd.hdr.portx >= MAX_CHANS) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid channel number, %d, in command '%c', from network TNC.\n", + cmd.hdr.portx, cmd.hdr.datakind); + cmd.hdr.portx = 0; // avoid subscript out of bounds, try to keep going. + } + +/* + * Call to/from fields are 10 bytes but contents must not exceed 9 characters. + * It's not guaranteed that unused bytes will contain 0 so we + * don't issue error message in this case. + */ + cmd.hdr.call_from[sizeof(cmd.hdr.call_from)-1] = '\0'; + cmd.hdr.call_to[sizeof(cmd.hdr.call_to)-1] = '\0'; + +/* + * Following data must fit in available buffer. + * Leave room for an extra nul byte terminator at end later. + */ + + int data_len = netle2host(cmd.hdr.data_len_NETLE); + + if (data_len < 0 || data_len > (int)(sizeof(cmd.data) - 1)) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid message from network TNC.\n"); + dw_printf ("Data Length of %d is out of range.\n", data_len); + + /* This is a bad situation. */ + /* If we tried to read again, the header probably won't be there. */ + /* No point in trying to continue reading. */ + + dw_printf ("Closing connection to TNC.\n"); + dwsock_close (s_tnc_sock); + s_tnc_sock = -1; + continue; + } + + cmd.data[0] = '\0'; + + if (data_len > 0) { + n = SOCK_RECV (s_tnc_sock, cmd.data, data_len); + if (n != data_len) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Error getting message data from network TNC.\n"); + dw_printf ("Tried to read %d bytes but got only %d.\n", data_len, n); + dw_printf ("Closing socket to network TNC.\n\n"); + dwsock_close (s_tnc_sock); + s_tnc_sock = -1; + continue; + } + if (n >= 0) { + cmd.data[n] = '\0'; // Terminate so it can be used as a C string. + } + + process_from_tnc (&cmd); + + } // additional data after command header + } // s_tnc_sock != -1 + } // while (1) + + return (0); // unreachable but shutup warning. + +} // end tnc_listen_thread + + +/* + * The user supplied application must supply functions to handle or ignore + * messages that come from the TNC. + */ + +static void process_from_tnc (struct agw_cmd_s *cmd) +{ + int data_len = netle2host(cmd->hdr.data_len_NETLE); + //int session; + + + switch (cmd->hdr.datakind) { + + case 'C': // AX.25 Connection Received + { + //agw_cb_C_connection_received (cmd->hdr.portx, cmd->hdr.call_from, cmd->hdr.call_to, data_len, cmd->data); + // TODO: compute session id + // There are two different cases to consider here. + if (strncmp(cmd->data, "*** CONNECTED To Station", 24) == 0) { + // Incoming: Other station initiated the connect request. + on_C_connection_received (cmd->hdr.portx, cmd->hdr.call_from, cmd->hdr.call_to, 1, cmd->data); + } + else if (strncmp(cmd->data, "*** CONNECTED With Station", 26) == 0) { + // Outgoing: Other station accepted my connect request. + on_C_connection_received (cmd->hdr.portx, cmd->hdr.call_from, cmd->hdr.call_to, 0, cmd->data); + } + else { +// TBD + } + } + break; + + case 'D': // Connected AX.25 Data + // FIXME: should probably add pid here. + agw_cb_D_connected_data (cmd->hdr.portx, cmd->hdr.call_from, cmd->hdr.call_to, data_len, cmd->data); + break; + + case 'd': // Disconnected + agw_cb_d_disconnected (cmd->hdr.portx, cmd->hdr.call_from, cmd->hdr.call_to, data_len, cmd->data); + break; + + case 'R': // Reply to Request for version number. + break; + + case 'G': // Port Information. + // Data part should be fields separated by semicolon. + // First field is number of ports (we call them channels). + // Other fields are of the form "Port99 comment" where first is number 1. + { + int num_chan = 1; // FIXME: FIXME: actually parse it. + char *chans[20]; + chans[0] = "Port1 blah blah"; + chans[1] = "Port2 blah blah"; + agw_cb_G_port_information (num_chan, chans); + } + break; + +// TODO: Maybe fill in more someday. + + case 'g': // Reply to capabilities of a port. + break; + case 'K': // Received AX.25 frame in raw format. (Enabled with 'k' command.) + break; + case 'U': // Received AX.25 frame in monitor format. (Enabled with 'm' command.) + break; + case 'y': // Outstanding frames waiting on a Port + break; + + case 'Y': // How many frames waiting for transmit for a particular station + { + int *p = (int*)(cmd->data); + int frame_count = netle2host(*p); + agw_cb_Y_outstanding_frames_for_station (cmd->hdr.portx, cmd->hdr.call_from, cmd->hdr.call_to, frame_count); + } + break; + + default: + break; + } + +} // end process_from_tnc + + + +/*------------------------------------------------------------------- + * + * Name: agwlib_X_register_callsign + * + * Purpose: Tell TNC to accept incoming connect requests to given callsign. + * + * Inputs: chan - Radio channel number, first is 0. + * + * call_from - My callsign or alias. + * + * Returns: Number of bytes sent for success, -1 for error. + * + *--------------------------------------------------------------------*/ + +int agwlib_X_register_callsign (int chan, char *call_from) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.portx = chan; + cmd.hdr.datakind = 'X'; + strlcpy (cmd.hdr.call_from, call_from, sizeof(cmd.hdr.call_from)); + return (SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE))); +} + + +/*------------------------------------------------------------------- + * + * Name: agwlib_x_unregister_callsign + * + * Purpose: Tell TNC to stop accepting incoming connect requests to given callsign. + * + * Inputs: chan - Radio channel number, first is 0. + * + * call_from - My callsign or alias. + * + * Returns: Number of bytes sent for success, -1 for error. + * + * FIXME: question do we need channel here? + * + *--------------------------------------------------------------------*/ + +int agwlib_x_unregister_callsign (int chan, char *call_from) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.portx = chan; + cmd.hdr.datakind = 'x'; + strlcpy (cmd.hdr.call_from, call_from, sizeof(cmd.hdr.call_from)); + return (SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE))); +} + + +/*------------------------------------------------------------------- + * + * Name: agwlib_G_ask_port_information + * + * Purpose: Tell TNC to stop accepting incoming connect requests to given callsign. + * + * Inputs: call_from - My callsign or alias. + * + * Returns: 0 for success, -1 for error. TODO: all like this. + * + *--------------------------------------------------------------------*/ + +int agwlib_G_ask_port_information (void) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.datakind = 'G'; + int n = SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE)); + return (n > 0 ? 0 : -1); +} + + + +/*------------------------------------------------------------------- + * + * Name: agwlib_C_connect + * + * Purpose: Tell TNC to start sequence for connecting to remote station. + * + * Inputs: chan - Radio channel number, first is 0. + * + * call_from - My callsign. + * + * call_to - Callsign (or alias) of remote station. + * + * Returns: Number of bytes sent for success, -1 for error. + * + * Description: This only starts the sequence and does not wait. + * Success or failure will be indicated sometime later by ? + * + *--------------------------------------------------------------------*/ + +int agwlib_C_connect (int chan, char *call_from, char *call_to) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.portx = chan; + cmd.hdr.datakind = 'C'; + cmd.hdr.pid = 0xF0; // Shouldn't matter because this appears + // only in Information frame, not connect sequence. + strlcpy (cmd.hdr.call_from, call_from, sizeof(cmd.hdr.call_from)); + strlcpy (cmd.hdr.call_to, call_to, sizeof(cmd.hdr.call_to)); + return (SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE))); +} + + + +/*------------------------------------------------------------------- + * + * Name: agwlib_d_disconnect + * + * Purpose: Tell TNC to disconnect from remote station. + * + * Inputs: chan - Radio channel number, first is 0. + * + * call_from - My callsign. + * + * call_to - Callsign (or alias) of remote station. + * + * Returns: Number of bytes sent for success, -1 for error. + * + * Description: This only starts the sequence and does not wait. + * Success or failure will be indicated sometime later by ? + * + *--------------------------------------------------------------------*/ + +int agwlib_d_disconnect (int chan, char *call_from, char *call_to) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.portx = chan; + cmd.hdr.datakind = 'd'; + strlcpy (cmd.hdr.call_from, call_from, sizeof(cmd.hdr.call_from)); + strlcpy (cmd.hdr.call_to, call_to, sizeof(cmd.hdr.call_to)); + return (SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE))); +} + + + +/*------------------------------------------------------------------- + * + * Name: agwlib_D_send_connected_data + * + * Purpose: Send connected data to remote station. + * + * Inputs: chan - Radio channel number, first is 0. + * + * pid - Protocol ID. Normally 0xFo for Ax.25. + * + * call_from - My callsign. + * + * call_to - Callsign (or alias) of remote station. + * + * data_len - Number of bytes for Information part. + * + * data - Content for Information part. + * + * Returns: Number of bytes sent for success, -1 for error. + * + * Description: This should only be done when we are known to have + * an established link to other station. + * + *--------------------------------------------------------------------*/ + +int agwlib_D_send_connected_data (int chan, int pid, char *call_from, char *call_to, int data_len, char *data) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.portx = chan; + cmd.hdr.datakind = 'D'; + cmd.hdr.pid = pid; // Normally 0xF0 but other special cases are possible. + strlcpy (cmd.hdr.call_from, call_from, sizeof(cmd.hdr.call_from)); + strlcpy (cmd.hdr.call_to, call_to, sizeof(cmd.hdr.call_to)); + cmd.hdr.data_len_NETLE = host2netle(data_len); + +// FIXME: DANGER possible buffer overflow, Need checking. + + assert (data_len <= sizeof(cmd.data)); + + memcpy (cmd.data, data, data_len); + return (SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE))); +} + + +/*------------------------------------------------------------------- + * + * Name: agwlib_Y_outstanding_frames_for_station + * + * Purpose: Ask how many frames remain to be sent to station on other end of link. + * + * Inputs: chan - Radio channel number, first is 0. + * + * call_from - My call [ or is it Station which initiated the link? (sent SABM/SABME) ] + * + * call_to - Remote station call [ or is it Station which accepted the link? ] + * + * Returns: Number of bytes sent for success, -1 for error. + * + * Description: We expect to get a 'Y' frame response shortly. + * + * This would be useful for a couple different purposes. + * + * When sending bulk data, we want to keep a fair amount queued up to take + * advantage of large window sizes (MAXFRAME, EMAXFRAME). On the other + * hand we don't want to get TOO far ahead when transferring a large file. + * + * Before disconnecting from another station, it would be good to know + * that it actually received the last message we sent. For this reason, + * I think it would be good for this to include frames that were + * transmitted but not yet acknowledged. (Even if it was transmitted once, + * it could still be transmitted again, if lost, so you could say it is + * still waiting for transmission.) + * + * See server.c for a more precise definition of exactly how this is defined. + * + *--------------------------------------------------------------------*/ + +int agwlib_Y_outstanding_frames_for_station (int chan, char *call_from, char *call_to) +{ + struct agw_cmd_s cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + cmd.hdr.portx = chan; + cmd.hdr.datakind = 'Y'; + strlcpy (cmd.hdr.call_from, call_from, sizeof(cmd.hdr.call_from)); + strlcpy (cmd.hdr.call_to, call_to, sizeof(cmd.hdr.call_to)); + return (SOCK_SEND(s_tnc_sock, (char*)(&cmd), sizeof(cmd.hdr) + netle2host(cmd.hdr.data_len_NETLE))); +} + + + +/* end agwlib.c */ diff --git a/src/agwlib.h b/src/agwlib.h new file mode 100644 index 00000000..688f9184 --- /dev/null +++ b/src/agwlib.h @@ -0,0 +1,45 @@ + +#ifndef AGWLIB_H +#define AGWLIB_H 1 + + +// Call at beginning to start it up. + +int agwlib_init (char *host, char *port, int (*init_func)(void)); + + + +// Send commands to TNC. + + +int agwlib_X_register_callsign (int chan, char *call_from); + +int agwlib_x_unregister_callsign (int chan, char *call_from); + +int agwlib_G_ask_port_information (void); + +int agwlib_C_connect (int chan, char *call_from, char *call_to); + +int agwlib_d_disconnect (int chan, char *call_from, char *call_to); + +int agwlib_D_send_connected_data (int chan, int pid, char *call_from, char *call_to, int data_len, char *data); + +int agwlib_Y_outstanding_frames_for_station (int chan, char *call_from, char *call_to); + + + +// The application must define these. + +void agw_cb_C_connection_received (int chan, char *call_from, char *call_to, int data_len, char *data); +void on_C_connection_received (int chan, char *call_from, char *call_to, int incoming, char *data); + +void agw_cb_d_disconnected (int chan, char *call_from, char *call_to, int data_len, char *data); + +void agw_cb_D_connected_data (int chan, char *call_from, char *call_to, int data_len, char *data); + +void agw_cb_G_port_information (int num_chan, char *chan_descriptions[]); + +void agw_cb_Y_outstanding_frames_for_station (int chan, char *call_from, char *call_to, int frame_count); + + +#endif \ No newline at end of file diff --git a/src/ais.c b/src/ais.c new file mode 100644 index 00000000..938fa012 --- /dev/null +++ b/src/ais.c @@ -0,0 +1,710 @@ + +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2020 John Langner, WB2OSZ +// +// 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, see . +// + + +/******************************************************************************** + * + * File: ais.c + * + * Purpose: Functions for processing received AIS transmissions and + * converting to NMEA sentence representation. + * + * References: AIVDM/AIVDO protocol decoding by Eric S. Raymond + * https://gpsd.gitlab.io/gpsd/AIVDM.html + * + * Sample recording with about 100 messages. Test with "atest -B AIS xxx.wav" + * https://github.com/freerange/ais-on-sdr/wiki/example-data/long-beach-160-messages.wav + * + * Useful on-line decoder for AIS NMEA sentences. + * https://www.aggsoft.com/ais-decoder.htm + * + * Future? Add an interface to feed AIS data into aprs.fi. + * https://aprs.fi/page/ais_feeding + * + *******************************************************************************/ + +#include "direwolf.h" + +#include +#include +#include +#include +#include + +#include "textcolor.h" +#include "ais.h" + +// Lengths, in bits, for the AIS message types. + +#define NUM_TYPES 27 +static const struct { + short min; + short max; +} valid_len[NUM_TYPES+1] = { + { -1, -1 }, // 0 not used + { 168, 168 }, // 1 + { 168, 168 }, // 2 + { 168, 168 }, // 3 + { 168, 168 }, // 4 + { 424, 424 }, // 5 + { 72, 1008 }, // 6 multipurpose + { 72, 168 }, // 7 increments of 32 bits + { 168, 1008 }, // 8 multipurpose + { 168, 168 }, // 9 + { 72, 72 }, // 10 + { 168, 168 }, // 11 + { 72, 1008 }, // 12 + { 72, 168 }, // 13 increments of 32 bits + { 40, 1008 }, // 14 + { 88, 160 }, // 15 + { 96, 114 }, // 16 96 or 114, not range + { 80, 816 }, // 17 + { 168, 168 }, // 18 + { 312, 312 }, // 19 + { 72, 160 }, // 20 + { 272, 360 }, // 21 + { 168, 168 }, // 22 + { 160, 160 }, // 23 + { 160, 168 }, // 24 + { 40, 168 }, // 25 + { 60, 1064 }, // 26 + { 96, 168 } // 27 96 or 168, not range +}; + +static void save_ship_data(char *mssi, char *shipname, char *callsign, char *destination); +static void get_ship_data(char *mssi, char *comment, int comment_size); + + +/*------------------------------------------------------------------- + * + * Functions to get and set element of a bit vector. + * + *--------------------------------------------------------------------*/ + +static const unsigned char mask[8] = { 0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01 }; + +static inline unsigned int get_bit (unsigned char *base, unsigned int offset) +{ + return ( (base[offset >> 3] & mask[offset & 0x7]) != 0); +} + +static inline void set_bit (unsigned char *base, unsigned int offset, int val) +{ + if (val) { + base[offset >> 3] |= mask[offset & 0x7]; + } + else { + base[offset >> 3] &= ~ mask[offset & 0x7]; + } +} + + +/*------------------------------------------------------------------- + * + * Extract a variable length field from a bit vector. + * + *--------------------------------------------------------------------*/ + +static unsigned int get_field (unsigned char *base, unsigned int start, unsigned int len) +{ + unsigned int result = 0; + for (int k = 0; k < len; k++) { + result <<= 1; + result |= get_bit (base, start + k); + } + return (result); +} + +static void set_field (unsigned char *base, unsigned int start, unsigned int len, unsigned int val) +{ + for (int k = 0; k < len; k++) { + set_bit (base, start + k, (val >> (len - 1 - k) ) & 1); + } +} + + +static int get_field_signed (unsigned char *base, unsigned int start, unsigned int len) +{ + int result = (int) get_field(base, start, len); + // Sign extend. + result <<= (32 - len); + result >>= (32 - len); + return (result); +} + +static double get_field_lat (unsigned char *base, unsigned int start, unsigned int len) +{ + // Latitude of 0x3412140 (91 deg) means not available. + // Message type 27 uses lower resolution, 17 bits rather than 27. + // It encodes minutes/10 rather than normal minutes/10000. + + int n = get_field_signed(base, start, len); + if (len == 17) { + return ((n == 91*600) ? G_UNKNOWN : (double)n / 600.0); + } + else { + return ((n == 91*600000) ? G_UNKNOWN : (double)n / 600000.0); + } +} + +static double get_field_lon (unsigned char *base, unsigned int start, unsigned int len) +{ + // Longitude of 0x6791AC0 (181 deg) means not available. + // Message type 27 uses lower resolution, 18 bits rather than 28. + // It encodes minutes/10 rather than normal minutes/10000. + + int n = get_field_signed(base, start, len); + if (len == 18) { + return ((n == 181*600) ? G_UNKNOWN : (double)n / 600.0); + } + else { + return ((n == 181*600000) ? G_UNKNOWN : (double)n / 600000.0); + } +} + +static float get_field_speed (unsigned char *base, unsigned int start, unsigned int len) +{ + // Raw 1023 means not available. + // Multiply by 0.1 to get knots. + // For aircraft it is knots, not deciknots. + + // Message type 27 uses lower resolution, 6 bits rather than 10. + // It encodes minutes/10 rather than normal minutes/10000. + + int n = get_field(base, start, len); + if (len == 6) { + return ((n == 63) ? G_UNKNOWN : (float)n); + } + else { + return ((n == 1023) ? G_UNKNOWN : (float)n * 0.1); + } +} + +static float get_field_course (unsigned char *base, unsigned int start, unsigned int len) +{ + // Raw 3600 means not available. + // Multiply by 0.1 to get degrees + // Message type 27 uses lower resolution, 9 bits rather than 12. + // It encodes degrees rather than normal degrees/10. + + int n = get_field(base, start, len); + if (len == 9) { + return ((n == 360) ? G_UNKNOWN : (float)n); + } + else { + return ((n == 3600) ? G_UNKNOWN : (float)n * 0.1); + } +} + +static int get_field_ascii (unsigned char *base, unsigned int start, unsigned int len) +{ + assert (len == 6); + int ch = get_field(base, start, len); + if (ch < 32) ch += 64; + return (ch); +} + +static void get_field_string (unsigned char *base, unsigned int start, unsigned int len, char *result) +{ + assert (len % 6 == 0); + int nc = len / 6; // Number of characters. + // Caller better provide space for at least this +1. + // No bounds checking here. + for (int i = 0; i < nc; i++) { + result[i] = get_field_ascii (base, start + i * 6, 6); + } + result[nc] = '\0'; + // Officially it should be terminated/padded with @ but we also see trailing spaces. + char *p = strchr(result, '@'); + if (p != NULL) *p = '\0'; + for (int k = strlen(result) - 1; k >= 0 && result[k] == ' '; k--) { + result[k] = '\0'; + } +} + + + +/*------------------------------------------------------------------- + * + * Convert between 6 bit values and printable characters used in + * in the AIS NMEA sentences. + * + *--------------------------------------------------------------------*/ + +// Characters '0' thru 'W' become values 0 thru 39. +// Characters '`' thru 'w' become values 40 thru 63. + +static int char_to_sextet (char ch) +{ + if (ch >= '0' && ch <= 'W') { + return (ch - '0'); + } + else if (ch >= '`' && ch <= 'w') { + return (ch - '`' + 40); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid character \"%c\" found in AIS NMEA sentence payload.\n", ch); + return (0); + } +} + + +// Values 0 thru 39 become characters '0' thru 'W'. +// Values 40 thru 63 become characters '`' thru 'w'. +// This is known as "Payload Armoring." + +static int sextet_to_char (int val) +{ + if (val >= 0 && val <= 39) { + return ('0' + val); + } + else if (val >= 40 && val <= 63) { + return ('`' + val - 40); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid 6 bit value %d from AIS HDLC payload.\n", val); + return ('0'); + } +} + + +/*------------------------------------------------------------------- + * + * Convert AIS binary block (from HDLC frame) to NMEA sentence. + * + * In: Pointer to AIS binary block and number of bytes. + * Out: NMEA sentence. Provide size to avoid string overflow. + * + *--------------------------------------------------------------------*/ + +void ais_to_nmea (unsigned char *ais, int ais_len, char *nmea, int nmea_size) +{ + char payload[256]; + // Number of resulting characters for payload. + int ns = (ais_len * 8 + 5) / 6; + if (ns+1 > sizeof(payload)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("AIS HDLC payload of %d bytes is too large.\n", ais_len); + ns = sizeof(payload) - 1; + } + for (int k = 0; k < ns; k++) { + payload[k] = sextet_to_char(get_field(ais, k*6, 6)); + } + payload[ns] = '\0'; + + strlcpy (nmea, "!AIVDM,1,1,,A,", nmea_size); + strlcat (nmea, payload, nmea_size); + + // If the number of bytes in is not a multiple of 3, this does not + // produce a whole number of characters out. Extra padding bits were + // added to get the last character. Include this number so the + // decoding application can drop this number of bits from the end. + // At least, I think that is the way it should work. + // The examples all have 0. + char pad_bits[8]; + snprintf (pad_bits, sizeof(pad_bits), ",%d", ns * 6 - ais_len * 8); + strlcat (nmea, pad_bits, nmea_size); + + // Finally the NMEA style checksum. + int cs = 0; + for (char *p = nmea + 1; *p != '\0'; p++) { + cs ^= *p; + } + char checksum[8]; + snprintf (checksum, sizeof(checksum), "*%02X", cs & 0x7f); + strlcat (nmea, checksum, nmea_size); +} + + +/*------------------------------------------------------------------- + * + * Name: ais_parse + * + * Purpose: Parse AIS sentence and extract interesting parts. + * + * Inputs: sentence NMEA sentence. + * + * quiet Suppress printing of error messages. + * + * Outputs: descr Description of AIS message type. + * mssi 9 digit identifier. + * odlat latitude. + * odlon longitude. + * ofknots speed, knots. + * ofcourse direction of travel. + * ofalt_m altitude, meters. + * symtab APRS symbol table. + * symbol APRS symbol code. + * + * Returns: 0 for success, -1 for error. + * + *--------------------------------------------------------------------*/ + +// Maximum NMEA sentence length is 82, including CR/LF. +// Make buffer considerably larger to be safe. +#define NMEA_MAX_LEN 240 + +int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mssi, int mssi_size, double *odlat, double *odlon, + float *ofknots, float *ofcourse, float *ofalt_m, char *symtab, char *symbol, char *comment, int comment_size) +{ + char stemp[NMEA_MAX_LEN]; /* Make copy because parsing is destructive. */ + + strlcpy (mssi, "?", mssi_size); + *odlat = G_UNKNOWN; + *odlon = G_UNKNOWN; + *ofknots = G_UNKNOWN; + *ofcourse = G_UNKNOWN; + *ofalt_m = G_UNKNOWN; + + strlcpy (stemp, sentence, sizeof(stemp)); + +// Verify and remove checksum. + + unsigned char cs = 0; + char *p; + + for (p = stemp+1; *p != '*' && *p != '\0'; p++) { + cs ^= *p; + } + + p = strchr (stemp, '*'); + if (p == NULL) { + if ( ! quiet) { + text_color_set (DW_COLOR_INFO); + dw_printf("Missing AIS sentence checksum.\n"); + } + return (-1); + } + if (cs != strtoul(p+1, NULL, 16)) { + if ( ! quiet) { + text_color_set (DW_COLOR_ERROR); + dw_printf("AIS sentence checksum error. Expected %02x but found %s.\n", cs, p+1); + } + return (-1); + } + *p = '\0'; // Remove the checksum. + +// Extract the comma separated fields. + + char *next; + + char *talker; /* Expecting !AIVDM */ + char *frag_count; /* ignored */ + char *frag_num; /* ignored */ + char *msg_id; /* ignored */ + char *radio_chan; /* ignored */ + char *payload; /* Encoded as 6 bits per character. */ + char *fill_bits; /* Number of bits to discard. */ + + next = stemp; + talker = strsep(&next, ","); + frag_count = strsep(&next, ","); + frag_num = strsep(&next, ","); + msg_id = strsep(&next, ","); + radio_chan = strsep(&next, ","); + payload = strsep(&next, ","); + fill_bits = strsep(&next, ","); + + /* Suppress the 'set but not used' compiler warnings. */ + /* Alternatively, we might use __attribute__((unused)) */ + + (void)(talker); + (void)(frag_count); + (void)(frag_num); + (void)(msg_id); + (void)(radio_chan); + + if (payload == NULL || strlen(payload) == 0) { + if ( ! quiet) { + text_color_set (DW_COLOR_ERROR); + dw_printf("Payload is missing from AIS sentence.\n"); + } + return (-1); + } + +// Convert character representation to bit vector. + + unsigned char ais[256]; + memset (ais, 0, sizeof(ais)); + + int plen = strlen(payload); + + for (int k = 0; k < plen; k++) { + set_field (ais, k*6, 6, char_to_sextet(payload[k])); + } + +// Verify number of filler bits. + + int nfill = atoi(fill_bits); + int nbytes = (plen * 6) / 8; + + if (nfill != plen * 6 - nbytes * 8) { + if ( ! quiet) { + text_color_set (DW_COLOR_ERROR); + dw_printf("Number of filler bits is %d when %d is expected.\n", + nfill, plen * 6 - nbytes * 8); + } + } + + +// Extract the fields of interest from a few message types. +// Don't get too carried away. + + int type = get_field(ais, 0, 6); + + if (type >= 1 && type <= 27) { + snprintf (mssi, mssi_size, "%09d", get_field(ais, 8, 30)); + } + switch (type) { + + case 1: // Position Report Class A + case 2: + case 3: + + snprintf (descr, descr_size, "AIS %d: Position Report Class A", type); + *symtab = '/'; + *symbol = 's'; // Power boat (ship) side view + *odlon = get_field_lon(ais, 61, 28); + *odlat = get_field_lat(ais, 89, 27); + *ofknots = get_field_speed(ais, 50, 10); + *ofcourse = get_field_course(ais, 116, 12); + get_ship_data(mssi, comment, comment_size); + break; + + case 4: // Base Station Report + + snprintf (descr, descr_size, "AIS %d: Base Station Report", type); + *symtab = '\\'; + *symbol = 'L'; // Lighthouse + //year = get_field(ais, 38, 14); + //month = get_field(ais, 52, 4); + //day = get_field(ais, 56, 5); + //hour = get_field(ais, 61, 5); + //minute = get_field(ais, 66, 6); + //second = get_field(ais, 72, 6); + *odlon = get_field_lon(ais, 79, 28); + *odlat = get_field_lat(ais, 107, 27); + // Is this suitable or not? Doesn't hurt, I suppose. + get_ship_data(mssi, comment, comment_size); + break; + + case 5: // Static and Voyage Related Data + + snprintf (descr, descr_size, "AIS %d: Static and Voyage Related Data", type); + *symtab = '/'; + *symbol = 's'; // Power boat (ship) side view + { + char callsign[12]; + char shipname[24]; + char destination[24]; + get_field_string(ais, 70, 42, callsign); + get_field_string(ais, 112, 120, shipname); + get_field_string(ais, 302, 120, destination); + save_ship_data(mssi, shipname, callsign, destination); + get_ship_data(mssi, comment, comment_size); + } + break; + + + case 9: // Standard SAR Aircraft Position Report + + snprintf (descr, descr_size, "AIS %d: SAR Aircraft Position Report", type); + *symtab = '/'; + *symbol = '\''; // Small AIRCRAFT + *ofalt_m = get_field(ais, 38, 12); // meters, 4095 means not available + *odlon = get_field_lon(ais, 61, 28); + *odlat = get_field_lat(ais, 89, 27); + *ofknots = get_field_speed(ais, 50, 10); // plane is knots, not knots/10 + if (*ofknots != G_UNKNOWN) *ofknots = *ofknots * 10.0; + *ofcourse = get_field_course(ais, 116, 12); + get_ship_data(mssi, comment, comment_size); + break; + + case 18: // Standard Class B CS Position Report + // As an oversimplification, Class A is commercial, B is recreational. + + snprintf (descr, descr_size, "AIS %d: Standard Class B CS Position Report", type); + *symtab = '/'; + *symbol = 'Y'; // YACHT (sail) + *odlon = get_field_lon(ais, 57, 28); + *odlat = get_field_lat(ais, 85, 27); + get_ship_data(mssi, comment, comment_size); + break; + + case 19: // Extended Class B CS Position Report + + snprintf (descr, descr_size, "AIS %d: Extended Class B CS Position Report", type); + *symtab = '/'; + *symbol = 'Y'; // YACHT (sail) + *odlon = get_field_lon(ais, 57, 28); + *odlat = get_field_lat(ais, 85, 27); + get_ship_data(mssi, comment, comment_size); + break; + + case 27: // Long Range AIS Broadcast message + + snprintf (descr, descr_size, "AIS %d: Long Range AIS Broadcast message", type); + *symtab = '\\'; + *symbol = 's'; // OVERLAY SHIP/boat (top view) + *odlon = get_field_lon(ais, 44, 18); // Note: minutes/10 rather than usual /10000. + *odlat = get_field_lat(ais, 62, 17); + *ofknots = get_field_speed(ais, 79, 6); // Note: knots, not deciknots. + *ofcourse = get_field_course(ais, 85, 9); // Note: degrees, not decidegrees. + get_ship_data(mssi, comment, comment_size); + break; + + default: + snprintf (descr, descr_size, "AIS message type %d", type); + break; + } + + return (0); + +} /* end ais_parse */ + + + +/*------------------------------------------------------------------- + * + * Name: ais_check_length + * + * Purpose: Verify frame length against expected. + * + * Inputs: type Message type, 1 - 27. + * + * length Number of data octets in in frame. + * + * Returns: -1 Invalid message type. + * 0 Good length. + * 1 Unexpected length. + * + *--------------------------------------------------------------------*/ + +int ais_check_length (int type, int length) +{ + if (type >= 1 && type <= NUM_TYPES) { + int b = length * 8; + if (b >= valid_len[type].min && b <= valid_len[type].max) { + return (0); // Good. + } + else { + //text_color_set (DW_COLOR_ERROR); + //dw_printf("AIS ERROR: type %d, has %d bits when %d to %d expected.\n", + // type, b, valid_len[type].min, valid_len[type].max); + return (1); // Length out of range. + } + } + else { + //text_color_set (DW_COLOR_ERROR); + //dw_printf("AIS ERROR: message type %d is invalid.\n", type); + return (-1); // Invalid type. + } + +} // end ais_check_length + + + +/*------------------------------------------------------------------- + * + * Name: save_ship_data + * + * Purpose: Save shipname, etc., from "Static and Voyage Related Data" + * so it can be combined later with the position reports. + * + * Inputs: mssi + * shipname + * callsign + * destination + * + *--------------------------------------------------------------------*/ + +struct ship_data_s { + struct ship_data_s *pnext; + char mssi[9+1]; + char shipname[20+1]; + char callsign[7+1]; + char destination[20+1]; +}; + +// Just use a single linked list for now. +// If I get ambitious, I might use a hash table. +// I don't think we need a critical region because all channels +// should be serialized thru the receive queue. + +static struct ship_data_s *ships = NULL; + + +static void save_ship_data(char *mssi, char *shipname, char *callsign, char *destination) +{ + // Get list node, either existing or new. + struct ship_data_s *p = ships; + while (p != NULL) { + if (strcmp(mssi, p->mssi) == 0) { + break; + } + p = p->pnext; + } + if (p == NULL) { + p = calloc(sizeof(struct ship_data_s),1); + p->pnext = ships; + ships = p; + } + + strlcpy (p->mssi, mssi, sizeof(p->mssi)); + strlcpy (p->shipname, shipname, sizeof(p->shipname)); + strlcpy (p->callsign, callsign, sizeof(p->callsign)); + strlcpy (p->destination, destination, sizeof(p->destination)); +} + +/*------------------------------------------------------------------- + * + * Name: save_ship_data + * + * Purpose: Get ship data for specified mssi. + * + * Inputs: mssi + * + * Outputs: comment - If mssi is found, return in single string here, + * suitable for the comment field. + * + *--------------------------------------------------------------------*/ + +static void get_ship_data(char *mssi, char *comment, int comment_size) +{ + struct ship_data_s *p = ships; + while (p != NULL) { + if (strcmp(mssi, p->mssi) == 0) { + break; + } + p = p->pnext; + } + if (p != NULL) { + if (strlen(p->destination) > 0) { + snprintf (comment, comment_size, "%s, %s, dest. %s", p->shipname, p->callsign, p->destination); + } + else { + snprintf (comment, comment_size, "%s, %s", p->shipname, p->callsign); + } + } +} + + +// end ais.c diff --git a/src/ais.h b/src/ais.h new file mode 100644 index 00000000..6b962884 --- /dev/null +++ b/src/ais.h @@ -0,0 +1,8 @@ + + +void ais_to_nmea (unsigned char *ais, int ais_len, char *nema, int nema_size); + +int ais_parse (char *sentence, int quiet, char *descr, int descr_size, char *mssi, int mssi_size, double *odlat, double *odlon, + float *ofknots, float *ofcourse, float *ofalt_m, char *symtab, char *symbol, char *comment, int comment_size); + +int ais_check_length (int type, int length); diff --git a/src/appserver.c b/src/appserver.c new file mode 100644 index 00000000..b0ef7d87 --- /dev/null +++ b/src/appserver.c @@ -0,0 +1,741 @@ +// ****** PRELIMINARY - needs work ****** + +#define DEBUG 1 + + +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// + + + +#include "direwolf.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "ax25_pad.h" +#include "textcolor.h" +#include "agwlib.h" // Network TNC interface. + + +/*------------------------------------------------------------------ + * + * Module: appserver.c + * + * Purpose: Simple application server for connected mode AX.25. + * + * This demonstrates how you can write a application that will wait for + * a connection from another station and respond to commands. + * It can be used as a starting point for developing your own applications. + * + * Description: This attaches to an instance of Dire Wolf via the AGW network interface. + * It processes commands from other radio stations and responds. + * + *---------------------------------------------------------------*/ + + +static void usage() +{ + text_color_set(DW_COLOR_ERROR); + dw_printf ("Usage: \n"); + dw_printf (" \n"); + dw_printf ("appserver [ -h hostname ] [ -p port ] mycall \n"); + dw_printf (" \n"); + dw_printf (" -h hostname for TNC. Default is localhost. \n"); + dw_printf (" \n"); + dw_printf (" -p tcp port for TNC. Default is 8000. \n"); + dw_printf (" \n"); + dw_printf (" mycall is required because that is the callsign for \n"); + dw_printf (" which the TNC will accept connections. \n"); + dw_printf (" \n"); + exit (EXIT_FAILURE); +} + + + + + +static char mycall[AX25_MAX_ADDR_LEN]; /* Callsign, with SSID, for the application. */ + /* Future? Could have multiple applications, on the same */ + /* radio channel, each with its own SSID. */ + +static char tnc_hostname[80]; /* DNS host name or IPv4 address. */ + /* Some of the code is there for IPv6 but */ + /* needs more work. */ + /* Defaults to "localhost" if not specified. */ + +static char tnc_port[8]; /* a TCP port number. Default 8000. */ + + + +/* + * Maintain information about connections from users which we will call "sessions." + * It should be possible to have multiple users connected at the same time. + * + * This allows a "who" command to see who is currently connected and a place to keep + * possible state information for each user. + * + * Each combination of channel & callsign is a separate session. + * The same user (callsign), on a different channel, is a different session. + */ + + +struct session_s { + + char client_addr[AX25_MAX_ADDR_LEN]; // Callsign of other station. + // Clear to mean this table entry is not in use. + + int channel; // Radio channel. + + time_t login_time; // Time when connection established. + +// For the timing test. +// Send specified number of frames, optional length. +// When finished summarize with statistics. + + time_t tt_start_time; + volatile int tt_count; // Number to send. + int tt_length; // Bytes in info part. + int tt_next; // Next sequence to send. + + volatile int tx_queue_len; // Number in transmit queue. For flow control. +}; + +#define MAX_SESSIONS 12 + +static struct session_s session[MAX_SESSIONS]; + +static int find_session (int chan, char *addr, int create); +static void poll_timing_test (void); + + + +/*------------------------------------------------------------------ + * + * Name: main + * + * Purpose: Attach to Dire Wolf TNC, wait for requests from users. + * + * Usage: Described above. + * + *---------------------------------------------------------------*/ + + +int main (int argc, char *argv[]) +{ + int c; + char *p; + +#if __WIN32__ + setvbuf(stdout, NULL, _IONBF, 0); +#else + setlinebuf (stdout); +#endif + + memset (session, 0, sizeof(session)); + + strlcpy (tnc_hostname, "localhost", sizeof(tnc_hostname)); + strlcpy (tnc_port, "8000", sizeof(tnc_port)); + +/* + * Extract command line args. + */ + + while ((c = getopt (argc, argv, "h:p:")) != -1) { + switch (c) { + + case 'h': + strlcpy (tnc_hostname, optarg, sizeof(tnc_hostname)); + break; + + case 'p': + strlcpy (tnc_port, optarg, sizeof(tnc_port)); + break; + + default: + usage (); + } + } + + if (argv[optind] == NULL) { + usage (); + } + + strlcpy (mycall, argv[optind], sizeof(mycall)); + + // Force to upper case. + for (p = mycall; *p != '\0'; p++) { + if (islower(*p)) { + *p = toupper(*p); + } + } + +/* + * Establish a TCP socket to the network TNC. + * It starts up a thread, which listens for messages from the TNC, + * and calls the corresponding agw_cb_... callback functions. + * + * After attaching to the TNC, the specified init function is called. + * We pass it to the library, rather than doing it here, so it can + * repeated automatically if the TNC goes away and comes back again. + * We need to reestablish what it knows about the application. + */ + + if (agwlib_init (tnc_hostname, tnc_port, agwlib_G_ask_port_information) != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not attach to network TNC %s:%s.\n", tnc_hostname, tnc_port); + exit (EXIT_FAILURE); + } + + +/* + * Send command to ask what channels are available. + * The response will be handled by agw_cb_G_port_information. + */ +// FIXME: Need to do this again if we lose TNC and reattach to it. + + /// should happen automatically now. agwlib_G_ask_port_information (); + + + while (1) { + SLEEP_SEC(1); // other places based on 1 second assumption. + poll_timing_test (); + } + + +} /* end main */ + + + +static void poll_timing_test (void) +{ + int s; + for (s = 0; s < MAX_SESSIONS; s++) { + + if (session[s].tt_count == 0) { + continue; // nothing to do + } + else if (session[s].tt_next <= session[s].tt_count) { + int rem = session[s].tt_count - session[s].tt_next + 1; // remaining to send. + agwlib_Y_outstanding_frames_for_station (session[s].channel, mycall, session[s].client_addr); + SLEEP_MS(10); + if (session[s].tx_queue_len > 128) continue; // enough queued up for now. + if (rem > 64) rem = 64; // add no more than 64 at a time. + int i; + for (i = 0; i < rem; i++) { + char c = 'a'; + char stuff[AX25_MAX_INFO_LEN+2]; + snprintf (stuff, sizeof(stuff), "%06d ", session[s].tt_next); + int k; + for (k = strlen(stuff); k < session[s].tt_length - 1; k++) { + stuff[k] = c; + c++; + if (c == 'z' + 1) c = 'A'; + if (c == 'Z' + 1) c = '0'; + if (c == '9' + 1) c = 'a'; + } + stuff[k++] = '\r'; + stuff[k++] = '\0'; + agwlib_D_send_connected_data (session[s].channel, 0xF0, mycall, session[s].client_addr, strlen(stuff), stuff); + session[s].tt_next++; + } + } + else { + // All done queuing up the packets. + // Wait until they have all been sent and ack'ed by other end. + + agwlib_Y_outstanding_frames_for_station (session[s].channel, mycall, session[s].client_addr); + SLEEP_MS(10); + + if (session[s].tx_queue_len > 0) continue; // not done yet. + + int elapsed = time(NULL) - session[s].tt_start_time; + if (elapsed <= 0) elapsed = 1; // avoid divide by 0 + + int byte_count = session[s].tt_count * session[s].tt_length; + char summary[100]; + snprintf (summary, sizeof(summary), "%d bytes in %d seconds, %d bytes/sec, efficiency %d%% at 1200, %d%% at 9600.\r", + byte_count, elapsed, byte_count/elapsed, + byte_count * 8 * 100 / elapsed / 1200, + byte_count * 8 * 100 / elapsed / 9600); + + agwlib_D_send_connected_data (session[s].channel, 0xF0, mycall, session[s].client_addr, strlen(summary), summary); + session[s].tt_count = 0; // all done. + } + } + +} // end poll_timing_test + + + +/*------------------------------------------------------------------- + * + * Name: agw_cb_C_connection_received + * + * Purpose: Callback for the "connection received" command from the TNC. + * + * Inputs: chan - Radio channel, first is 0. + * + * call_from - Address of other station. + * + * call_to - Callsign I responded to. (could be an alias.) + * + * data_len - Length of data field. + * + * data - Should look something like this for incoming: + * *** CONNECTED to Station xxx\r + * + * Description: Add to the sessions table. + * + *--------------------------------------------------------------------*/ + +/*------------------------------------------------------------------- + * + * Name: on_C_connection_received + * + * Purpose: Callback for the "connection received" command from the TNC. + * + * Inputs: chan - Radio channel, first is 0. + * + * call_from - Address of other station. + * + * call_to - My call. + * In the case of an incoming connect request (i.e. to + * a server) this is the callsign I responded to. + * It is possible to define additional aliases and respond + * to any one of them. It would be possible to have a server + * that responds to multiple names and behaves differently + * depending on the name. + * + * incoming - true(1) if other station made connect request. + * false(0) if I made request and other statio accepted. + * + * data - Should look something like this for incoming: + * *** CONNECTED to Station xxx\r + * and this for my request being accepted: + * *** CONNECTED With Station xxx\r + * + * session_id - Session id to be used in data transfer and + * other control functions related to this connection. + * Think of it like a file handle. Once it is open + * we usually don't care about the name anymore and + * and just refer to the handle. This is used to + * keep track of multiple connections at the same + * time. e.g. a server could be handling multiple + * clients at once on the same or different channels. + * + * Description: Add to the table of clients. + * + *--------------------------------------------------------------------*/ + + +// old void agw_cb_C_connection_received (int chan, char *call_from, char *call_to, int data_len, char *data) +void on_C_connection_received (int chan, char *call_from, char *call_to, int incoming, char *data) +{ + int s; + char *p; + char greeting[256]; + + + for (p = data; *p != '\0'; p++) { + if (! isprint(*p)) *p = '\0'; // Remove any \r character at end. + } + + s = find_session (chan, call_from, 1); + + if (s >= 0) { + + text_color_set(DW_COLOR_INFO); + dw_printf ("Begin session %d: %s\n", s, data); + +// Send greeting. + + snprintf (greeting, sizeof(greeting), "Welcome! Type ? for list of commands or HELP for details.\r"); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + } + else { + + text_color_set(DW_COLOR_INFO); + dw_printf ("Too many users already: %s\n", data); + +// Sorry, too many users already. + + snprintf (greeting, sizeof(greeting), "Sorry, maximum number of users has been exceeded. Try again later.\r"); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + + // FIXME: Ideally we'd want to wait until nothing in the outgoing queue + // to that station so we know the rejection message was received. + SLEEP_SEC (10); + agwlib_d_disconnect (chan, mycall, call_from); + } + +} /* end agw_cb_C_connection_received */ + + + +/*------------------------------------------------------------------- + * + * Name: agw_cb_d_disconnected + * + * Purpose: Process the "disconnected" command from the TNC. + * + * Inputs: chan - Radio channel. + * + * call_from - Address of other station. + * + * call_to - Callsign I responded to. (could be aliases.) + * + * data_len - Length of data field. + * + * data - Should look something like one of these: + * *** DISCONNECTED RETRYOUT With xxx\r + * *** DISCONNECTED From Station xxx\r + * + * Description: Remove from the sessions table. + * + *--------------------------------------------------------------------*/ + + + +void agw_cb_d_disconnected (int chan, char *call_from, char *call_to, int data_len, char *data) +{ + int s; + char *p; + + s = find_session (chan, call_from, 0); + + for (p = data; *p != '\0'; p++) { + if (! isprint(*p)) *p = '\0'; // Remove any \r character at end. + } + + text_color_set(DW_COLOR_INFO); + dw_printf ("End session %d: %s\n", s, data); + +// Remove from session table. + + if (s >= 0) { + memset (&(session[s]), 0, sizeof(struct session_s)); + } + +} /* end agw_cb_d_disconnected */ + + + +/*------------------------------------------------------------------- + * + * Name: agw_cb_D_connected_data + * + * Purpose: Process "connected ax.25 data" from the TNC. + * + * Inputs: chan - Radio channel. + * + * addr - Address of other station. + * + * msg - What the user sent us. Probably a command. + * + * Global In: tnc_sock - Socket for TNC. + * + * Description: Remove from the session table. + * + *--------------------------------------------------------------------*/ + + +void agw_cb_D_connected_data (int chan, char *call_from, char *call_to, int data_len, char *data) +{ + int s; + char *p; + char logit[AX25_MAX_INFO_LEN+100]; + char *pcmd; + char *save; + + s = find_session (chan, call_from, 0); + + for (p = data; *p != '\0'; p++) { + if (! isprint(*p)) *p = '\0'; // Remove any \r character at end. + } + + // TODO: Should timestamp to all output. + + snprintf (logit, sizeof(logit), "%d,%d,%s: %s\n", s, chan, call_from, data); + text_color_set(DW_COLOR_INFO); + dw_printf ("%s", logit); + + if (s < 0) { + // Uh oh. Data from some station when not connected. + text_color_set(DW_COLOR_ERROR); + dw_printf ("Internal error. Incoming data, no corresponding session.\n"); + return; + } + +// Process the command from user. + + pcmd = strtok_r (data, " ", &save); + if (pcmd == NULL || strlen(pcmd) == 0) { + + char greeting[80]; + strlcpy (greeting, "Type ? for list of commands or HELP for details.\r", sizeof(greeting)); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + return; + } + + if (strcasecmp(pcmd, "who") == 0) { + +// who - list people currently logged in. + + int n; + char greeting[128]; + + snprintf (greeting, sizeof(greeting), "Session Channel User Since\r"); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + + for (n = 0; n < MAX_SESSIONS; n++) { + if (session[n].client_addr[0]) { +// I think compiler is confused. It says up to 520 characters can be written. +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wformat-truncation" + snprintf (greeting, sizeof(greeting), " %2d %d %-9s [time later]\r", + n, session[n].channel, session[n].client_addr); +#pragma GCC diagnostic pop + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + } + } + } + else if (strcasecmp(pcmd, "test") == 0) { + +// test - timing test +// Send specified number of frames with optional length. + + char *pcount = strtok_r (NULL, " ", &save); + char *plength = strtok_r (NULL, " ", &save); + + session[s].tt_start_time = time(NULL); + session[s].tt_next = 1; + session[s].tt_length = 256; + session[s].tt_count = 1; + + if (plength != NULL) { + session[s].tt_length = atoi(plength); + if (session[s].tt_length < 16) session[s].tt_length = 16; + if (session[s].tt_length > AX25_MAX_INFO_LEN) session[s].tt_length = AX25_MAX_INFO_LEN; + } + if (pcount != NULL) { + session[s].tt_count = atoi(pcount); + } + + // The background polling will take it from here. + } + else if (strcasecmp(pcmd, "bye") == 0) { + +// bye - disconnect. + + char greeting[80]; + strlcpy (greeting, "Thank you folks for kindly droppin' in. Y'all come on back now, ya hear?\r", sizeof(greeting)); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + // Ideally we'd want to wait until nothing in the outgoing queue + // to that station so we know the message was received. + SLEEP_SEC (10); + agwlib_d_disconnect (chan, mycall, call_from); + } + else if (strcasecmp(pcmd, "help") == 0 || strcasecmp(pcmd, "?") == 0) { + +// help. + + char greeting[80]; + strlcpy (greeting, "Help not yet available.\r", sizeof(greeting)); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + } + else { + +// command not recognized. + + char greeting[80]; + strlcpy (greeting, "Invalid command. Type ? for list of commands or HELP for details.\r", sizeof(greeting)); + agwlib_D_send_connected_data (chan, 0xF0, mycall, call_from, strlen(greeting), greeting); + } + +} /* end agw_cb_D_connected_data */ + + + + +/*------------------------------------------------------------------- + * + * Name: agw_cb_G_port_information + * + * Purpose: Process the port information "radio channels available" response from the TNC. + * + * + * Inputs: num_chan_avail - Number of radio channels available. + * + * chan_descriptions - Array of string pointers to form "Port99 description". + * Port1 is channel 0. + * + *--------------------------------------------------------------------*/ + +void agw_cb_G_port_information (int num_chan_avail, char *chan_descriptions[]) +{ + char *p; + int n; + + text_color_set(DW_COLOR_INFO); + dw_printf("TNC has %d radio channel%s available:\n", num_chan_avail, (num_chan_avail != 1) ? "s" : ""); + + for (n = 0; n < num_chan_avail; n++) { + + p = chan_descriptions[n]; + + // Expecting something like this: "Port1 first soundcard mono" + + if (strncasecmp(p, "Port", 4) == 0 && isdigit(p[4])) { + + int chan = atoi(p+4) - 1; // "Port1" is our channel 0. + if (chan >= 0 && chan < MAX_CHANS) { + + char *desc = p + 4; + while (*desc != '\0' && (*desc == ' ' || isdigit(*desc))) { + desc++; + } + + text_color_set(DW_COLOR_INFO); + dw_printf(" Channel %d: %s\n", chan, desc); + + // Later? Use 'g' to get speed and maybe other properties? + // Though I'm not sure why we would care here. + +/* + * Send command to register my callsign for incoming connect requests. + */ + + agwlib_X_register_callsign (chan, mycall); + + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf("Radio channel number is out of bounds: %s\n", p); + } + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf("Radio channel description not in expected format: %s\n", p); + } + } + +} /* end agw_cb_G_port_information */ + + +/*------------------------------------------------------------------- + * + * Name: agw_cb_Y_outstanding_frames_for_station + * + * Purpose: Process the "disconnected" command from the TNC. + * + * Inputs: chan - Radio channel. + * + * call_from - Should be my call. + * + * call_to - Callsign of other station. + * + * frame_count + * + * Description: Remove from the sessions table. + * + *--------------------------------------------------------------------*/ + + + +void agw_cb_Y_outstanding_frames_for_station (int chan, char *call_from, char *call_to, int frame_count) +{ + int s; + + s = find_session (chan, call_to, 0); + + text_color_set(DW_COLOR_DEBUG); // FIXME temporary + dw_printf ("debug ----------------------> session %d, callback Y outstanding frame_count %d\n", s, frame_count); + +// Update the transmit queue length + + if (s >= 0) { + session[s].tx_queue_len = frame_count; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Oops! Did not expect to be here.\n"); + } + +} /* end agw_cb_Y_outstanding_frames_for_station */ + + + +/*------------------------------------------------------------------- + * + * Name: find_session + * + * Purpose: Given a channel number and address (callsign), find existing + * table entry or create a new one. + * + * Inputs: chan - Radio channel number. + * + * addr - Address of station contacting us. + * + * create - If true, try create a new entry if not already there. + * + * Returns: "session id" which is an index into "session" array or -1 for failure. + * + *--------------------------------------------------------------------*/ + +static int find_session (int chan, char *addr, int create) +{ + int i; + int s = -1; + +// Is it there already? + + +//#if DEBUG +// +// text_color_set(DW_COLOR_DEBUG); +// dw_printf("find_session (%d, %s, %d)\n", chan, addr, create); +//#endif + + for (i = 0; i < MAX_SESSIONS; i++) { + if (session[i].channel == chan && strcmp(session[i].client_addr, addr) == 0) { + s = i; + break; + } + } + + if (s >= 0) return (s); + + if (! create) return (-1); + +// No, and there is a request to add a new entry. +// See if we have any available space. + + s = -1; + for (i = 0; i < MAX_SESSIONS; i++) { + if (strlen(session[i].client_addr) == 0) { + s = i; + break; + } + } + + if (s < 0) return (-1); + + strlcpy (session[s].client_addr, addr, sizeof(session[s].client_addr)); + session[s].channel = chan; + session[s].login_time = time(NULL); + + return (s); + +} /* end find_session */ + + +/* end appserver.c */ diff --git a/aprs_tt.c b/src/aprs_tt.c similarity index 91% rename from aprs_tt.c rename to src/aprs_tt.c index cf00cc8b..7b125759 100644 --- a/aprs_tt.c +++ b/src/aprs_tt.c @@ -115,6 +115,8 @@ static int find_ttloc_match (char *e, char *xstr, char *ystr, char *zstr, char * static void check_result (void); #endif +static int tt_debug = 0; + /*------------------------------------------------------------------ * @@ -122,7 +124,8 @@ static void check_result (void); * * Purpose: Initialize the APRStt gateway at system startup time. * - * Inputs: Configuration options gathered by config.c. + * Inputs: P - Pointer to configuration options gathered by config.c. + * debug - Debug printing control. * * Global out: Make our own local copy of the structure here. * @@ -164,9 +167,10 @@ static struct ttloc_s test_config[] = { #endif -void aprs_tt_init (struct tt_config_s *p) +void aprs_tt_init (struct tt_config_s *p, int debug) { int c; + tt_debug = debug; #if TT_MAIN /* For unit testing. */ @@ -208,7 +212,7 @@ void aprs_tt_init (struct tt_config_s *p) * The complete message is then processed. * The touch tone decoder sends $ if no activity * for some amount of time, perhaps 5 seconds. - * A partially accumulated messge is discarded if + * A partially accumulated message is discarded if * there is a long gap. * * '.' means no activity during processing period. @@ -422,7 +426,7 @@ void aprs_tt_sequence (int chan, char *msg) * Anything from script, above, will override other predefined responses. */ - char audible_response[1000]; + char audible_response[sizeof(script_response) + 16]; snprintf (audible_response, sizeof(audible_response), "APRSTT>%s:%s", @@ -483,7 +487,19 @@ static int parse_fields (char *msg) //text_color_set(DW_COLOR_DEBUG); //dw_printf ("parse_fields (%s).\n", msg); - strlcpy (stemp, msg, sizeof(stemp)); +// Make a copy of msg because strtok corrupts the original. +// While we are at it, remove any blanks. +// This should not happen with DTMF reception but could happen +// in manually crafted strings for testing. + + int n = 0; + for (char *m = msg; *m != '\0' && n < sizeof(stemp)-1; m++) { + if (*m != ' ') { + stemp[n++] = *m; + } + } + stemp[n] = '\0'; + e = strtok_r (stemp, "*#", &save); while (e != NULL) { @@ -551,7 +567,7 @@ static int parse_fields (char *msg) default: text_color_set(DW_COLOR_ERROR); - dw_printf ("Field does not start with A, B, C, or digit: \"%s\"\n", msg); + dw_printf ("Field does not start with A, B, C, or digit: \"%s\"\n", e); return (TT_ERROR_D_MSG); } @@ -574,7 +590,7 @@ static int parse_fields (char *msg) * Purpose: Expand compact form "macro" to full format then process. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should contain only digits. * * Returns: 0 for success or one of the TT_ERROR_... codes. @@ -689,18 +705,16 @@ static int expand_macro (char *e) * Purpose: Extract traditional format callsign or object name from touch tone sequence. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. - * In this case, it should start with "A". + * APRStt message. + * In this case, it should start with "A" then a digit. * * Outputs: m_callsign * * m_symtab_or_overlay - Set to 0-9 or A-Z if specified. * - * m_symbol_code - Always set to 'A'. - * NO! This should be applied only if we - * have the default value at this point. - * The symbol might have been explicitly - * set already and we don't want to overwrite that. + * m_symbol_code - Always set to 'A' (Box, DTMF or RFID) + * If you want a different symbol, use the new + * object name format and separate symbol specification. * * Returns: 0 for success or one of the TT_ERROR_... codes. * @@ -752,6 +766,11 @@ static int parse_callsign (char *e) int len; char tttemp[40], stemp[30]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("APRStt parse callsign (starts with A then digit): \"%s\"\n", e); + } + assert (*e == 'A'); len = strlen(e); @@ -762,6 +781,10 @@ static int parse_callsign (char *e) if (len == 4 && isdigit(e[1]) && isdigit(e[2]) && isdigit(e[3])) { strlcpy (m_callsign, e+1, sizeof(m_callsign)); + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Special case, 3 digit tactical call: \"%s\"\n", m_callsign); + } return (0); } @@ -779,7 +802,7 @@ static int parse_callsign (char *e) return (cs_err); } - strncpy (m_callsign, e+1, 3); + memcpy (m_callsign, e+1, 3); m_callsign[3] = '\0'; if (len == 7) { @@ -789,10 +812,20 @@ static int parse_callsign (char *e) tt_two_key_to_text (tttemp, 0, stemp); m_symbol_code = APRSTT_DEFAULT_SYMBOL; m_symtab_or_overlay = stemp[0]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Three digit abbreviation1: callsign \"%s\", symbol code '%c (Box DTMF)', overlay '%c', checksum %c\n", + m_callsign, m_symbol_code, m_symtab_or_overlay, e[len-1]); + } } else { m_symbol_code = APRSTT_DEFAULT_SYMBOL; m_symtab_or_overlay = e[len-2]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Three digit abbreviation2: callsign \"%s\", symbol code '%c' (Box DTMF), overlay '%c', checksum %c\n", + m_callsign, m_symbol_code, m_symtab_or_overlay, e[len-1]); + } } return (0); } @@ -810,7 +843,7 @@ static int parse_callsign (char *e) } if (isupper(e[len-2])) { - strncpy (tttemp, e+1, len-4); + memcpy (tttemp, e+1, len-4); tttemp[len-4] = '\0'; tt_two_key_to_text (tttemp, 0, m_callsign); @@ -820,14 +853,24 @@ static int parse_callsign (char *e) tt_two_key_to_text (tttemp, 0, stemp); m_symbol_code = APRSTT_DEFAULT_SYMBOL; m_symtab_or_overlay = stemp[0]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Callsign in two key format1: callsign \"%s\", symbol code '%c' (Box DTMF), overlay '%c', checksum %c\n", + m_callsign, m_symbol_code, m_symtab_or_overlay, e[len-1]); + } } else { - strncpy (tttemp, e+1, len-3); + memcpy (tttemp, e+1, len-3); tttemp[len-3] = '\0'; tt_two_key_to_text (tttemp, 0, m_callsign); m_symbol_code = APRSTT_DEFAULT_SYMBOL; m_symtab_or_overlay = e[len-2]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Callsign in two key format2: callsign \"%s\", symbol code '%c' (Box DTMF), overlay '%c', checksum %c\n", + m_callsign, m_symbol_code, m_symtab_or_overlay, e[len-1]); + } } return (0); } @@ -845,7 +888,7 @@ static int parse_callsign (char *e) * Purpose: Extract object name from touch tone sequence. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should start with "AA". * * Outputs: m_callsign @@ -864,9 +907,11 @@ static int parse_callsign (char *e) static int parse_object_name (char *e) { int len; - //int c_length; - //char tttemp[40]; - //char stemp[30]; + + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("APRStt parse object name (starts with AA): \"%s\"\n", e); + } assert (e[0] == 'A'); assert (e[1] == 'A'); @@ -882,6 +927,10 @@ static int parse_object_name (char *e) if (tt_two_key_to_text (e+2, 0, m_callsign) == 0) { m_callsign[9] = '\0'; /* truncate to 9 */ m_ssid = 0; /* No ssid for object name */ + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Object name in two key format: \"%s\"\n", m_callsign); + } return (0); } } @@ -901,7 +950,7 @@ static int parse_object_name (char *e) * Purpose: Extract symbol from touch tone sequence. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should start with "AB". * * Outputs: m_symtab_or_overlay @@ -935,6 +984,11 @@ static int parse_symbol (char *e) int nn; char stemp[10]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("APRStt parse symbol (starts with AB): \"%s\"\n", e); + } + assert (e[0] == 'A'); assert (e[1] == 'B'); @@ -959,12 +1013,22 @@ static int parse_symbol (char *e) case '1': m_symtab_or_overlay = '/'; m_symbol_code = 32 + nn; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("symbol code '%c', primary symbol table '%c'\n", + m_symbol_code, m_symtab_or_overlay); + } return (0); break; case '2': m_symtab_or_overlay = '\\'; m_symbol_code = 32 + nn; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("symbol code '%c', alternate symbol table '%c'\n", + m_symbol_code, m_symtab_or_overlay); + } return (0); break; @@ -973,6 +1037,11 @@ static int parse_symbol (char *e) if (tt_two_key_to_text (e+5, 0, stemp) == 0) { m_symbol_code = 32 + nn; m_symtab_or_overlay = stemp[0]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("symbol code '%c', alternate symbol table with overlay '%c'\n", + m_symbol_code, m_symtab_or_overlay); + } return (0); } } @@ -995,7 +1064,7 @@ static int parse_symbol (char *e) * Purpose: Extract QIKcom-2 / APRStt 3 ten digit call or five digit suffix. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should start with "AC". * * Outputs: m_callsign @@ -1018,6 +1087,11 @@ static int parse_aprstt3_call (char *e) assert (e[0] == 'A'); assert (e[1] == 'C'); + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("APRStt parse QIKcom-2 / APRStt 3 ten digit call or five digit suffix (starts with AC): \"%s\"\n", e); + } + if (strlen(e) == 2+10) { char call[12]; @@ -1073,7 +1147,7 @@ static int parse_aprstt3_call (char *e) * Purpose: Extract location from touch tone sequence. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should start with "B". * * Outputs: m_latitude @@ -1125,6 +1199,11 @@ static int parse_location (char *e) char mh[20]; char stemp[32]; + if (tt_debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("APRStt parse location (starts with B): \"%s\"\n", e); + // TODO: more detail later... + } assert (*e == 'B'); @@ -1204,13 +1283,26 @@ static int parse_location (char *e) lat0 = tt_config.ttloc_ptr[ipat].grid.lat0; lat9 = tt_config.ttloc_ptr[ipat].grid.lat9; + double yrange = lat9 - lat0; y = atof(ystr); - m_latitude = lat0 + y * (lat9-lat0) / (pow(10., strlen(ystr)) - 1.); - + double user_y_max = round(pow(10., strlen(ystr)) - 1.); // e.g. 999 for 3 digits + m_latitude = lat0 + yrange * y / user_y_max; +#if 0 + dw_printf ("TTLOC_GRID LAT min=%f, max=%f, range=%f\n", lat0, lat9, yrange); + dw_printf ("TTLOC_GRID LAT user_y=%f, user_y_max=%f\n", y, user_y_max); + dw_printf ("TTLOC_GRID LAT min + yrange * user_y / user_y_range = %f\n", m_latitude); +#endif lon0 = tt_config.ttloc_ptr[ipat].grid.lon0; lon9 = tt_config.ttloc_ptr[ipat].grid.lon9; + double xrange = lon9 - lon0; x = atof(xstr); - m_longitude = lon0 + x * (lon9-lon0) / (pow(10., strlen(xstr)) - 1.); + double user_x_max = round(pow(10., strlen(xstr)) - 1.); + m_longitude = lon0 + xrange * x / user_x_max; +#if 0 + dw_printf ("TTLOC_GRID LON min=%f, max=%f, range=%f\n", lon0, lon9, xrange); + dw_printf ("TTLOC_GRID LON user_x=%f, user_x_max=%f\n", x, user_x_max); + dw_printf ("TTLOC_GRID LON min + xrange * user_x / user_x_range = %f\n", m_longitude); +#endif m_dao[2] = e[0]; m_dao[3] = e[1]; @@ -1415,7 +1507,7 @@ static int parse_location (char *e) * defined in the configuration file. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should start with "B". * * valstrsize - size of the outputs so we can check for buffer overflow. @@ -1566,7 +1658,7 @@ static int find_ttloc_match (char *e, char *xstr, char *ystr, char *zstr, char * * Purpose: Extract comment / status or other special information from touch tone message. * * Inputs: e - An "entry" extracted from a complete - * APRStt messsage. + * APRStt message. * In this case, it should start with "C". * * Outputs: m_comment @@ -1708,7 +1800,7 @@ static void raw_tt_data_to_app (int chan, char *msg) alevel.mark = -2; alevel.space = -2; - dlq_rec_frame (chan, -1, 0, pp, alevel, RETRY_NONE, "tt"); + dlq_rec_frame (chan, -1, 0, pp, alevel, 0, RETRY_NONE, "tt"); } else { text_color_set(DW_COLOR_ERROR); @@ -1893,6 +1985,7 @@ static const struct { /* Latitude comes out ok, 37.9137 -> 55.82 min. */ /* Longitude -81.1254 -> 8.20 min */ { "B21234*A67979#", "679", "12", "7A", "", "", "12.3400", "56.1200", "!TB2!" }, + { "B533686*A67979#", "679", "12", "7A", "", "", "37.9222", "81.1143", "!TB5!" }, // TODO: should test other coordinate systems. @@ -1985,7 +2078,7 @@ static void check_result (void) int main (int argc, char *argv[]) { - aprs_tt_init (NULL); + aprs_tt_init (NULL, 0); error_count = 0; diff --git a/aprs_tt.h b/src/aprs_tt.h similarity index 97% rename from aprs_tt.h rename to src/aprs_tt.h index 8cc845d8..4d33f487 100644 --- a/aprs_tt.h +++ b/src/aprs_tt.h @@ -123,7 +123,7 @@ struct tt_config_s { int obj_recv_chan; /* Channel to listen for tones. */ int obj_xmit_chan; /* Channel to transmit object report. */ - /* -1 for none. This could happpen if we */ + /* -1 for none. This could happen if we */ /* are only sending to application */ /* and/or IGate. */ @@ -165,7 +165,7 @@ struct tt_config_s { -void aprs_tt_init (struct tt_config_s *p_config); +void aprs_tt_init (struct tt_config_s *p_config, int debug); void aprs_tt_button (int chan, char button); diff --git a/atest.c b/src/atest.c similarity index 57% rename from atest.c rename to src/atest.c index c78094a7..c5f4ec50 100644 --- a/atest.c +++ b/src/atest.c @@ -2,7 +2,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016, 2019, 2021, 2022, 2023 John Langner, WB2OSZ // // 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 @@ -23,12 +23,12 @@ * * Name: atest.c * - * Purpose: Test fixture for the AFSK demodulator. + * Purpose: Test fixture for the Dire Wolf demodulators. * - * Inputs: Takes audio from a .WAV file insted of the audio device. + * Inputs: Takes audio from a .WAV file instead of the audio device. * - * Description: This can be used to test the AFSK demodulator under - * controlled and reproducable conditions for tweaking. + * Description: This can be used to test the demodulators under + * controlled and reproducible conditions for tweaking. * * For example * @@ -68,6 +68,7 @@ #include #include #include +#include #define ATEST_C 1 @@ -81,7 +82,9 @@ #include "dlq.h" #include "ptt.h" #include "dtime_now.h" - +#include "fx25.h" +#include "il2p.h" +#include "hdlc_rec.h" #if 0 /* Typical but not flexible enough. */ @@ -105,7 +108,7 @@ struct wav_header { /* .WAV file header. */ /* 8 bit samples are unsigned bytes */ /* in range of 0 .. 255. */ - /* 16 bit samples are signed short */ + /* 16 bit samples are little endian signed short */ /* in range of -32768 .. +32767. */ static struct { @@ -137,9 +140,13 @@ static struct { static FILE *fp; static int e_o_f; -static int packets_decoded = 0; +static int packets_decoded_one = 0; +static int packets_decoded_total = 0; static int decimate = 0; /* Reduce that sampling rate if set. */ - /* 1 = normal, 2 = half, etc. */ + /* 1 = normal, 2 = half, 3 = 1/3, etc. */ + +static int upsample = 0; /* Upsample for G3RUH decoder. */ + /* Non-zero will override the default. */ static struct audio_s my_audio_config; @@ -174,6 +181,21 @@ static int sample_number = -1; /* Sample number from the file. */ /* Use to print timestamp, relative to beginning */ /* of file, when frame was decoded. */ +// command line options. + +static int B_opt = DEFAULT_BAUD; // Bits per second. Need to change all baud references to bps. +static int g_opt = 0; // G3RUH modem regardless of speed. +static int j_opt = 0; /* 2400 bps PSK compatible with direwolf <= 1.5 */ +static int J_opt = 0; /* 2400 bps PSK compatible MFJ-2400 and maybe others. */ +static int h_opt = 0; // Hexadecimal display of received packet. +static char P_opt[16] = ""; // Demodulator profiles. +static int d_x_opt = 1; // FX.25 debug. +static int d_o_opt = 0; // "-d o" option for DCD output control. */ +static int d_2_opt = 0; // "-d 2" option for IL2P details. */ +static int dcd_count = 0; +static int dcd_missing_errors = 0; + + int main (int argc, char *argv[]) { @@ -182,7 +204,8 @@ int main (int argc, char *argv[]) int channel; double start_time; // Time when we started so we can measure elapsed time. - double duration; // Length of the audio file in seconds. + double one_filetime = 0; // Length of one audio file in seconds. + double total_filetime = 0; // Length of all audio files in seconds. double elapsed; // Time it took us to process it. @@ -207,49 +230,6 @@ int main (int argc, char *argv[]) my_audio_config.adev[0].samples_per_sec = DEFAULT_SAMPLES_PER_SEC; my_audio_config.adev[0].bits_per_sample = DEFAULT_BITS_PER_SAMPLE; - // Results v0.9: - // - // fix_bits = 0 971 packets, 69 sec - // fix_bits = SINGLE 990 64 - // fix_bits = DOUBLE 992 65 - // fix_bits = TRIPLE 992 67 - // fix_bits = TWO_SEP 1004 476 - - // Essentially no difference in time for those with order N time. - // Time increases greatly for the one with order N^2 time. - - - // Results: version 1.1, decoder C, my_audio_config.fix_bits = RETRY_MAX - 1 - // - // 971 NONE - // +19 SINGLE - // +2 DOUBLE - // +12 TWO_SEP - // +3 REMOVE_MANY - // ---- - // 1007 Total in 1008 sec. More than twice as long as earlier version. - - // Results: version 1.1, decoders ABC, my_audio_config.fix_bits = RETRY_MAX - 1 - // - // 976 NONE - // +21 SINGLE - // +1 DOUBLE - // +22 TWO_SEP - // +1 MANY - // +3 REMOVE_MANY - // ---- - // 1024 Total in 2042 sec. - // About 34 minutes of CPU time for a roughly 40 minute CD. - // Many computers wouldn't be able to keep up. - - // The SINGLE and TWO_SEP techniques are the most effective. - // Should we reorder the enum values so that TWO_SEP - // comes after SINGLE? That way "FIX_BITS 2" would - // use the two most productive techniques and not waste - // time on the others. People with plenty of CPU power - // to spare can still specify larger numbers for the other - // techniques with less return on investment. - for (channel=0; channel MAX_BAUD) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Use a more reasonable bit rate in range of %d - %d.\n", MIN_BAUD, MAX_BAUD); - exit (EXIT_FAILURE); - } - - /* We have similar logic in direwolf.c, config.c, gen_packets.c, and atest.c, */ - /* that need to be kept in sync. Maybe it could be a common function someday. */ - - if (my_audio_config.achan[0].baud < 600) { - my_audio_config.achan[0].modem_type = MODEM_AFSK; - my_audio_config.achan[0].mark_freq = 1600; - my_audio_config.achan[0].space_freq = 1800; - strlcpy (my_audio_config.achan[0].profiles, "D", sizeof(my_audio_config.achan[0].profiles)); - } - else if (my_audio_config.achan[0].baud < 1800) { - my_audio_config.achan[0].modem_type = MODEM_AFSK; - my_audio_config.achan[0].mark_freq = DEFAULT_MARK_FREQ; - my_audio_config.achan[0].space_freq = DEFAULT_SPACE_FREQ; - // Should default to E+ or something similar later. + /* Also implies modem type based on speed. */ + /* Special cases AIS, EAS rather than number. */ + if (strcasecmp(optarg, "AIS") == 0) { + B_opt = 0xA15A15; // See special case below. } - else if (my_audio_config.achan[0].baud < 3600) { - my_audio_config.achan[0].modem_type = MODEM_QPSK; - my_audio_config.achan[0].mark_freq = 0; - my_audio_config.achan[0].space_freq = 0; - strlcpy (my_audio_config.achan[0].profiles, "", sizeof(my_audio_config.achan[0].profiles)); - dw_printf ("Using V.26 QPSK rather than AFSK.\n"); - } - else if (my_audio_config.achan[0].baud < 7200) { - my_audio_config.achan[0].modem_type = MODEM_8PSK; - my_audio_config.achan[0].mark_freq = 0; - my_audio_config.achan[0].space_freq = 0; - strlcpy (my_audio_config.achan[0].profiles, "", sizeof(my_audio_config.achan[0].profiles)); - dw_printf ("Using V.27 8PSK rather than AFSK.\n"); + else if (strcasecmp(optarg, "EAS") == 0) { + B_opt = 0xEA5EA5; // See special case below. } else { - my_audio_config.achan[0].modem_type = MODEM_SCRAMBLE; - my_audio_config.achan[0].mark_freq = 0; - my_audio_config.achan[0].space_freq = 0; - strlcpy (my_audio_config.achan[0].profiles, " ", sizeof(my_audio_config.achan[0].profiles)); // avoid getting default later. - dw_printf ("Using scrambled baseband signal rather than AFSK.\n"); + B_opt = atoi(optarg); } break; + case 'g': /* -G Force G3RUH regardless of speed. */ + + g_opt = 1; + break; + + case 'j': /* -j V.26 compatible with earlier direwolf. */ + + j_opt = 1; + break; + + case 'J': /* -J V.26 compatible with MFJ-2400. */ + + J_opt = 1; + break; + case 'P': /* -P for modem profile. */ - dw_printf ("Demodulator profile set to \"%s\"\n", optarg); - strlcpy (my_audio_config.achan[0].profiles, optarg, sizeof(my_audio_config.achan[0].profiles)); + // Wait until after other options processed. + strlcpy (P_opt, optarg, sizeof(P_opt)); break; case 'D': /* -D reduce sampling rate for lower CPU usage. */ @@ -367,7 +323,24 @@ int main (int argc, char *argv[]) my_audio_config.achan[0].decimate = decimate; break; - case 'F': /* -D set "fix bits" level. */ + case 'U': /* -U upsample for G3RUH to improve performance */ + /* when the sample rate to baud ratio is low. */ + /* Actually it is set automatically and this will */ + /* override the normal calculation. */ + + upsample = atoi(optarg); + + dw_printf ("Multiply audio sample rate by %d\n", upsample); + if (upsample < 1 || upsample > 4) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Unreasonable value for -U.\n"); + exit (EXIT_FAILURE); + } + dw_printf ("Multiply audio sample rate by %d\n", upsample); + my_audio_config.achan[0].upsample = upsample; + break; + + case 'F': /* -F set "fix bits" level. */ my_audio_config.achan[0].fix_bits = atoi(optarg); @@ -403,6 +376,28 @@ int main (int argc, char *argv[]) decode_only = 2; break; + case 'h': /* Hexadecimal display. */ + + h_opt = 1; + break; + + case 'e': /* Receive Bit Error Rate (BER). */ + + my_audio_config.recv_ber = atof(optarg); + break; + + case 'd': /* Debug message options. */ + + for (char *p=optarg; *p!='\0'; p++) { + switch (*p) { + case 'x': d_x_opt++; break; // FX.25 + case 'o': d_o_opt++; break; // DCD output control + case '2': d_2_opt++; break; // IL2P debug out + default: break; + } + } + break; + case '?': /* Unknown option message was already printed. */ @@ -418,6 +413,124 @@ int main (int argc, char *argv[]) } } +/* + * Set modem type based on data rate. + * (Could be overridden by -g, -j, or -J later.) + */ + /* 300 implies 1600/1800 AFSK. */ + /* 1200 implies 1200/2200 AFSK. */ + /* 2400 implies V.26 QPSK. */ + /* 4800 implies V.27 8PSK. */ + /* 9600 implies G3RUH baseband scrambled. */ + + my_audio_config.achan[0].baud = B_opt; + + + /* We have similar logic in direwolf.c, config.c, gen_packets.c, and atest.c, */ + /* that need to be kept in sync. Maybe it could be a common function someday. */ + + if (my_audio_config.achan[0].baud == 100) { // What was this for? + my_audio_config.achan[0].modem_type = MODEM_AFSK; + my_audio_config.achan[0].mark_freq = 1615; + my_audio_config.achan[0].space_freq = 1785; + } + else if (my_audio_config.achan[0].baud < 600) { // e.g. HF SSB packet + my_audio_config.achan[0].modem_type = MODEM_AFSK; + my_audio_config.achan[0].mark_freq = 1600; + my_audio_config.achan[0].space_freq = 1800; + // Previously we had a "D" which was fine tuned for 300 bps. + // In v1.7, it's not clear if we should use "B" or just stick with "A". + } + else if (my_audio_config.achan[0].baud < 1800) { // common 1200 + my_audio_config.achan[0].modem_type = MODEM_AFSK; + my_audio_config.achan[0].mark_freq = DEFAULT_MARK_FREQ; + my_audio_config.achan[0].space_freq = DEFAULT_SPACE_FREQ; + } + else if (my_audio_config.achan[0].baud < 3600) { + my_audio_config.achan[0].modem_type = MODEM_QPSK; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + strlcpy (my_audio_config.achan[0].profiles, "", sizeof(my_audio_config.achan[0].profiles)); + } + else if (my_audio_config.achan[0].baud < 7200) { + my_audio_config.achan[0].modem_type = MODEM_8PSK; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + strlcpy (my_audio_config.achan[0].profiles, "", sizeof(my_audio_config.achan[0].profiles)); + } + else if (my_audio_config.achan[0].baud == 0xA15A15) { // Hack for different use of 9600 + my_audio_config.achan[0].modem_type = MODEM_AIS; + my_audio_config.achan[0].baud = 9600; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + strlcpy (my_audio_config.achan[0].profiles, " ", sizeof(my_audio_config.achan[0].profiles)); // avoid getting default later. + } + else if (my_audio_config.achan[0].baud == 0xEA5EA5) { + my_audio_config.achan[0].modem_type = MODEM_EAS; + my_audio_config.achan[0].baud = 521; // Actually 520.83 but we have an integer field here. + // Will make more precise in afsk demod init. + my_audio_config.achan[0].mark_freq = 2083; // Actually 2083.3 - logic 1. + my_audio_config.achan[0].space_freq = 1563; // Actually 1562.5 - logic 0. + strlcpy (my_audio_config.achan[0].profiles, "A", sizeof(my_audio_config.achan[0].profiles)); + } + else { + my_audio_config.achan[0].modem_type = MODEM_SCRAMBLE; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + strlcpy (my_audio_config.achan[0].profiles, " ", sizeof(my_audio_config.achan[0].profiles)); // avoid getting default later. + } + + if (my_audio_config.achan[0].baud < MIN_BAUD || my_audio_config.achan[0].baud > MAX_BAUD) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Use a more reasonable bit rate in range of %d - %d.\n", MIN_BAUD, MAX_BAUD); + exit (EXIT_FAILURE); + } + +/* + * -g option means force g3RUH regardless of speed. + */ + + if (g_opt) { + my_audio_config.achan[0].modem_type = MODEM_SCRAMBLE; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + strlcpy (my_audio_config.achan[0].profiles, " ", sizeof(my_audio_config.achan[0].profiles)); // avoid getting default later. + } + +/* + * We have two different incompatible flavors of V.26. + */ + if (j_opt) { + + // V.26 compatible with earlier versions of direwolf. + // Example: -B 2400 -j or simply -j + + my_audio_config.achan[0].v26_alternative = V26_A; + my_audio_config.achan[0].modem_type = MODEM_QPSK; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + my_audio_config.achan[0].baud = 2400; + strlcpy (my_audio_config.achan[0].profiles, "", sizeof(my_audio_config.achan[0].profiles)); + } + if (J_opt) { + + // V.26 compatible with MFJ and maybe others. + // Example: -B 2400 -J or simply -J + + my_audio_config.achan[0].v26_alternative = V26_B; + my_audio_config.achan[0].modem_type = MODEM_QPSK; + my_audio_config.achan[0].mark_freq = 0; + my_audio_config.achan[0].space_freq = 0; + my_audio_config.achan[0].baud = 2400; + strlcpy (my_audio_config.achan[0].profiles, "", sizeof(my_audio_config.achan[0].profiles)); + } + + // Needs to be after -B, -j, -J. + if (strlen(P_opt) > 0) { + dw_printf ("Demodulator profile set to \"%s\"\n", P_opt); + strlcpy (my_audio_config.achan[0].profiles, P_opt, sizeof(my_audio_config.achan[0].profiles)); + } + memcpy (&my_audio_config.achan[1], &my_audio_config.achan[0], sizeof(my_audio_config.achan[0])); @@ -427,6 +540,13 @@ int main (int argc, char *argv[]) usage (); } + fx25_init (d_x_opt); + il2p_init (d_2_opt); + + start_time = dtime_now(); + + while (optind < argc) { + fp = fopen(argv[optind], "rb"); if (fp == NULL) { text_color_set(DW_COLOR_ERROR); @@ -435,8 +555,6 @@ int main (int argc, char *argv[]) exit (EXIT_FAILURE); } - start_time = dtime_now(); - /* * Read the file header. * Doesn't handle all possible cases but good enough for our purposes. @@ -501,25 +619,31 @@ int main (int argc, char *argv[]) my_audio_config.adev[0].bits_per_sample = format.wbitspersample; my_audio_config.adev[0].num_channels = format.nchannels; - my_audio_config.achan[0].valid = 1; - if (format.nchannels == 2) my_audio_config.achan[1].valid = 1; + my_audio_config.chan_medium[0] = MEDIUM_RADIO; + if (format.nchannels == 2) { + my_audio_config.chan_medium[1] = MEDIUM_RADIO; + } text_color_set(DW_COLOR_INFO); dw_printf ("%d samples per second. %d bits per sample. %d audio channels.\n", my_audio_config.adev[0].samples_per_sec, my_audio_config.adev[0].bits_per_sample, my_audio_config.adev[0].num_channels); - duration = (double) wav_data.datasize / + one_filetime = (double) wav_data.datasize / ((my_audio_config.adev[0].bits_per_sample / 8) * my_audio_config.adev[0].num_channels * my_audio_config.adev[0].samples_per_sec); + total_filetime += one_filetime; + dw_printf ("%d audio bytes in file. Duration = %.1f seconds.\n", (int)(wav_data.datasize), - duration); + one_filetime); dw_printf ("Fix Bits level = %d\n", my_audio_config.achan[0].fix_bits); /* * Initialize the AFSK demodulator and HDLC decoder. + * Needs to be done for each file because they could have different sample rates. */ multi_modem_init (&my_audio_config); + packets_decoded_one = 0; e_o_f = 0; @@ -572,17 +696,27 @@ int main (int argc, char *argv[]) } #endif + dw_printf ("%d from %s\n", packets_decoded_one, argv[optind]); + packets_decoded_total += packets_decoded_one; + + fclose (fp); + optind++; + } elapsed = dtime_now() - start_time; - dw_printf ("%d packets decoded in %.3f seconds. %.1f x realtime\n", packets_decoded, elapsed, duration/elapsed); + dw_printf ("%d packets decoded in %.3f seconds. %.1f x realtime\n", packets_decoded_total, elapsed, total_filetime/elapsed); + if (d_o_opt) { + dw_printf ("DCD count = %d\n", dcd_count); + dw_printf ("DCD missing errors = %d\n", dcd_missing_errors); + } - if (error_if_less_than != -1 && packets_decoded < error_if_less_than) { + if (error_if_less_than != -1 && packets_decoded_total < error_if_less_than) { text_color_set(DW_COLOR_ERROR); dw_printf ("\n * * * TEST FAILED: number decoded is less than %d * * * \n", error_if_less_than); exit (EXIT_FAILURE); } - if (error_if_greater_than != -1 && packets_decoded > error_if_greater_than) { + if (error_if_greater_than != -1 && packets_decoded_total > error_if_greater_than) { text_color_set(DW_COLOR_ERROR); dw_printf ("\n * * * TEST FAILED: number decoded is greater than %d * * * \n", error_if_greater_than); exit (EXIT_FAILURE); @@ -619,42 +753,22 @@ int audio_get (int a) -/* - * Rather than queuing up frames with bad FCS, - * try to fix them immediately. - */ - -void rdq_append (rrbb_t rrbb) -{ - int chan, subchan, slice; - alevel_t alevel; - - - chan = rrbb_get_chan(rrbb); - subchan = rrbb_get_subchan(rrbb); - slice = rrbb_get_slice(rrbb); - alevel = rrbb_get_audio_level(rrbb); - - hdlc_rec2_try_to_fix_later (rrbb, chan, subchan, slice, alevel); - rrbb_delete (rrbb); -} - - /* * This is called when we have a good frame. */ -void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, char *spectrum) +void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, fec_type_t fec_type, retry_t retries, char *spectrum) { char stemp[500]; unsigned char *pinfo; int info_len; int h; - char heard[AX25_MAX_ADDR_LEN]; + char heard[2 * AX25_MAX_ADDR_LEN + 20]; char alevel_text[AX25_ALEVEL_TO_TEXT_SIZE]; - packets_decoded++; + packets_decoded_one++; + if ( ! hdlc_rec_data_detect_any(chan)) dcd_missing_errors++; ax25_format_addrs (pp, stemp); @@ -680,7 +794,7 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev text_color_set(DW_COLOR_DEBUG); dw_printf ("\n"); - dw_printf("DECODED[%d] ", packets_decoded ); + dw_printf("DECODED[%d] ", packets_decoded_one ); /* Insert time stamp relative to start of file. */ @@ -688,32 +802,55 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev int min = (int)(sec / 60.); sec -= min * 60; - dw_printf ("%d:%07.4f ", min, sec); + dw_printf ("%d:%06.3f ", min, sec); if (h != AX25_SOURCE) { dw_printf ("Digipeater "); } ax25_alevel_to_text (alevel, alevel_text); - if (my_audio_config.achan[chan].fix_bits == RETRY_NONE && my_audio_config.achan[chan].passall == 0) { - dw_printf ("%s audio level = %s %s\n", heard, alevel_text, spectrum); + /* As suggested by KJ4ERJ, if we are receiving from */ + /* WIDEn-0, it is quite likely (but not guaranteed), that */ + /* we are actually hearing the preceding station in the path. */ + + if (h >= AX25_REPEATER_2 && + strncmp(heard, "WIDE", 4) == 0 && + isdigit(heard[4]) && + heard[5] == '\0') { + + char probably_really[AX25_MAX_ADDR_LEN]; + ax25_get_addr_with_ssid(pp, h-1, probably_really); + + strlcat (heard, " (probably ", sizeof(heard)); + strlcat (heard, probably_really, sizeof(heard)); + strlcat (heard, ")", sizeof(heard)); } - else { - dw_printf ("%s audio level = %s [%s] %s\n", heard, alevel_text, retry_text[(int)retries], spectrum); + + switch (fec_type) { + + case fec_type_fx25: + dw_printf ("%s audio level = %s FX.25 %s\n", heard, alevel_text, spectrum); + break; + + case fec_type_il2p: + dw_printf ("%s audio level = %s IL2P %s\n", heard, alevel_text, spectrum); + break; + + case fec_type_none: + default: + if (my_audio_config.achan[chan].fix_bits == RETRY_NONE && my_audio_config.achan[chan].passall == 0) { + // No fix_bits or passall specified. + dw_printf ("%s audio level = %s %s\n", heard, alevel_text, spectrum); + } + else { + assert (retries >= RETRY_NONE && retries <= RETRY_MAX); // validate array index. + dw_printf ("%s audio level = %s [%s] %s\n", heard, alevel_text, retry_text[(int)retries], spectrum); + } + break; } #endif -//#if defined(EXPERIMENT_G) || defined(EXPERIMENT_H) -// int j; -// -// for (j=0; j #include #include +#include #if USE_ALSA #include -#else -#include -#ifdef __OpenBSD__ -#include +#elif USE_SNDIO +#include +#include #else #include #endif -#endif #include "audio.h" @@ -111,6 +110,9 @@ static struct adev_s { int bytes_per_frame; /* number of bytes for a sample from all channels. */ /* e.g. 4 for stereo 16 bit. */ +#elif USE_SNDIO + struct sio_hdl *sndio_in_handle; + struct sio_hdl *sndio_out_handle; #else int oss_audio_device_fd; /* Single device, both directions. */ @@ -141,6 +143,9 @@ static struct adev_s { #if USE_ALSA static int set_alsa_params (int a, snd_pcm_t *handle, struct audio_s *pa, char *name, char *dir); //static void alsa_select_device (char *pick_dev, int direction, char *result); +#elif USE_SNDIO +static int set_sndio_params (int a, struct sio_hdl *handle, struct audio_s *pa, char *devname, char *inout); +static int poll_sndio (struct sio_hdl *hdl, int events); #else static int set_oss_params (int a, int fd, struct audio_s *pa); #endif @@ -201,7 +206,7 @@ static int calcbufsize(int rate, int chans, int bits) * more restrictive in its capabilities. * It might say, the best I can do is mono, 8 bit, 8000/sec. * - * The sofware modem must use this ACTUAL information + * The software modem must use this ACTUAL information * that the device is supplying, that could be different * than what the user specified. * @@ -212,7 +217,9 @@ static int calcbufsize(int rate, int chans, int bits) int audio_open (struct audio_s *pa) { +#if !USE_SNDIO int err; +#endif int chan; int a; char audio_in_name[30]; @@ -224,7 +231,11 @@ int audio_open (struct audio_s *pa) memset (adev, 0, sizeof(adev)); for (a=0; aadev[a].adevice_in, O_RDWR); @@ -430,6 +463,10 @@ int audio_open (struct audio_s *pa) text_color_set(DW_COLOR_ERROR); dw_printf ("Could not open audio device %s for output\n%s\n", audio_out_name, snd_strerror(err)); + if (err == -EBUSY) { + dw_printf ("This means that some other application is using that device.\n"); + dw_printf ("The solution is to identify that other application and stop it.\n"); + } return (-1); } @@ -439,6 +476,27 @@ int audio_open (struct audio_s *pa) return (-1); } +#elif USE_SNDIO + adev[a].sndio_out_handle = sio_open (audio_out_name, SIO_PLAY, 0); + if (adev[a].sndio_out_handle == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not open audio device %s for output\n", + audio_out_name); + return (-1); + } + + adev[a].outbuf_size_in_bytes = set_sndio_params (a, adev[a].sndio_out_handle, pa, audio_out_name, "output"); + + if (adev[a].inbuf_size_in_bytes <= 0 || adev[a].outbuf_size_in_bytes <= 0) { + return (-1); + } + + if (!sio_start (adev[a].sndio_out_handle)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not start audio device %s for output\n", + audio_out_name); + return (-1); + } #endif /* @@ -675,6 +733,112 @@ static int set_alsa_params (int a, snd_pcm_t *handle, struct audio_s *pa, char * } /* end alsa_set_params */ +#elif USE_SNDIO + +/* + * Set parameters for sound card. (sndio) + * + * See /usr/include/sndio.h for details. + */ + +static int set_sndio_params (int a, struct sio_hdl *handle, struct audio_s *pa, char *devname, char *inout) +{ + + struct sio_par q, r; + + /* Signed 16 bit little endian or unsigned 8 bit. */ + sio_initpar (&q); + q.bits = pa->adev[a].bits_per_sample; + q.bps = (q.bits + 7) / 8; + q.sig = (q.bits == 8) ? 0 : 1; + q.le = 1; /* always little endian */ + q.msb = 0; /* LSB aligned */ + q.rchan = q.pchan = pa->adev[a].num_channels; + q.rate = pa->adev[a].samples_per_sec; + q.xrun = SIO_IGNORE; + q.appbufsz = calcbufsize(pa->adev[a].samples_per_sec, pa->adev[a].num_channels, pa->adev[a].bits_per_sample); + + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("suggest buffer size %d bytes for %s %s.\n", + q.appbufsz, devname, inout); +#endif + + /* challenge new setting */ + if (!sio_setpar (handle, &q)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not set hardware parameter for %s %s.\n", + devname, inout); + return (-1); + } + + /* get response */ + if (!sio_getpar (handle, &r)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not obtain current hardware setting for %s %s.\n", + devname, inout); + return (-1); + } + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("audio buffer size %d bytes for %s %s.\n", + r.appbufsz, devname, inout); +#endif + if (q.rate != r.rate) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Asked for %d samples/sec but got %d for %s %s.", + pa->adev[a].samples_per_sec, r.rate, devname, inout); + pa->adev[a].samples_per_sec = r.rate; + } + + /* not supported */ + if (q.bits != r.bits || q.bps != r.bps || q.sig != r.sig || + (q.bits > 8 && q.le != r.le) || + (*inout == 'o' && q.pchan != r.pchan) || + (*inout == 'i' && q.rchan != r.rchan)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Unsupported format for %s %s.\n", devname, inout); + return (-1); + } + + return r.appbufsz; + +} /* end set_sndio_params */ + +static int poll_sndio (struct sio_hdl *hdl, int events) +{ + struct pollfd *pfds; + int nfds, revents; + + nfds = sio_nfds (hdl); + pfds = alloca (nfds * sizeof(struct pollfd)); + + do { + nfds = sio_pollfd (hdl, pfds, events); + if (nfds < 1) { + /* no need to wait */ + return (0); + } + if (poll (pfds, nfds, -1) < 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("poll %d\n", errno); + return (-1); + } + revents = sio_revents (hdl, pfds); + } while (!(revents & (events | POLLHUP))); + + /* unrecoverable error occurred */ + if (revents & POLLHUP) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("waited for %s, POLLHUP received\n", (events & POLLIN) ? "POLLIN" : "POLLOUT"); + return (-1); + } + + return (0); +} + #else @@ -769,7 +933,7 @@ static int set_oss_params (int a, int fd, struct audio_s *pa) * This was long ago under different conditions. * Should study this again some day. * - * Your milage may vary. + * Your mileage may vary. */ err = ioctl (fd, SNDCTL_DSP_GETBLKSIZE, &ossbuf_size_in_bytes); if (err == -1) { @@ -842,7 +1006,9 @@ __attribute__((hot)) int audio_get (int a) { int n; +#if USE_ALSA int retries = 0; +#endif #if STATISTICS /* Gather numbers for read from audio device. */ @@ -921,11 +1087,24 @@ int audio_get (int a) /* Error */ // TODO: Needs more study and testing. - // TODO: print n. should snd_strerror use n or errno? - // Audio input device error: Unknown error + // Only expected error conditions: + // -EBADFD PCM is not in the right state (SND_PCM_STATE_PREPARED or SND_PCM_STATE_RUNNING) + // -EPIPE an overrun occurred + // -ESTRPIPE a suspend event occurred (stream is suspended and waiting for an application recovery) + + // Data overrun is displayed as "broken pipe" which seems a little misleading. + // Add our own message which says something about CPU being too slow. text_color_set(DW_COLOR_ERROR); - dw_printf ("Audio input device %d error: %s\n", a, snd_strerror(n)); + dw_printf ("Audio input device %d error code %d: %s\n", a, n, snd_strerror(n)); + + if (n == (-EPIPE)) { + dw_printf ("This is most likely caused by the CPU being too slow to keep up with the audio stream.\n"); + dw_printf ("Use the \"top\" command, in another command window, to look at CPU usage.\n"); + dw_printf ("This might be a temporary condition so we will attempt to recover a few times before giving up.\n"); + dw_printf ("If using a very slow CPU, try reducing the CPU load by using -P- command\n"); + dw_printf ("line option for 9600 bps or -D3 for slower AFSK .\n"); + } audio_stats (a, save_audio_config_p->adev[a].num_channels, @@ -959,7 +1138,28 @@ int audio_get (int a) } -#else /* end ALSA, begin OSS */ +#elif USE_SNDIO + + while (adev[a].inbuf_next >= adev[a].inbuf_len) { + + assert (adev[a].sndio_in_handle != NULL); + if (poll_sndio (adev[a].sndio_in_handle, POLLIN) < 0) { + adev[a].inbuf_len = 0; + adev[a].inbuf_next = 0; + return (-1); + } + + n = sio_read (adev[a].sndio_in_handle, adev[a].inbuf_ptr, adev[a].inbuf_size_in_bytes); + adev[a].inbuf_len = n; + adev[a].inbuf_next = 0; + + audio_stats (a, + save_audio_config_p->adev[a].num_channels, + n / (save_audio_config_p->adev[a].num_channels * save_audio_config_p->adev[a].bits_per_sample / 8), + save_audio_config_p->statistics_interval); + } + +#else /* begin OSS */ /* Fixed in 1.2. This was formerly outside of the switch */ /* so the OSS version did not process stdin or UDP. */ @@ -1193,6 +1393,17 @@ int audio_flush (int a) snd_pcm_recover (adev[a].audio_out_handle, k, 1); } + else if (k == -ESTRPIPE) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Driver suspended, recovering\n"); + snd_pcm_recover(adev[a].audio_out_handle, k, 1); + } + else if (k == -EBADFD) { + k = snd_pcm_prepare (adev[a].audio_out_handle); + if(k < 0) { + dw_printf ("Error preparing after bad state: %s\n", snd_strerror(k)); + } + } else if (k < 0) { text_color_set(DW_COLOR_ERROR); dw_printf ("Audio write error: %s\n", snd_strerror(k)); @@ -1200,7 +1411,10 @@ int audio_flush (int a) /* Some other error condition. */ /* Try again. What do we have to lose? */ - snd_pcm_recover (adev[a].audio_out_handle, k, 1); + k = snd_pcm_prepare (adev[a].audio_out_handle); + if(k < 0) { + dw_printf ("Error preparing after error: %s\n", snd_strerror(k)); + } } else if (k != adev[a].outbuf_len / adev[a].bytes_per_frame) { text_color_set(DW_COLOR_ERROR); @@ -1225,6 +1439,37 @@ int audio_flush (int a) adev[a].outbuf_len = 0; return (-1); +#elif USE_SNDIO + + int k; + unsigned char *ptr; + int len; + + ptr = adev[a].outbuf_ptr; + len = adev[a].outbuf_len; + + while (len > 0) { + assert (adev[a].sndio_out_handle != NULL); + if (poll_sndio (adev[a].sndio_out_handle, POLLOUT) < 0) { + text_color_set(DW_COLOR_ERROR); + perror("Can't write to audio device"); + adev[a].outbuf_len = 0; + return (-1); + } + + k = sio_write (adev[a].sndio_out_handle, ptr, len); +#if DEBUGx + text_color_set(DW_COLOR_DEBUG); + dw_printf ("audio_flush(): write %d returns %d\n", len, k); + fflush (stdout); +#endif + ptr += k; + len -= k; + } + + adev[a].outbuf_len = 0; + return (0); + #else /* OSS */ int k; @@ -1295,7 +1540,7 @@ int audio_flush (int a) * (3) Call this function, which might or might not wait long enough. * (4) Add (1) and (2) resulting in when PTT should be turned off. * (5) Take difference between current time and desired PPT off time - * and wait for additoinal time if required. + * and wait for additional time if required. * *----------------------------------------------------------------*/ @@ -1326,6 +1571,10 @@ void audio_wait (int a) * Either way, the caller will now compensate for it. */ +#elif USE_SNDIO + + poll_sndio (adev[a].sndio_out_handle, POLLOUT); + #else assert (adev[a].oss_audio_device_fd > 0); @@ -1371,7 +1620,22 @@ int audio_close (void) snd_pcm_close (adev[a].audio_in_handle); snd_pcm_close (adev[a].audio_out_handle); - + + adev[a].audio_in_handle = adev[a].audio_out_handle = NULL; + +#elif USE_SNDIO + + if (adev[a].sndio_in_handle != NULL && adev[a].sndio_out_handle != NULL) { + + audio_wait (a); + + sio_stop (adev[a].sndio_in_handle); + sio_stop (adev[a].sndio_out_handle); + sio_close (adev[a].sndio_in_handle); + sio_close (adev[a].sndio_out_handle); + + adev[a].sndio_in_handle = adev[a].sndio_out_handle = NULL; + #else if (adev[a].oss_audio_device_fd > 0) { diff --git a/audio.h b/src/audio.h similarity index 63% rename from audio.h rename to src/audio.h index 71d37517..cb5ca94e 100644 --- a/audio.h +++ b/src/audio.h @@ -18,7 +18,7 @@ #include "direwolf.h" /* for MAX_CHANS used throughout the application. */ #include "ax25_pad.h" /* for AX25_MAX_ADDR_LEN */ - +#include "version.h" /* @@ -29,8 +29,9 @@ enum ptt_method_e { PTT_METHOD_NONE, /* VOX or no transmit. */ PTT_METHOD_SERIAL, /* Serial port RTS or DTR. */ PTT_METHOD_GPIO, /* General purpose I/O, Linux only. */ - PTT_METHOD_LPT, /* Parallel printer port, Linux only. */ - PTT_METHOD_HAMLIB }; /* HAMLib, Linux only. */ + PTT_METHOD_LPT, /* Parallel printer port, Linux only. */ + PTT_METHOD_HAMLIB, /* HAMLib, Linux only. */ + PTT_METHOD_CM108 }; /* GPIO pin of CM108/CM119/etc. Linux only. */ typedef enum ptt_method_e ptt_method_t; @@ -52,6 +53,13 @@ typedef enum retry_e { RETRY_INVERT_TWO_SEP=4, RETRY_MAX = 5} retry_t; +// Type of communication medium associated with the channel. + +enum medium_e { MEDIUM_NONE = 0, // Channel is not valid for use. + MEDIUM_RADIO, // Internal modem for radio. + MEDIUM_IGATE, // Access IGate as ordinary channel. + MEDIUM_NETTNC }; // Remote network TNC. (possible future) + typedef enum sanity_e { SANITY_APRS, SANITY_AX25, SANITY_NONE } sanity_t; @@ -64,7 +72,7 @@ struct audio_s { struct adev_param_s { - /* Properites of the sound device. */ + /* Properties of the sound device. */ int defined; /* Was device defined? */ /* First one defaults to yes. */ @@ -94,22 +102,72 @@ struct audio_s { /* This is the probability, in per cent, of randomly corrupting it. */ /* Normally this is 0. 25 would mean corrupt it 25% of the time. */ - int recv_error_rate; /* Similar but the % probablity of dropping a received frame. */ + int recv_error_rate; /* Similar but the % probability of dropping a received frame. */ + float recv_ber; /* Receive Bit Error Rate (BER). */ + /* Probability of inverting a bit coming out of the modem. */ - /* Properties for each audio channel, common to receive and transmit. */ - /* Can be different for each radio channel. */ + //int fx25_xmit_enable; /* Enable transmission of FX.25. */ + /* See fx25_init.c for explanation of values. */ + /* Initially this applies to all channels. */ + /* This should probably be per channel. One step at a time. */ + /* v1.7 - replaced by layer2_xmit==LAYER2_FX25 */ + int fx25_auto_enable; /* Turn on FX.25 for current connected mode session */ + /* under poor conditions. */ + /* Set to 0 to disable feature. */ + /* I put it here, rather than with the rest of the link layer */ + /* parameters because it is really a part of the HDLC layer */ + /* and is part of the KISS TNC functionality rather than our data link layer. */ + /* Future: not used yet. */ - struct achan_param_s { - int valid; /* Is this channel valid? */ + char timestamp_format[40]; /* -T option */ + /* Precede received & transmitted frames with timestamp. */ + /* Command line option uses "strftime" format string. */ + + + + /* originally a "channel" was always connected to an internal modem. */ + /* In version 1.6, this is generalized so that a channel (as seen by client application) */ + /* can be connected to something else. Initially, this will allow application */ + /* access to the IGate. Later we might have network TNCs or other internal functions. */ + + // Properties for all channels. + + enum medium_e chan_medium[MAX_TOTAL_CHANS]; + // MEDIUM_NONE for invalid. + // MEDIUM_RADIO for internal modem. (only possibility earlier) + // MEDIUM_IGATE allows application access to IGate. + // MEDIUM_NETTNC for external TNC via TCP. + + int igate_vchannel; /* Virtual channel mapped to APRS-IS. */ + /* -1 for none. */ + /* Redundant but it makes things quicker and simpler */ + /* than always searching thru above. */ + + /* Properties for each radio channel, common to receive and transmit. */ + /* Can be different for each radio channel. */ + + struct achan_param_s { + // Currently, we have a fixed mapping from audio sources to channel. + // + // ADEVICE CHANNEL (mono) (stereo) + // 0 0 0, 1 + // 1 2 2, 3 + // 2 4 4, 5 + // + // A future feauture might allow the user to specify a different audio source. + // This would allow multiple modems (with associated channel) to share an audio source. + // int audio_source; // Default would be [0,1,2,3,4,5] + + // What else should be moved out of structure and enlarged when NETTNC is implemented. ??? char mycall[AX25_MAX_ADDR_LEN]; /* Call associated with this radio channel. */ /* Could all be the same or different. */ - enum modem_t { MODEM_AFSK, MODEM_BASEBAND, MODEM_SCRAMBLE, MODEM_QPSK, MODEM_8PSK, MODEM_OFF } modem_type; + enum modem_t { MODEM_AFSK, MODEM_BASEBAND, MODEM_SCRAMBLE, MODEM_QPSK, MODEM_8PSK, MODEM_OFF, MODEM_16_QAM, MODEM_64_QAM, MODEM_AIS, MODEM_EAS } modem_type; /* Usual AFSK. */ /* Baseband signal. Not used yet. */ @@ -117,6 +175,32 @@ struct audio_s { /* Might try MFJ-2400 / CCITT v.26 / Bell 201 someday. */ /* No modem. Might want this for DTMF only channel. */ + enum layer2_t { LAYER2_AX25 = 0, LAYER2_FX25, LAYER2_IL2P } layer2_xmit; + + // IL2P - New for version 1.7. + // New layer 2 with FEC. Much less overhead than FX.25 but no longer backward compatible. + // Only applies to transmit. + // Listening for FEC sync word should add negligible overhead so + // we leave reception enabled all the time as we do with FX.25. + // TODO: FX.25 should probably be put here rather than global for all channels. + + int fx25_strength; // Strength of FX.25 FEC. + // 16, 23, 64 for specific number of parity symbols. + // 1 for automatic selection based on frame size. + + int il2p_max_fec; // 1 for max FEC length, 0 for automatic based on size. + + int il2p_invert_polarity; // 1 means invert on transmit. Receive handles either automatically. + + enum v26_e { V26_UNSPECIFIED=0, V26_A, V26_B } v26_alternative; + + // Original implementation used alternative A for 2400 bbps PSK. + // Years later, we discover that MFJ-2400 used alternative B. + // It's likely the others did too. it also works a little better. + // Default to MFJ compatible and print warning if user did not + // pick one explicitly. + +#define V26_DEFAULT V26_B enum dtmf_decode_t { DTMF_DECODE_OFF, DTMF_DECODE_ON } dtmf_decode; @@ -133,8 +217,7 @@ struct audio_s { int decimate; /* Reduce AFSK sample rate by this factor to */ /* decrease computational requirements. */ - int interleave; /* If > 1, interleave samples among multiple decoders. */ - /* Quick hack for experiment. */ + int upsample; /* Upsample by this factor for G3RUH. */ int mark_freq; /* Two tones for AFSK modulation, in Hz. */ int space_freq; /* Standard tones are 1200 and 2200 for 1200 baud. */ @@ -191,10 +274,19 @@ struct audio_s { struct { - ptt_method_t ptt_method; /* none, serial port, GPIO, LPT, HAMLIB. */ + ptt_method_t ptt_method; /* none, serial port, GPIO, LPT, HAMLIB, CM108. */ - char ptt_device[20]; /* Serial device name for PTT. e.g. COM1 or /dev/ttyS0 */ + char ptt_device[128]; /* Serial device name for PTT. e.g. COM1 or /dev/ttyS0 */ /* Also used for HAMLIB. Could be host:port when model is 1. */ + /* For years, 20 characters was plenty then we start getting extreme names like this: */ + /* /dev/serial/by-id/usb-FTDI_Navigator__CAT___2nd_PTT__00000000-if00-port0 */ + /* /dev/serial/by-id/usb-Prolific_Technology_Inc._USB-Serial_Controller_D-if00-port0 */ + /* Issue 104, changed to 100 bytes in version 1.5. */ + + /* This same field is also used for CM108/CM119 GPIO PTT which will */ + /* have a name like /dev/hidraw1 for Linux or */ + /* \\?\hid#vid_0d8c&pid_0008&mi_03#8&39d3555&0&0000#{4d1e55b2-f16f-11cf-88cb-001111000030} */ + /* for Windows. Largest observed was 95 but add some extra to be safe. */ ptt_line_t ptt_line; /* Control line when using serial port. PTT_LINE_RTS, PTT_LINE_DTR. */ ptt_line_t ptt_line2; /* Optional second one: PTT_LINE_NONE when not used. */ @@ -202,16 +294,19 @@ struct audio_s { int out_gpio_num; /* GPIO number. Originally this was only for PTT. */ /* It is now more general. */ /* octrl array is indexed by PTT, DCD, or CONnected indicator. */ + /* For CM108/CM119, this should be in range of 1-8. */ -#define MAX_GPIO_NAME_LEN 16 // 12 would cover any case I've seen so this should be safe +#define MAX_GPIO_NAME_LEN 20 // 12 would cover any case I've seen so this should be safe char out_gpio_name[MAX_GPIO_NAME_LEN]; - /* orginally, gpio number NN was assumed to simply */ + /* originally, gpio number NN was assumed to simply */ /* have the name gpioNN but this turned out not to be */ /* the case for CubieBoard where it was longer. */ /* This is filled in by ptt_init so we don't have to */ /* recalculate it each time we access it. */ + /* This could probably be collapsed into ptt_device instead of being separate. */ + int ptt_lpt_bit; /* Bit number for parallel printer port. */ /* Bit 0 = pin 2, ..., bit 7 = pin 9. */ @@ -221,6 +316,8 @@ struct audio_s { #ifdef USE_HAMLIB int ptt_model; /* HAMLIB model. -1 for AUTO. 2 for rigctld. Others are radio model. */ + int ptt_rate; /* Serial port speed when using hamlib CAT control for PTT. */ + /* If zero, hamlib will come up with a default for pariticular rig. */ #endif } octrl[NUM_OCTYPES]; @@ -239,7 +336,7 @@ struct audio_s { int in_gpio_num; /* GPIO number */ char in_gpio_name[MAX_GPIO_NAME_LEN]; - /* orginally, gpio number NN was assumed to simply */ + /* originally, gpio number NN was assumed to simply */ /* have the name gpioNN but this turned out not to be */ /* the case for CubieBoard where it was longer. */ /* This is filled in by ptt_init so we don't have to */ @@ -253,7 +350,7 @@ struct audio_s { int dwait; /* First wait extra time for receiver squelch. */ /* Default 0 units of 10 mS each . */ - int slottime; /* Slot time in 10 mS units for persistance algorithm. */ + int slottime; /* Slot time in 10 mS units for persistence algorithm. */ /* Typical value is 10 meaning 100 milliseconds. */ int persist; /* Sets probability for transmitting after each */ @@ -270,7 +367,10 @@ struct audio_s { /* are done sending the data. This is to avoid */ /* dropping PTT too soon and chopping off the end */ /* of the frame. Again 10 mS units. */ - /* At this point, I'm thinking of 10 as the default. */ + /* At this point, I'm thinking of 10 (= 100 mS) as the default */ + /* because we're not quite sure when the soundcard audio stops. */ + + int fulldup; /* Full Duplex. */ } achan[MAX_CHANS]; @@ -282,15 +382,18 @@ struct audio_s { }; -#if __WIN32__ || __APPLE__ +#if __WIN32__ #define DEFAULT_ADEVICE "" /* Windows: Empty string = default audio device. */ -#else -#if USE_ALSA +#elif __APPLE__ +#define DEFAULT_ADEVICE "" /* Mac OSX: Empty string = default audio device. */ +#elif USE_ALSA #define DEFAULT_ADEVICE "default" /* Use default device for ALSA. */ +#elif USE_SNDIO +#define DEFAULT_ADEVICE "default" /* Use default device for sndio. */ #else -#define DEFAULT_ADEVICE "/dev/dsp" /* First audio device for OSS. */ +#define DEFAULT_ADEVICE "/dev/dsp" /* First audio device for OSS. (FreeBSD) */ #endif -#endif + /* @@ -323,7 +426,8 @@ struct audio_s { #define DEFAULT_BITS_PER_SAMPLE 16 -#define DEFAULT_FIX_BITS RETRY_INVERT_SINGLE +#define DEFAULT_FIX_BITS RETRY_NONE // Interesting research project but even a single bit fix up + // will occasionally let corrupted packets through. /* * Standard for AFSK on VHF FM. @@ -353,11 +457,11 @@ struct audio_s { */ #define DEFAULT_DWAIT 0 -#define DEFAULT_SLOTTIME 10 +#define DEFAULT_SLOTTIME 10 // *10mS = 100mS #define DEFAULT_PERSIST 63 -#define DEFAULT_TXDELAY 30 -#define DEFAULT_TXTAIL 10 - +#define DEFAULT_TXDELAY 30 // *10mS = 300mS +#define DEFAULT_TXTAIL 10 // *10mS = 100mS +#define DEFAULT_FULLDUP 0 // false = half duplex /* * Note that we have two versions of these in audio.c and audio_win.c. diff --git a/audio_portaudio.c b/src/audio_portaudio.c similarity index 97% rename from audio_portaudio.c rename to src/audio_portaudio.c index 6d53f6af..cb6ccf10 100644 --- a/audio_portaudio.c +++ b/src/audio_portaudio.c @@ -156,7 +156,7 @@ static int calcbufsize(int rate, int chans, int bits) * the same device name for more then one connected device * (ie two SignaLinks). Appending a Portaudio device index to the * the device name ensure we can find the correct one. And if it's not - * available return the first occurence that matches the device name. + * available return the first occurrence that matches the device name. *----------------------------------------------------------------*/ static int searchPADevice(struct adev_s *dev, char *_devName, int reqDeviceNo, int io_flag) { @@ -213,7 +213,21 @@ static int pa_devNN(char *deviceStr, char *_devName, size_t length, int *_devNo) while(*cPtr) { cVal = *cPtr++; if(cVal == ':') break; - if(((cVal >= ' ') && (cVal <= '~')) && (count < length)) { + + // See Issue 417. + // Originally this copied only printable ASCII characters (space thru ~). + // That is a problem for some locales that use UTF-8 characters in the device name. + // original: if(((cVal >= ' ') && (cVal <= '~')) && (count < length)) { + + // At first I was thinking we should keep the test for < ' ' but then I + // remembered that char type can be signed or unsigned depending on implementation. + // If characters are signed then a value above 0x7f would be considered negative. + + // It seems to me that the test for buffer full is off by one. + // count could reach length, leaving no room for a nul terminator. + // Compare has been changed so count is limited to length minus 1. + + if(count < length - 1) { _devName[count++] = cVal; } @@ -513,7 +527,7 @@ static int paOutput16CB( const void *inputBuffer, void *outputBuffer, * more restrictive in its capabilities. * It might say, the best I can do is mono, 8 bit, 8000/sec. * - * The sofware modem must use this ACTUAL information + * The software modem must use this ACTUAL information * that the device is supplying, that could be different * than what the user specified. * @@ -1149,7 +1163,7 @@ int audio_put (int a, int c) static double start = 0, end = 0, diff = 0; if(adev[a].outbuf_len == 0) - start = dtime_now(); + start = dtime_monotonic(); #endif if(c >= 0) { @@ -1178,7 +1192,7 @@ int audio_put (int a, int c) #ifdef __TIMED__ count += frames; if(c < 0) { // When the Ax25 frames are flushed. - end = dtime_now(); + end = dtime_monotonic(); diff = end - start; if(count) dw_printf ("Transfer Time:%3.9f No of Frames:%d Per frame:%3.9f speed:%f\n", @@ -1246,7 +1260,7 @@ int audio_flush (int a) * (3) Call this function, which might or might not wait long enough. * (4) Add (1) and (2) resulting in when PTT should be turned off. * (5) Take difference between current time and desired PPT off time - * and wait for additoinal time if required. + * and wait for additional time if required. * *----------------------------------------------------------------*/ diff --git a/audio_stats.c b/src/audio_stats.c similarity index 99% rename from audio_stats.c rename to src/audio_stats.c index b6549cab..7b94b1ee 100644 --- a/audio_stats.c +++ b/src/audio_stats.c @@ -65,7 +65,6 @@ #include "audio_stats.h" #include "textcolor.h" -#include "dtime_now.h" #include "demod.h" /* for alevel_t & demod_get_audio_level() */ diff --git a/audio_stats.h b/src/audio_stats.h similarity index 100% rename from audio_stats.h rename to src/audio_stats.h diff --git a/audio_win.c b/src/audio_win.c similarity index 94% rename from audio_win.c rename to src/audio_win.c index 5cbb39b0..85a1548b 100644 --- a/audio_win.c +++ b/src/audio_win.c @@ -84,7 +84,7 @@ static struct audio_s *save_audio_config_p; */ /* - * Originally, we had an abitrary buf time of 40 mS. + * Originally, we had an arbitrary buf time of 40 mS. * * For mono, the buffer size was rounded up from 3528 to 4k so * it was really about 50 mS per buffer or about 20 per second. @@ -209,7 +209,7 @@ static struct adev_s { * more restrictive in its capabilities. * It might say, the best I can do is mono, 8 bit, 8000/sec. * - * The sofware modem must use this ACTUAL information + * The software modem must use this ACTUAL information * that the device is supplying, that could be different * than what the user specified. * @@ -222,8 +222,8 @@ static struct adev_s { *----------------------------------------------------------------*/ -static void CALLBACK in_callback (HWAVEIN handle, UINT msg, DWORD instance, DWORD param1, DWORD param2); -static void CALLBACK out_callback (HWAVEOUT handle, UINT msg, DWORD instance, DWORD param1, DWORD param2); +static void CALLBACK in_callback (HWAVEIN handle, UINT msg, DWORD_PTR instance, DWORD_PTR param1, DWORD_PTR param2); +static void CALLBACK out_callback (HWAVEOUT handle, UINT msg, DWORD_PTR instance, DWORD_PTR param1, DWORD_PTR param2); int audio_open (struct audio_s *pa) { @@ -313,10 +313,15 @@ int audio_open (struct audio_s *pa) /* Does config file have a number? */ /* If so, it is an index into list of devices. */ + /* Originally only a single digit was recognized. */ + /* v 1.5 also recognizes two digits. (Issue 116) */ if (strlen(pa->adev[a].adevice_in) == 1 && isdigit(pa->adev[a].adevice_in[0])) { in_dev_no[a] = atoi(pa->adev[a].adevice_in); } + else if (strlen(pa->adev[a].adevice_in) == 2 && isdigit(pa->adev[a].adevice_in[0]) && isdigit(pa->adev[a].adevice_in[1])) { + in_dev_no[a] = atoi(pa->adev[a].adevice_in); + } /* Otherwise, does it have search string? */ @@ -344,6 +349,9 @@ int audio_open (struct audio_s *pa) if (strlen(pa->adev[a].adevice_out) == 1 && isdigit(pa->adev[a].adevice_out[0])) { out_dev_no[a] = atoi(pa->adev[a].adevice_out); } + else if (strlen(pa->adev[a].adevice_out) == 2 && isdigit(pa->adev[a].adevice_out[0]) && isdigit(pa->adev[a].adevice_out[1])) { + out_dev_no[a] = atoi(pa->adev[a].adevice_out); + } if ((UINT)(out_dev_no[a]) == WAVE_MAPPER && strlen(pa->adev[a].adevice_out) >= 1) { num_devices = waveOutGetNumDevs(); @@ -553,6 +561,8 @@ int audio_open (struct audio_s *pa) */ case AUDIO_IN_TYPE_SOUNDCARD: + // Use InitializeCriticalSectionAndSpinCount to avoid exceptions in low memory situations? + InitializeCriticalSection (&(A->in_cs)); err = waveInOpen (&(A->audio_in_handle), in_dev_no[a], &wf, (DWORD_PTR)in_callback, a, CALLBACK_FUNCTION); @@ -676,24 +686,25 @@ int audio_open (struct audio_s *pa) * Called when input audio block is ready. */ -static void CALLBACK in_callback (HWAVEIN handle, UINT msg, DWORD instance, DWORD param1, DWORD param2) +static void CALLBACK in_callback (HWAVEIN handle, UINT msg, DWORD_PTR instance, DWORD_PTR param1, DWORD_PTR param2) { - int a = instance; - -//dw_printf ("in_callback, handle = %d, a = %d\n", (int)handle, a); + //dw_printf ("in_callback, handle = %p, msg = %d, instance = %I64d\n", handle, msg, instance); + int a = instance; assert (a >= 0 && a < MAX_ADEVS); struct adev_s *A = &(adev[a]); - if (msg == WIM_DATA) { WAVEHDR *p = (WAVEHDR*)param1; - p->dwUser = -1; /* needs to be unprepared. */ + p->dwUser = 0x5a5a5a5a; /* needs to be unprepared. */ + /* dwUser can be 32 or 64 bit unsigned int. */ p->lpNext = NULL; + // dw_printf ("dwBytesRecorded = %ld\n", p->dwBytesRecorded); + EnterCriticalSection (&(A->in_cs)); if (A->in_headp == NULL) { @@ -718,7 +729,7 @@ static void CALLBACK in_callback (HWAVEIN handle, UINT msg, DWORD instance, DWOR */ -static void CALLBACK out_callback (HWAVEOUT handle, UINT msg, DWORD instance, DWORD param1, DWORD param2) +static void CALLBACK out_callback (HWAVEOUT handle, UINT msg, DWORD_PTR instance, DWORD_PTR param1, DWORD_PTR param2) { if (msg == WOM_DONE) { @@ -799,7 +810,7 @@ int audio_get (int a) p = (WAVEHDR*)(A->in_headp); /* no need to be volatile at this point */ - if (p->dwUser == (DWORD)(-1)) { + if (p->dwUser == 0x5a5a5a5a) { // dwUser can be 32 or bit unsigned. waveInUnprepareHeader(A->audio_in_handle, p, sizeof(WAVEHDR)); p->dwUser = 0; /* Index for next byte. */ @@ -842,7 +853,7 @@ int audio_get (int a) assert (A->udp_sock > 0); - res = recv (A->udp_sock, A->stream_data, SDR_UDP_BUF_MAXLEN, 0); + res = SOCK_RECV (A->udp_sock, A->stream_data, SDR_UDP_BUF_MAXLEN); if (res <= 0) { text_color_set(DW_COLOR_ERROR); dw_printf ("Can't read from udp socket, errno %d", WSAGetLastError()); @@ -912,7 +923,7 @@ int audio_get (int a) * c - One byte in range of 0 - 255. * * - * Global In: out_current - index of output buffer currenly being filled. + * Global In: out_current - index of output buffer currently being filled. * * Returns: Normally non-negative. * -1 for any type of error. @@ -943,7 +954,15 @@ int audio_put (int a, int c) timeout--; if (timeout <= 0) { text_color_set(DW_COLOR_ERROR); + +// TODO: open issues 78 & 165. How can we avoid/improve this? + dw_printf ("Audio output failure waiting for buffer.\n"); + dw_printf ("This can occur when we are producing audio output for\n"); + dw_printf ("transmit and the operating system doesn't provide buffer\n"); + dw_printf ("space after waiting and retrying many times.\n"); + //dw_printf ("In recent years, this has been reported only when running the\n"); + //dw_printf ("Windows version with VMWare on a Macintosh.\n"); ptt_term (); return (-1); } @@ -1055,7 +1074,7 @@ int audio_flush (int a) * (3) Call this function, which might or might not wait long enough. * (4) Add (1) and (2) resulting in when PTT should be turned off. * (5) Take difference between current time and desired PPT off time - * and wait for additoinal time if required. + * and wait for additional time if required. * *----------------------------------------------------------------*/ diff --git a/ax25_link.c b/src/ax25_link.c similarity index 85% rename from ax25_link.c rename to src/ax25_link.c index 324334fa..3043a046 100644 --- a/ax25_link.c +++ b/src/ax25_link.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2016 John Langner, WB2OSZ +// Copyright (C) 2016, 2017, 2018, 2023 John Langner, WB2OSZ // // 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 @@ -26,6 +26,10 @@ * Establish connections and transfer data in the proper * order with retries. * + * Using the term "data link" is rather unfortunate because it causes + * confusion to someone familiar with the OSI networking model. + * This corresponds to the layer 4 transport, not layer 2 data link. + * * Description: * * Typical sequence for establishing a connection @@ -33,8 +37,8 @@ * get refused, and fall back to trying version 2.0. * * - * State Client App State Mach. Peer - * ----- ---------- ----------- ---- + * State Client App State Machine Peer + * ----- ---------- ------------- ---- * * 0 disc * Conn. Req ---> @@ -51,8 +55,8 @@ * Typical sequence when other end initiates connection. * * - * State Client App State Mach. Peer - * ----- ---------- ----------- ---- + * State Client App State Machine Peer + * ----- ---------- ------------- ---- * * 0 disc * <--- SABME or SABM @@ -63,20 +67,33 @@ * * *note: * - * By reading the v2.2 spec, I expected a 2.0 implementation to send - * FRMR in response to SABME. In testing, I found that the KPC-3+ sent DM. + * After carefully studying the v2.2 spec, I expected a 2.0 implementation to send + * FRMR in response to SABME. This is important. If a v2.2 implementation + * gets FRMR, in response to SABME, it switches to v2.0 and sends SABM instead. * - * After consulting the 2.0 spec, it says, that when in disconnected - * mode, it will respond to any command other than SABM or UI frame with a DM - * response with P/F set to 1. I can see where they might get that idea. + * The v2.0 protocol spec, section 2.3.4.3.3.1, states that FRMR should be sent when + * an invalid or not implemented command is received. That all fits together. * - * I think the rule about sending FRMR for any unrecognized command should take precedence. + * In testing, I found that the KPC-3+ sent DM. + * + * I can see where they might get that idea. + * The v2.0 spec says that when in disconnected mode, it should respond to any + * command other than SABM or UI frame with a DM response with P/F set to 1. + * I think it was implemented wrong. 2.3.4.3.3.1 should take precedence. + * + * The TM-D710 does absolutely nothing in response to SABME. + * Not responding at all is just plain wrong. To work around this, I put + * in a special hack to start sending SABM after a certain number of + * SABME go unanswered. There is more discussion in the User Guide. * - * * References: * * AX.25 Amateur Packet-Radio Link-Layer Protocol Version 2.0, October 1984 * - * https://www.tapr.org/pub_ax25.html + * https://www.tapr.org/pub_ax25.html + * http://lea.hamradio.si/~s53mv/nbp/nbp/AX25V20.pdf + * + * At first glance, they look pretty much the same, but the second one + * is more complete with 4 appendices, including a state table. * * * AX.25 Link Access Protocol for Amateur Packet Radio Version 2.2 Revision: July 1998 * @@ -92,17 +109,26 @@ * "This is a new version of the 1998 standard. It has had all figures * redone using Microsoft Visio. Errors in the SDL have been corrected." * - * The SDL diagrams are dated 2006. I wish I knew about this version earlier - * before doing most of the implementation. :-( + * The SDL diagrams are dated 2006. I wish I had known about this version, with + * several corrections, before doing most of the implementation. :-( + * + * The title page still says July 1998 so it's not immediately obvious this + * is different than the one on the TAPR site. * + * * AX.25 ... Latest revision, in progress. + * + * http://www.nj7p.org/ + * + * This is currently being revised in cooperation with software authors + * who have noticed some issues during implementation. * * The functions here are based on the SDL diagrams but turned inside out. * It seems more intuitive to have a function for each type of input and then decide * what to do depending on the state. This also reduces duplicate code because we * often see the same flow chart segments, for the same input, appearing in multiple states. * - * Erratum: The protocol spec has many places that appear to be errors or are ambiguous so I wasn't - * sure what to do. These should be annotated with Errata comments so we can easily go + * Errata: The protocol spec has many places that appear to be errors or are ambiguous so I wasn't + * sure what to do. These should be annotated with "erratum" comments so we can easily go * back and revisit them. * * X.25: The AX.25 protocol is based on, but does not necessarily adhere to, the X.25 protocol. @@ -110,11 +136,12 @@ * * http://www.itu.int/rec/T-REC-X.25-199610-I/en/ * - * Current Status: This is still under development. + * Version 1.4, released April 2017: * - * Features partially tested: + * Features tested reasonably well: * * Connect to/from a KPC-3+ and send I frames in both directions. + * Same with TM-D710A. * v2.2 connect between two instances of direwolf. (Can't find another v2.2 for testing.) * Modulo 8 & 128 sequence numbers. * Recovery from simulated transmission errors using either REJ or SREJ. @@ -128,6 +155,14 @@ * T3 timer. * Compatibility with additional types of TNC. * + * Version 1.5, December 2017: + * + * Implemented Multi Selective Reject. + * More efficient generation of SREJ frames. + * Reduced number of duplicate I frames sent for both REJ and SREJ cases. + * Avoided unnecessary RR when I frame could take care of the ack. + * (This led to issue 132 where outgoing data sometimes got stuck in the queue.) + * *------------------------------------------------------------------*/ #include "direwolf.h" @@ -157,9 +192,11 @@ #define MAX(a,b) ((a)>(b)?(a):(b)) // Debug switches for different types of information. +// Should have command line options instead of changing source and recompiling. -static int s_debug_protocol_errors = 1; // Less serious Protocol errors. +static int s_debug_protocol_errors = 0; // Less serious Protocol errors. // Useful for debugging but unnecessarily alarming other times. + // Was it intentially left on for release 1.6? static int s_debug_client_app = 0; // Interaction with client application. // dl_connect_request, dl_data_request, dl_data_indication, etc. @@ -215,7 +252,7 @@ typedef struct ax25_dlsm_s { // notifications about state changes. - char addrs[AX25_MAX_REPEATERS][AX25_MAX_ADDR_LEN]; + char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; // Up to 10 addresses, same order as in frame. int num_addr; // Number of addresses. Should be in range 2 .. 10. @@ -224,7 +261,7 @@ typedef struct ax25_dlsm_s { // addrs[OWNCALL] is owncall for this end of link. // Note that we are acting on behalf of // a client application so the APRS mycall - // might not be relevent. + // might not be relevant. #define PEERCALL AX25_DESTINATION // addrs[PEERCALL] is call for other end. @@ -239,10 +276,10 @@ typedef struct ax25_dlsm_s { // Determines whether we have one or two control // octets. 128 allows a much larger window size. - int srej_enabled; // Is other end capable of processing SREJ? (Am I allowed to send it?) - // Starts out as false for v2.0 or true for v2.2. - // Can be changed when exchanging capabilities. - // Can be used only with modulo 128. + enum srej_e srej_enable; // Is other end capable of processing SREJ? (Am I allowed to send it?) + // Starts out as 'srej_none' for v2.0 or 'srej_single' for v2.2. + // Can be changed to 'srej_multi' with XID exchange. + // Should be used only with modulo 128. (Is this enforced?) int n1_paclen; // Maximum length of information field, in bytes. // Starts out as 256 but can be negotiated higher. @@ -257,7 +294,7 @@ typedef struct ax25_dlsm_s { int k_maxframe; // Window size. Defaults to 4 (mod 8) or 32 (mod 128). // Maximum number of unacknowledged information // frames that can be outstanding. - // "MAXFRAME" parameter in configuration file. + // "MAXFRAME" or "EMAXFRAME" parameter in configuration file. int rc; // Retry count. Give up after n2. @@ -288,11 +325,19 @@ typedef struct ax25_dlsm_s { int reject_exception; // A REJ frame has been sent to the remote station. (boolean) + // This is used only when receiving an I frame, in states 3 & 4, SREJ not enabled. + // When an I frame has an unexpected N(S), + // - if not already set, set it and send REJ. + // When an I frame with expected N(S) is received, clear it. + // This would prevent us from sending additional REJ while + // waiting for result from first one. + // What happens if the REJ gets lost? Is it resent somehow? + int own_receiver_busy; // Layer 3 is busy and can't receive I frames. // We have no API to convey this information so it should always be 0. int acknowledge_pending; // I frames have been successfully received but not yet - // acknowledged to the remote station. + // acknowledged TO the remote station. // Set when receiving the next expected I frame and P=0. // This gets cleared by sending any I, RR, RNR, REJ. // Cleared when sending SREJ with F=1. @@ -304,7 +349,7 @@ typedef struct ax25_dlsm_s { // Sometimes the flow chart has SAT instead of SRT. // I think that is a typographical error. - float t1v; // How long to wait for an acknowlegement before resending. + float t1v; // How long to wait for an acknowledgement before resending. // Value used when starting timer T1, in seconds. // "FRACK" parameter in some implementations. // Typically it might be 3 seconds after frame has been @@ -381,7 +426,7 @@ typedef struct ax25_dlsm_s { // Counting outgoing could probably be done in lm_data_request so // it would not have to be scattered all over the place. TBD - int count_recv_frame_type[frame_not_AX25+1]; + int count_recv_frame_type[frame_not_AX25+1]; int peak_rc_value; // Peak value of retry count (rc). @@ -390,6 +435,9 @@ typedef struct ax25_dlsm_s { cdata_t *i_frame_queue; // Connected data from client which has not been transmitted yet. // Linked list. + // The name is misleading because these are just blocks of + // data, not "I frames" at this point. The name comes from + // the protocol specification. cdata_t *txdata_by_ns[128]; // Data which has already been transmitted. // Indexed by N(S) in case it gets lost and needs to be sent again. @@ -469,6 +517,7 @@ static reg_callsign_t *reg_callsign_list = NULL; text_color_set(DW_COLOR_DEBUG); \ dw_printf ("V(S) = %d at %s %d\n", S->vs, __func__, __LINE__); \ } \ + assert (S->vs >= 0 && S->vs < S->modulo); \ } // If other guy acks reception of an I frame, we should never get an REJ or SREJ @@ -480,6 +529,7 @@ static reg_callsign_t *reg_callsign_list = NULL; text_color_set(DW_COLOR_DEBUG); \ dw_printf ("V(A) = %d at %s %d\n", S->va, __func__, __LINE__); \ } \ + assert (S->va >= 0 && S->va < S->modulo); \ int x = AX25MODULO(n-1, S->modulo, __FILE__, __func__, __LINE__); \ while (S->txdata_by_ns[x] != NULL) { \ cdata_delete (S->txdata_by_ns[x]); \ @@ -493,12 +543,25 @@ static reg_callsign_t *reg_callsign_list = NULL; text_color_set(DW_COLOR_DEBUG); \ dw_printf ("V(R) = %d at %s %d\n", S->vr, __func__, __LINE__); \ } \ + assert (S->vr >= 0 && S->vr < S->modulo); \ + } + +#define SET_RC(n) { S->rc = (n); \ + if (s_debug_variables) { \ + text_color_set(DW_COLOR_DEBUG); \ + dw_printf ("rc = %d at %s %d, state = %d\n", S->rc, __func__, __LINE__, S->state); \ + } \ } //TODO: Make this a macro so we can simplify calls yet keep debug output if something goes wrong. +#if 0 +#define AX25MODULO(n) ax25modulo((n), S->modulo, __FILE__, __func__, __LINE__) +static int ax25modulo(int n, int m, const char *file, const char *func, int line) +#else static int AX25MODULO(int n, int m, const char *file, const char *func, int line) +#endif { if (m != 8 && m != 128) { text_color_set(DW_COLOR_ERROR); @@ -510,6 +573,13 @@ static int AX25MODULO(int n, int m, const char *file, const char *func, int line } +// Test whether we can send more or if we need to wait +// because we have reached 'maxframe' outstanding frames. +// Argument must be 'S'. + +#define WITHIN_WINDOW_SIZE(x) (x->vs != AX25MODULO(x->va + x->k_maxframe, x->modulo, __FILE__, __func__, __LINE__)) + + // Timer macros to provide debug output with location from where they are called. #define START_T1 start_t1(S, __func__, __LINE__) @@ -526,10 +596,10 @@ static int AX25MODULO(int n, int m, const char *file, const char *func, int line #define PAUSE_TM201 pause_tm201(S, __func__, __LINE__) #define RESUME_TM201 resume_tm201(S, __func__, __LINE__) +// TODO: add SELECT_T1_VALUE for debugging. -static void dl_data_indication (ax25_dlsm_t *S, int pid, char *data, int len); -static void lm_seize_confirm (ax25_dlsm_t *S); +static void dl_data_indication (ax25_dlsm_t *S, int pid, char *data, int len); static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid, char *info_ptr, int info_len); static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *info_ptr, int info_len); @@ -537,7 +607,7 @@ static int is_ns_in_window (ax25_dlsm_t *S, int ns); static void send_srej_frames (ax25_dlsm_t *S, int *resend, int count, int allow_f1); static void rr_rnr_frame (ax25_dlsm_t *S, int ready, cmdres_t cr, int pf, int nr); static void rej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr); -static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr); +static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr, unsigned char *info_ptr, int info_len); static void sabm_e_frame (ax25_dlsm_t *S, int extended, int p); static void disc_frame (ax25_dlsm_t *S, int f); @@ -762,6 +832,11 @@ static ax25_dlsm_t *get_link_handle (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LE // Create new data link state machine. p = calloc (sizeof(ax25_dlsm_t), 1); + if (p == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } p->magic1 = MAGIC1; p->start_time = dtime_now(); p->stream_id = next_stream_id++; @@ -825,6 +900,7 @@ static ax25_dlsm_t *get_link_handle (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LE // // dl_connect_request // dl_disconnect_request +// dl_outstanding_frames_request - (mine) Ask about outgoing queue for a link. // dl_data_request - send connected data // dl_unit_data_request - not implemented. APRS & KISS bypass this // dl_flow_off - not implemented. Not in AGW API. @@ -885,6 +961,8 @@ void dl_connect_request (dlq_item_t *E) { ax25_dlsm_t *S; int ok_to_create = 1; + int old_version; + int n; if (s_debug_client_app) { text_color_set(DW_COLOR_DEBUG); @@ -902,7 +980,16 @@ void dl_connect_request (dlq_item_t *E) INIT_T1V_SRT; - if (g_misc_config_p->maxv22 == 0) { // Don't attempt v2.2. +// See if destination station is in list for v2.0 only. + + old_version = 0; + for (n = 0; n < g_misc_config_p->v20_count && ! old_version; n++) { + if (strcmp(E->addrs[AX25_DESTINATION],g_misc_config_p->v20_addrs[n]) == 0) { + old_version = 1; + } + } + + if (old_version || g_misc_config_p->maxv22 == 0) { // Don't attempt v2.2. set_version_2_0 (S); @@ -992,14 +1079,40 @@ void dl_disconnect_request (dlq_item_t *E) case state_1_awaiting_connection: case state_5_awaiting_v22_connection: -// TODO: FIXME requeue. Not sure what to do here. -// If we put it back in the queue we will get it back again probably still in same state. -// Need a way to defer it until the next state change. +// Erratum: The protocol spec says "requeue." If we put disconnect req back in the +// queue we will probably get it back again here while still in same state. +// I don't think we would want to delay it until the next state transition. + +// Suppose someone tried to connect to another station, which is not responding, and decided to cancel +// before all of the SABMe retries were used up. I think we would want to transmit a DISC, send a disc +// notice to the user, and go directly into disconnected state, rather than into awaiting release. + +// New code v1.7 dev, May 6 2023 + text_color_set(DW_COLOR_INFO); + dw_printf ("Stream %d: In progress connection attempt to %s terminated by user.\n", S->stream_id, S->addrs[PEERCALL]); + discard_i_queue (S); + SET_RC(0); + int p1 = 1; + int nopid0 = 0; + packet_t pp15 = ax25_u_frame (S->addrs, S->num_addr, cr_cmd, frame_type_U_DISC, p1, nopid0, NULL, 0); + lm_data_request (S->chan, TQ_PRIO_1_LO, pp15); + + STOP_T1; // started in establish_data_link. + STOP_T3; // probably don't need. + enter_new_state (S, state_0_disconnected, __func__, __LINE__); + server_link_terminated (S->chan, S->client, S->addrs[PEERCALL], S->addrs[OWNCALL], 0); break; - + case state_2_awaiting_release: { + // We have previously started the disconnect sequence and are waiting + // for a UA from the other guy. Meanwhile, the application got + // impatient and sent us another disconnect request. What should + // we do? Ignore it and let the disconnect sequence run its + // course? Or should we complete the sequence without waiting + // for the other guy to ack? + // Erratum. Flow chart simply says "DM (expedited)." // This is the only place we have expedited. Is this correct? @@ -1010,6 +1123,13 @@ void dl_disconnect_request (dlq_item_t *E) packet_t pp = ax25_u_frame (S->addrs, S->num_addr, cr, frame_type_U_DM, p, nopid, NULL, 0); lm_data_request (S->chan, TQ_PRIO_0_HI, pp); // HI means expedited. + // Erratum: Shouldn't we inform the user when going to disconnected state? + // Notifying the application, here, is my own enhancement. + + text_color_set(DW_COLOR_INFO); + dw_printf ("Stream %d: Disconnected from %s.\n", S->stream_id, S->addrs[PEERCALL]); + server_link_terminated (S->chan, S->client, S->addrs[PEERCALL], S->addrs[OWNCALL], 0); + STOP_T1; enter_new_state (S, state_0_disconnected, __func__, __LINE__); } @@ -1019,7 +1139,7 @@ void dl_disconnect_request (dlq_item_t *E) case state_4_timer_recovery: discard_i_queue (S); - S->rc = 0; // I think this should be 1 but I'm not that worried about it. + SET_RC(0); // I think this should be 1 but I'm not that worried about it. cmdres_t cmd = cr_cmd; int p = 1; @@ -1069,6 +1189,15 @@ void dl_disconnect_request (dlq_item_t *E) * * Erratum: Not sure how to interpret that. See example below for how it was implemented. * + * Version 1.6: Bug 252. Segmentation was occurring for a V2.0 link. From the spec: + * "The receipt of an XID response from the other station establishes that both + * stations are using AX.25 version 2.2 or higher and enables the use of the + * segmenter/reassembler and selective reject." + * "The segmenter/reassembler procedure is only enabled if both stations on the + * link are using AX.25 version 2.2 or higher." + * + * The Segmenter Ready State SDL has no decision based on protocol version. + * *------------------------------------------------------------------------------*/ static void data_request_good_size (ax25_dlsm_t *S, cdata_t *txdata); @@ -1078,8 +1207,6 @@ void dl_data_request (dlq_item_t *E) { ax25_dlsm_t *S; int ok_to_create = 1; - int nseg_to_follow; - int orig_offset, remaining_len; S = get_link_handle (E->addrs, E->num_addr, E->chan, E->client, ok_to_create); @@ -1097,6 +1224,39 @@ void dl_data_request (dlq_item_t *E) return; } +#define DIVROUNDUP(a,b) (((a)+(b)-1) / (b)) + +// Erratum: Don't do V2.2 segmentation for a V2.0 link. +// In this case, we can just split it into multiple frames not exceeding the specified max size. +// Hopefully the receiving end treats it like a stream and doesn't care about length of each frame. + + if (S->modulo == 8) { + + int num_frames = 0; + int remaining_len = E->txdata->len; + int offset = 0; + + while (remaining_len > 0) { + int this_len = MIN(remaining_len, S->n1_paclen); + + cdata_t *new_txdata = cdata_new(E->txdata->pid, E->txdata->data + offset, this_len); + data_request_good_size (S, new_txdata); + + offset += this_len; + remaining_len -= this_len; + num_frames++; + } + + if (num_frames != DIVROUNDUP(E->txdata->len, S->n1_paclen) || remaining_len != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("INTERNAL ERROR, Segmentation line %d, data length = %d, N1 = %d, num frames = %d, remaining len = %d\n", + __LINE__, E->txdata->len, S->n1_paclen, num_frames, remaining_len); + } + cdata_delete (E->txdata); + E->txdata = NULL; + return; + } + // More interesting case. // It is too large to fit in one frame so we segment it. @@ -1163,7 +1323,7 @@ void dl_data_request (dlq_item_t *E) // We will decrement this before putting it in the frame so the first // will have one less than this number. - nseg_to_follow = DIVROUNDUP(E->txdata->len + 1, S->n1_paclen - 1); + int nseg_to_follow = DIVROUNDUP(E->txdata->len + 1, S->n1_paclen - 1); if (nseg_to_follow < 2 || nseg_to_follow > 128) { text_color_set(DW_COLOR_ERROR); @@ -1174,8 +1334,8 @@ void dl_data_request (dlq_item_t *E) return; } - orig_offset = 0; - remaining_len = E->txdata->len; + int orig_offset = 0; + int remaining_len = E->txdata->len; // First segment. @@ -1307,7 +1467,27 @@ static void data_request_good_size (ax25_dlsm_t *S, cdata_t *txdata) break; } - i_frame_pop_off_queue (S); + // v1.5 change in strategy. + // New I frames, not sent yet, are delayed until after processing anything in the received transmission. + // Give the transmit process a kick unless other side is busy or we have reached our window size. + // Previously we had i_frame_pop_off_queue here which would start sending new stuff before we + // finished dealing with stuff already in progress. + + switch (S->state) { + + case state_3_connected: + case state_4_timer_recovery: + + if ( ( ! S->peer_receiver_busy ) && + WITHIN_WINDOW_SIZE(S) ) { + S->acknowledge_pending = 1; + lm_seize_request (S->chan); + } + break; + + default: + break; + } } /* end data_request_good_size */ @@ -1345,6 +1525,11 @@ void dl_register_callsign (dlq_item_t *E) } r = calloc(sizeof(reg_callsign_t),1); + if (r == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } strlcpy (r->callsign, E->addrs[0], sizeof(r->callsign)); r->chan = E->chan; r->client = E->client; @@ -1398,6 +1583,138 @@ void dl_unregister_callsign (dlq_item_t *E) +/*------------------------------------------------------------------------------ + * + * Name: dl_outstanding_frames_request + * + * Purpose: Client app wants to know how many frames are still on their way + * to other station. This is handy for flow control. We would like + * to keep the pipeline filled sufficiently to take advantage of a + * large window size (MAXFRAMES). It is also good to know that the + * the last packet sent was actually received before we commence + * the disconnect. + * + * Inputs: E - Event from the queue. + * The caller will free it. + * + * Outputs: This gets back to the AGW server which sends the 'Y' reply. + * + * Description: This is the sum of: + * - Incoming connected data, from application still in the queue. + * - I frames which have been transmitted but not yet acknowledged. + * + * Confusion: https://github.com/wb2osz/direwolf/issues/427 + * + * There are different, inconsistent versions of the protocol spec. + * + * One of them simply has: + * + * CallFrom is our call + * CallTo is the call of the other station + * + * A more detailed version has the same thing in the table of fields: + * + * CallFrom 10 bytes Our CallSign + * CallTo 10 bytes Other CallSign + * + * (My first implementation went with that.) + * + * HOWEVER, shortly after that, is contradictory information: + * + * Careful must be exercised to fill correctly both the CallFrom + * and CallTo fields to match the ones of an existing connection, + * otherwise AGWPE won’t return any information at all from this query. + * + * The order of the CallFrom and CallTo is not trivial, it should + * reflect the order used to start the connection, so + * + * * If we started the connection CallFrom=US and CallTo=THEM + * * If the other end started the connection CallFrom=THEM and CallTo=US + * + * This seems to make everything unnecessarily more complicated. + * We should only care about the stream going from the local station to the + * remote station. Why would it matter who reqested the link? The state + * machine doesn't even contain this information so the TNC doesn't know. + * The client app interface needs to behave differently for the two cases. + * + * The new code, below, May 2023, should handle both of those cases. + * + *------------------------------------------------------------------------------*/ + +void dl_outstanding_frames_request (dlq_item_t *E) +{ + ax25_dlsm_t *S; + const int ok_to_create = 0; // must exist already. + int reversed_addrs = 0; + + if (s_debug_client_app) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("dl_outstanding_frames_request ( to %s )\n", E->addrs[PEERCALL]); + } + + S = get_link_handle (E->addrs, E->num_addr, E->chan, E->client, ok_to_create); + if (S != NULL) { + reversed_addrs = 0; + } + else { + // Try swapping the addresses. + // this is communicating with the client app, not over the air, + // so we don't need to worry about digipeaters. + + char swapped[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; + memset (swapped, 0, sizeof(swapped)); + strlcpy (swapped[PEERCALL], E->addrs[OWNCALL], sizeof(swapped[PEERCALL])); + strlcpy (swapped[OWNCALL], E->addrs[PEERCALL], sizeof(swapped[OWNCALL])); + S = get_link_handle (swapped, E->num_addr, E->chan, E->client, ok_to_create); + if (S != NULL) { + reversed_addrs = 1; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't get outstanding frames for %s -> %s, chan %d\n", E->addrs[OWNCALL], E->addrs[PEERCALL], E->chan); + server_outstanding_frames_reply (E->chan, E->client, E->addrs[OWNCALL], E->addrs[PEERCALL], 0); + return; + } + } + +// Add up these +// +// cdata_t *i_frame_queue; // Connected data from client which has not been transmitted yet. +// // Linked list. +// // The name is misleading because these are just blocks of +// // data, not "I frames" at this point. The name comes from +// // the protocol specification. +// +// cdata_t *txdata_by_ns[128]; // Data which has already been transmitted. +// // Indexed by N(S) in case it gets lost and needs to be sent again. +// // Cleared out when we get ACK for it. + + int count1 = 0; + cdata_t *incoming; + for (incoming = S->i_frame_queue; incoming != NULL; incoming = incoming->next) { + count1++; + } + + int count2 = 0; + int k; + for (k = 0; k < S->modulo; k++) { + if (S->txdata_by_ns[k] != NULL) { + count2++; + } + } + + if (reversed_addrs) { + // Other end initiated the link. + server_outstanding_frames_reply (S->chan, S->client, S->addrs[PEERCALL], S->addrs[OWNCALL], count1 + count2); + } + else { + server_outstanding_frames_reply (S->chan, S->client, S->addrs[OWNCALL], S->addrs[PEERCALL], count1 + count2); + } + +} // end dl_outstanding_frames_request + + + /*------------------------------------------------------------------------------ * * Name: dl_client_cleanup @@ -1411,7 +1728,7 @@ void dl_unregister_callsign (dlq_item_t *E) * Description: By client application we mean something that attached with the * AGW network protocol. * - * Clean out anything related to the specfied client application. + * Clean out anything related to the specified client application. * This would include state machines and registered callsigns. * *------------------------------------------------------------------------------*/ @@ -1694,8 +2011,6 @@ static void dl_data_indication (ax25_dlsm_t *S, int pid, char *data, int len) * * Description: We need to pause the timers when the channel is busy. * - * Signal lm_seize_confirm when we have started to transmit. - * *------------------------------------------------------------------------------*/ static int dcd_status[MAX_CHANS]; @@ -1752,14 +2067,6 @@ void lm_channel_busy (dlq_item_t *E) S->radio_channel_busy = 1; PAUSE_T1; PAUSE_TM201; - - // Did channel become busy due to PTT turning on? - - if ( E->activity == OCTYPE_PTT && E->status == 1) { - - lm_seize_confirm (S); // C4.2. "This primitive indicates, to the Data-link State - // machine, that the transmission opportunity has arrived." - } } else if ( ! busy && S->radio_channel_busy) { S->radio_channel_busy = 0; @@ -1783,36 +2090,65 @@ void lm_channel_busy (dlq_item_t *E) * Description: C4.2. This primitive indicates to the Data-link State Machine that * the transmission opportunity has arrived. * + * Version 1.5: Originally this only invoked inquiry_response to provide an ack if not already + * taken care of by an earlier frame in this transmission. + * After noticing the unnecessary I frame duplication and differing N(R) in the same + * transmission, I came to the conclusion that we should delay sending of new + * (not resends as a result of rej or srej) frames until after after processing + * of everything in the incoming transmission. + * The protocol spec simply has "I frame pops off queue" without any indication about + * what might trigger this event. + * *------------------------------------------------------------------------------*/ -static void lm_seize_confirm (ax25_dlsm_t *S) +void lm_seize_confirm (dlq_item_t *E) { - switch (S->state) { + assert (E->chan >= 0 && E->chan < MAX_CHANS); - case state_0_disconnected: - case state_1_awaiting_connection: - case state_2_awaiting_release: - case state_5_awaiting_v22_connection: + ax25_dlsm_t *S; - break; + for (S = list_head; S != NULL; S = S->next) { - case state_3_connected: - case state_4_timer_recovery: + if (E->chan == S->chan) { - if (S->acknowledge_pending) { - S->acknowledge_pending = 0; - enquiry_response (S, frame_not_AX25, 0); - } -// Implemenation difference: The flow chart for state 3 has LM-RELEASE Request here. + switch (S->state) { + + case state_0_disconnected: + case state_1_awaiting_connection: + case state_2_awaiting_release: + case state_5_awaiting_v22_connection: + + break; + + case state_3_connected: + case state_4_timer_recovery: + + // v1.5 change in strategy. + // New I frames, not sent yet, are delayed until after processing anything in the received transmission. + // Previously we started sending new frames, from the client app, as soon as they arrived. + // Now, we first take care of those in progress before throwing more into the mix. + + i_frame_pop_off_queue(S); + + // Need an RR if we didn't have I frame send the necessary ack. + + if (S->acknowledge_pending) { + S->acknowledge_pending = 0; + enquiry_response (S, frame_not_AX25, 0); + } + +// Implementation difference: The flow chart for state 3 has LM-RELEASE Request here. // I don't think I need it because the transmitter will turn off // automatically once the queue is empty. // Erratum: The original spec had LM-SEIZE request here, for state 4, which didn't seem right. // The 2006 revision has LM-RELEASE Request so states 3 & 4 are the same. - break; + break; + } + } } } /* lm_seize_confirm */ @@ -1960,7 +2296,7 @@ void lm_data_indication (dlq_item_t *E) break; // Erratum: The AX.25 spec is not clear about whether SREJ should be command, response, or both. -// The underlying X.25 spec clearly says it is reponse only. Let's go with that. +// The underlying X.25 spec clearly says it is response only. Let's go with that. case frame_type_S_SREJ: case frame_type_U_DM: @@ -1983,7 +2319,7 @@ void lm_data_indication (dlq_item_t *E) case frame_type_U_UI: // Don't test at this point in case an APRS frame gets thru. // APRS doesn't specify what to put in the Source and Dest C bits. - // In practice we see all 4 possble combinations. + // In practice we see all 4 possible combinations. // I have an opinion about what would be "correct" (discussed elsewhere) // but in practice no one seems to care. break; @@ -2022,8 +2358,14 @@ void lm_data_indication (dlq_item_t *E) rej_frame (S, cr, pf, nr); break; - case frame_type_S_SREJ: // Selective Reject - Ask for single frame repeat - srej_frame (S, cr, pf, nr); + case frame_type_S_SREJ: // Selective Reject - Ask for selective frame(s) repeat + { + unsigned char *info_ptr; + int info_len; + + info_len = ax25_get_info (E->pp, &info_ptr); + srej_frame (S, cr, pf, nr, info_ptr, info_len); + } break; case frame_type_U_SABME: // Set Async Balanced Mode, Extended @@ -2083,7 +2425,20 @@ void lm_data_indication (dlq_item_t *E) break; } - i_frame_pop_off_queue (S); +// An incoming frame might have ack'ed frames we sent or indicated peer is no longer busy. +// Rather than putting this test in many places, where those conditions, may have changed, +// we will try to catch them all on this single path. +// Start transmission if we now have some outgoing data ready to go. +// (Added in 1.5 beta 3 for issue 132.) + + if ( S->i_frame_queue != NULL && + (S->state == state_3_connected || S->state == state_4_timer_recovery) && + ( ! S->peer_receiver_busy ) && + WITHIN_WINDOW_SIZE(S) ) { + + //S->acknowledge_pending = 1; + lm_seize_request (S->chan); + } } /* end lm_data_indication */ @@ -2198,7 +2553,14 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid // Look carefully. The original had two tiny differences between the two states. // In the 2006 version, these differences no longer exist. - if (info_len >= 0 && info_len <= S->n1_paclen) { + // Erratum: SDL asks: Is information field length <= N1 (paclen). + // (github issue 102 - Thanks to KK6WHJ for pointing this out.) + // Just because we are limiting the size of our transmitted data, it doesn't mean + // that the other end will be doing the same. With v2.2, the XID frame can be + // used to negotiate a maximum info length but with v2.0, there is no way for the + // other end to know our paclen value. + + if (info_len >= 0 && info_len <= AX25_MAX_INFO_LEN) { if (is_good_nr(S,nr)) { @@ -2209,7 +2571,7 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid // Erratum: Discrepancy between original and 2006 version. // Pattern noticed: Anytime we have "is_good_nr" which tests for V(A) <= N(R) <= V(S), - // we should always call "check_i_frame_acked" or at least set V(A) from N(R). + // we should always call "check_i_frame_ackd" or at least set V(A) from N(R). #if 0 // Erratum: original - states 3 & 4 differ here. @@ -2228,6 +2590,22 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid check_i_frame_ackd (S,nr); #endif +// Erratum: v1.5 - My addition. +// I noticed that we sometimes got stuck in state 4 and rc crept up slowly even though +// we received 'I' frames with N(R) values indicating that the other side received everything +// that we sent. Eventually rc could reach the limit and we would get an error. +// If we are in state 4, and other guy ack'ed last I frame we sent, transition to state 3. +// We had a similar situation for RR/RNR for cases other than response, F=1. + + if (S->state == state_4_timer_recovery && S->va == S->vs) { + + STOP_T1; + select_t1_value (S); + START_T3; + SET_RC(0); + enter_new_state (S, state_3_connected, __func__, __LINE__); + } + if (S->own_receiver_busy) { // This should be unreachable because we currently don't have a way to set own_receiver_busy. // But we might the capability someday so implement this while we are here. @@ -2239,7 +2617,7 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid int nr = S->vr; packet_t pp; - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RNR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RNR, S->modulo, nr, f, NULL, 0); // I wonder if this difference is intentional or if only one place was // was modified after a cut-n-paste of the flow chart segment. @@ -2259,13 +2637,17 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid #endif S->acknowledge_pending = 0; } + } - else { // N(R) out of expected range. + else { // Own receiver not busy. i_frame_continued (S, p, ns, pid, info_ptr, info_len); } + + } - else { + else { // N(R) not in expected range. + nr_error_recovery (S); // my enhancement. See below. enter_new_state (S, S->modulo == 128 ? state_5_awaiting_v22_connection : state_1_awaiting_connection, __func__, __LINE__); @@ -2275,7 +2657,7 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid // Wouldn't even get to CRC check if not octet aligned. text_color_set(DW_COLOR_ERROR); - dw_printf ("Stream %d: AX.25 Protocol Error O: Information part length, %d, not in range of 0 thru %d.\n", S->stream_id, info_len, S->n1_paclen); + dw_printf ("Stream %d: AX.25 Protocol Error O: Information part length, %d, not in range of 0 thru %d.\n", S->stream_id, info_len, AX25_MAX_INFO_LEN); establish_data_link (S); S->layer_3_initiated = 0; @@ -2325,6 +2707,8 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid * * 4.3.2.4. Selective Reject (SREJ) Command and Response * + * (Erratum: SREJ is only response with F bit.) + * * The selective reject, SREJ, frame is used by the receiving TNC to request retransmission of the single I frame * numbered N(R). If the P/F bit in the SREJ frame is set to "1", then I frames numbered up to N(R)-1 inclusive are * considered as acknowledged. However, if the P/F bit in the SREJ frame is set to "0", then the N(R) of the SREJ @@ -2372,6 +2756,9 @@ static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid * * 6.4.4.3. Selective Reject-Reject (SREJ/REJ) * + * (Erratum: REJ/SREJ should not be mixed. Basic (mod 8) allows only REJ. + * Extended (mod 128) gives you a choice of one or the other for a link.) + * * When an I frame is received with a correct FCS but its send sequence number N(S) does not match the current * receiver's receive state variable, and if N(S) indicates 2 or more frames are missing, a REJ frame is transmitted. * All subsequently received frames are discarded until the lost frame is correctly received. If only one frame is @@ -2409,7 +2796,7 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf if (S->rxdata_by_ns[ns] != NULL) { // There is a possibility that we might have another received frame stashed - // away from 8 or 128 (modulo) frames back. Remove it so it doesn't accidently + // away from 8 or 128 (modulo) frames back. Remove it so it doesn't accidentally // show up at some future inopportune time. cdata_delete (S->rxdata_by_ns[ns]); @@ -2450,7 +2837,7 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf cmdres_t cr = cr_res; // response with F set to 1. packet_t pp; - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; } @@ -2481,12 +2868,12 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf cmdres_t cr = cr_res; // response with F set to 1. packet_t pp; - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; } } - else if ( ! S->srej_enabled) { + else if (S->srej_enable == srej_none) { // The received sequence number is not the expected one and we can't use SREJ. // The old v2.0 approach is to send and REJ with the number we are expecting. @@ -2494,7 +2881,7 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf // we discard 3,4,5,6, and tell the other end to resend everything starting with 2. // At one time, I had some doubts about when to use command or response for REJ. -// I now believe that reponse, as implied by setting F in the flow chart, is correct. +// I now believe that response, as implied by setting F in the flow chart, is correct. int f = p; int nr = S->vr; // Next expected sequence number. @@ -2504,11 +2891,11 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf S->reject_exception = 1; if (s_debug_retry) { - text_color_set(DW_COLOR_ERROR); // make it more noticable. + text_color_set(DW_COLOR_ERROR); // make it more noticeable. dw_printf ("sending REJ, at %s %d, SREJ not enabled case, V(R)=%d", __func__, __LINE__, S->vr); } - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_REJ, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_REJ, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; @@ -2558,31 +2945,60 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf int nr = S->vr; packet_t pp; - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RNR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RNR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); } else if (S->rxdata_by_ns[ AX25MODULO(ns - 1, S->modulo, __FILE__, __func__, __LINE__)] == NULL) { // Ask for missing frames when we don't have N(S)-1 in the receive buffer. -// I would think that we would want to set F when sending N(R) = V(R) but it says use F=0 here. -// Don't understand why yet. +// In version 1.4: +// We end up sending more SREJ than necessary and and get back redundant information. Example: +// When we see 113 missing, we ask for a resend. +// When we see 115 & 116 missing, a cumulative SREJ asks for everything. +// The other end dutifully sends 113 twice. +// +// [0.4] DW1>DW0:(SREJ res, n(r)=113, f=0) +// [0.4] DW1>DW0:(SREJ res, n(r)=113, f=1)<0xe6><0xe8> +// +// [0L] DW0>DW1:(I cmd, n(s)=113, n(r)=11, p=0, pid=0xf0)0114 send data<0x0d> +// [0L] DW0>DW1:(I cmd, n(s)=113, n(r)=11, p=0, pid=0xf0)0114 send data<0x0d> +// [0L] DW0>DW1:(I cmd, n(s)=115, n(r)=11, p=0, pid=0xf0)0116 send data<0x0d> +// [0L] DW0>DW1:(I cmd, n(s)=116, n(r)=11, p=0, pid=0xf0)0117 send data<0x0d> -// I like my method better. It does not generate duplicate requests for gaps in the same transmission. -// This creates a cummulative list each time and would cause repeats to be sent more often than necessary. - int resend[128]; - int count = 0; +// Version 1.5: +// Don't generate duplicate requests for gaps in the same transmission. + +// Ideally, we might wait until carrier drops and then use one Multi-SREJ for entire transmission but +// we will keep that for another day. +// Probably need a flag similar to acknowledge_pending (or ask_resend_count, here) and the ask_for_resend array. +// It could then be processed first in lm_seize_confirm. + + int ask_for_resend[128]; + int ask_resend_count = 0; int x; - int allow_f1 = 0; // F=1 from X.25 2.4.6.4 b) 3) - for (x = S->vr; x != ns; x = AX25MODULO(x + 1, S->modulo, __FILE__, __func__, __LINE__)) { - if (S->rxdata_by_ns[x] == NULL) { - resend[count++] = AX25MODULO(x, S->modulo, __FILE__, __func__, __LINE__); - } +// Version 1.5 +// Erratum: AX.25 says use F=0 here. Doesn't make sense. +// We would want to set F when sending N(R) = V(R). +// int allow_f1 = 0; // F=1 from X.25 2.4.6.4 b) 3) + int allow_f1 = 1; // F=1 from X.25 2.4.6.4 b) 3) + +// send only for this gap, not cumulative from V(R). + + int last = AX25MODULO(ns - 1, S->modulo, __FILE__, __func__, __LINE__); + int first = last; + while (first != S->vr && S->rxdata_by_ns[AX25MODULO(first - 1, S->modulo, __FILE__, __func__, __LINE__)] == NULL) { + first = AX25MODULO(first - 1, S->modulo, __FILE__, __func__, __LINE__); } - - send_srej_frames (S, resend, count, allow_f1); + x = first; + do { + ask_for_resend[ask_resend_count++] = AX25MODULO(x, S->modulo, __FILE__, __func__, __LINE__); + x = AX25MODULO(x + 1, S->modulo, __FILE__, __func__, __LINE__); + } while (x != AX25MODULO(last + 1, S->modulo, __FILE__, __func__, __LINE__)); + + send_srej_frames (S, ask_for_resend, ask_resend_count, allow_f1); } } else { @@ -2634,8 +3050,8 @@ static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *inf // In this case, we need to ask for a resend of 1 & 2. // More generally, the range of V(R) thru N(S)-1. - int resend[128]; - int count = 0; + int ask_for_resend[128]; + int ask_resend_count = 0; int i; int allow_f1 = 1; @@ -2643,10 +3059,10 @@ text_color_set(DW_COLOR_ERROR); dw_printf ("%s:%d, zero exceptions, V(R)=%d, N(S)=%d\n", __func__, __LINE__, S->vr, ns); for (i = S->vr; i != ns; i = AX25MODULO(i+1, S->modulo, __FILE__, __func__, __LINE__)) { - resend[count++] = i; + ask_for_resend[ask_resend_count++] = i; } - send_srej_frames (S, resend, count, allow_f1); + send_srej_frames (S, ask_for_resend, ask_resend_count, allow_f1); } else { @@ -2661,7 +3077,7 @@ dw_printf ("%s:%d, zero exceptions, V(R)=%d, N(S)=%d\n", __func__, __LINE__, S-> // This can be achieved by searching S->rxdata_by_ns, starting with N(S)-1, and counting // how many empty slots we have before finding a saved frame. - int count = 0; + int ask_resend_count = 0; int first; text_color_set(DW_COLOR_ERROR); @@ -2672,8 +3088,8 @@ dw_printf ("%s:%d, %d srej exceptions, V(R)=%d, N(S)=%d\n", __func__, __LINE__, if (first == AX25MODULO(S->vr - 1, S->modulo, __FILE__, __func__, __LINE__)) { // Oops! Went too far. This I frame was already processed. text_color_set(DW_COLOR_ERROR); - dw_printf ("INTERNAL ERROR calulating what to put in SREJ, %s line %d\n", __func__, __LINE__); - dw_printf ("V(R)=%d, N(S)=%d, SREJ exception=%d, first=%d, count=%d\n", S->vr, ns, selective_reject_exception(S), first, count); + dw_printf ("INTERNAL ERROR calculating what to put in SREJ, %s line %d\n", __func__, __LINE__); + dw_printf ("V(R)=%d, N(S)=%d, SREJ exception=%d, first=%d, ask_resend_count=%d\n", S->vr, ns, selective_reject_exception(S), first, ask_resend_count); int k; for (k=0; k<128; k++) { if (S->rxdata_by_ns[k] != NULL) { @@ -2682,31 +3098,29 @@ dw_printf ("%s:%d, %d srej exceptions, V(R)=%d, N(S)=%d\n", __func__, __LINE__, } break; } - count++; + ask_resend_count++; first = AX25MODULO(first - 1, S->modulo, __FILE__, __func__, __LINE__); } // Go beyond the slot where we already have an I frame. first = AX25MODULO(first + 1, S->modulo, __FILE__, __func__, __LINE__); - // The count could be 0. e.g. We got 4 rather than 7 in this example. + // The ask_resend_count could be 0. e.g. We got 4 rather than 7 in this example. - if (count > 0) { - int resend[128]; + if (ask_resend_count > 0) { + int ask_for_resend[128]; int n; int allow_f1 = 1; - for (n = 0; n < count; n++) { - resend[n] = AX25MODULO(first + n, S->modulo, __FILE__, __func__, __LINE__);; + for (n = 0; n < ask_resend_count; n++) { + ask_for_resend[n] = AX25MODULO(first + n, S->modulo, __FILE__, __func__, __LINE__);; } - send_srej_frames (S, resend, count, allow_f1); + send_srej_frames (S, ask_for_resend, ask_resend_count, allow_f1); } } /* end SREJ exception */ - - #endif // my earlier attempt. @@ -2721,7 +3135,7 @@ dw_printf ("%s:%d, %d srej exceptions, V(R)=%d, N(S)=%d\n", __func__, __LINE__, S->acknowledge_pending = 0; #endif - } /* end srej_enabled */ + } /* end srej enabled */ } /* end i_frame_continued */ @@ -2789,16 +3203,22 @@ static int is_ns_in_window (ax25_dlsm_t *S, int ns) * Purpose: Ask for a resend of I frames with specified sequence numbers. * * Inputs: resend - Array of N(S) values for missing I frames. + * * count - Number of items in array. + * * allow_f1 - When true, set F=1 when asking for V(R). + * * X.25 section 2.4.6.4 b) 3) says F should be set to 0 * when receiving I frame out of sequence. + * * X.25 sections 2.4.6.11 & 2.3.5.2.2 say set F to 1 when * responding to command with P=1. (our enquiry_response function). * - * Future? The X.25 protocol spec allows additional sequence numbers in one frame + * Version 1.5: The X.25 protocol spec allows additional sequence numbers in one frame * by using the INFO part. - * It would be easy but we haven't done that here to remain compatible with AX.25. + * By default that feature is off but can be negotiated with XID. + * We should be able to use this between two direwolf stations while + * maintaining compatibility with the original AX.25 v2.2. * *------------------------------------------------------------------------------*/ @@ -2821,7 +3241,7 @@ static void send_srej_frames (ax25_dlsm_t *S, int *resend, int count, int allow_ if (s_debug_retry) { text_color_set(DW_COLOR_INFO); dw_printf ("%s line %d\n", __func__, __LINE__); - //dw_printf ("state=%d, count=%d, k=%d, V(R)=%d, SREJ exeception=%d\n", S->state, count, S->k_maxframe, S->vr, selective_reject_exception(S)); + //dw_printf ("state=%d, count=%d, k=%d, V(R)=%d, SREJ exception=%d\n", S->state, count, S->k_maxframe, S->vr, selective_reject_exception(S)); dw_printf ("state=%d, count=%d, k=%d, V(R)=%d\n", S->state, count, S->k_maxframe, S->vr); dw_printf ("resend[]="); @@ -2863,6 +3283,60 @@ static void send_srej_frames (ax25_dlsm_t *S, int *resend, int count, int allow_ dw_printf ("\n"); } +// Multi-SREJ - Use info part for additional sequence number(s) instead of sending separate SREJ for each. + + if (S->srej_enable == srej_multi && count > 1) { + + unsigned char info[128]; + int info_len = 0; + + for (i = 1; i < count; i++) { // skip first one + + if (resend[i] < 0 || resend[i] >= S->modulo) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("INTERNAL ERROR, additional nr=%d, modulo=%d, %s line %d\n", resend[i], S->modulo, __func__, __LINE__); + } + + // There is also a form to specify a range but I don't + // think it is worth the effort to generate it. Maybe later. + + if (S->modulo == 8) { + info[info_len++] = resend[i] << 5; + } + else { + info[info_len++] = resend[i] << 1; + } + } + + f = 0; + nr = resend[0]; + f = allow_f1 && (nr == S->vr); + // Possibly set if we are asking for the next after + // the last one received in contiguous order. + + // This could only apply to the first in + // the list so this would not go in the loop. + + if (f) { // In this case the other end is being + // informed of my V(R) so no additional + // RR etc. is needed. + // TODO: Need to think about this. + S->acknowledge_pending = 0; + } + + if (nr < 0 || nr >= S->modulo) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("INTERNAL ERROR, nr=%d, modulo=%d, %s line %d\n", nr, S->modulo, __func__, __LINE__); + nr = AX25MODULO(nr, S->modulo, __FILE__, __func__, __LINE__); + } + + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_SREJ, S->modulo, nr, f, info, info_len); + lm_data_request (S->chan, TQ_PRIO_1_LO, pp); + return; + } + +// Multi-SREJ not enabled. Send separate SREJ for each desired sequence number. + for (i = 0; i < count; i++) { nr = resend[i]; @@ -2883,7 +3357,7 @@ static void send_srej_frames (ax25_dlsm_t *S, int *resend, int count, int allow_ nr = AX25MODULO(nr, S->modulo, __FILE__, __func__, __LINE__); } - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_SREJ, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_SREJ, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); } @@ -2987,7 +3461,6 @@ static void rr_rnr_frame (ax25_dlsm_t *S, int ready, cmdres_t cr, int pf, int nr // dw_printf ("rr_rnr_frame (), line %d, state=%d, good nr=%d, calling check_i_frame_ackd\n", __LINE__, S->state, nr); check_i_frame_ackd (S, nr); - // keep current state. } else { if (s_debug_retry) { @@ -3008,9 +3481,11 @@ static void rr_rnr_frame (ax25_dlsm_t *S, int ready, cmdres_t cr, int pf, int nr if (cr == cr_res && pf == 1) { +// RR/RNR Response with F==1. + if (s_debug_retry) { text_color_set(DW_COLOR_DEBUG); - dw_printf ("rr_rnr_frame (), Response, pf=1, line %d, state=%d, good nr, calling check_i_frame_ackd\n", __LINE__, S->state); + dw_printf ("rr_rnr_frame (), Response, f=%d, line %d, state=%d, good nr, calling check_i_frame_ackd\n", pf, __LINE__, S->state); } STOP_T1; @@ -3021,7 +3496,7 @@ static void rr_rnr_frame (ax25_dlsm_t *S, int ready, cmdres_t cr, int pf, int nr SET_VA(nr); if (S->vs == S->va) { // all caught up with ack from other guy. START_T3; - S->rc =0; // My enhancement. See Erratum note in select_t1_value. + SET_RC(0); // My enhancement. See Erratum note in select_t1_value. enter_new_state (S, state_3_connected, __func__, __LINE__); } else { @@ -3049,21 +3524,44 @@ static void rr_rnr_frame (ax25_dlsm_t *S, int ready, cmdres_t cr, int pf, int nr } } else { + +// RR/RNR command, either P value. +// RR/RNR response, F==0 + if (cr == cr_cmd && pf == 1) { int f = 1; enquiry_response (S, ready ? frame_type_S_RR : frame_type_S_RNR, f); } if (is_good_nr(S,nr)) { + SET_VA(nr); - // REJ state 4 is identical but has conditional invoke_retransmission here. + +// Erratum: v1.5 - my addition. +// I noticed that we sometimes got stuck in state 4 and rc crept up slowly even though +// we received RR frames with N(R) values indicating that the other side received everything +// that we sent. Eventually rc could reach the limit and we would get an error. +// If we are in state 4, and other guy ack'ed last I frame we sent, transition to state 3. +// The same thing was done for receiving I frames after check_i_frame_ackd. + +// Thought: Could we simply call check_i_frame_ackd, for consistency, rather than only setting V(A)? + + if (cr == cr_res && pf == 0) { + + if (S->vs == S->va) { // all caught up with ack from other guy. + STOP_T1; + select_t1_value (S); + START_T3; + SET_RC(0); + enter_new_state (S, state_3_connected, __func__, __LINE__); + } + } } else { nr_error_recovery (S); enter_new_state (S, S->modulo == 128 ? state_5_awaiting_v22_connection : state_1_awaiting_connection, __func__, __LINE__); } } - break; } @@ -3261,7 +3759,7 @@ static void rej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr) SET_VA(nr); if (S->vs == S->va) { START_T3; - S->rc =0; // My enhancement. See Erratum note in select_t1_value. + SET_RC(0); // My enhancement. See Erratum note in select_t1_value. enter_new_state (S, state_3_connected, __func__, __LINE__); } else { @@ -3324,6 +3822,8 @@ static void rej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr) * cr - Is this command or response? * f - Final bit. When set, it is ack-ing up thru N(R)-1 * nr - N(R) from the frame. Peer has asked for a resend of I frame with this N(S). + * info - Information field, used only for Multi-SREJ + * info_len - Information field length, bytes. * * Description: 4.3.2.4. Selective Reject (SREJ) Command and Response * @@ -3354,7 +3854,7 @@ static void rej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr) * * The SREJ command/response initiates more-efficient error recovery by requesting the retransmission of a * single I frame following the detection of a sequence error. This is an advancement over the earlier versions in - * which the requested I frame was retransmitted togther with all additional I frames subsequently transmitted and + * which the requested I frame was retransmitted together with all additional I frames subsequently transmitted and * successfully received. * * When a TNC sends one or more SREJ commands, each with the P bit set to "0" or "1", or one or more SREJ @@ -3418,7 +3918,9 @@ static void rej_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, int nr) * *------------------------------------------------------------------------------*/ -static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) +static int resend_for_srej (ax25_dlsm_t *S, int nr, unsigned char *info, int info_len); + +static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr, unsigned char *info, int info_len) { switch (S->state) { @@ -3431,7 +3933,7 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) case state_5_awaiting_v22_connection: // Do nothing. - // Erratum: The original spec said stay in same state. + // Erratum: The original spec said stay in same state. (Seems correct.) // 2006 revision shows state 5 transitioning into 1. I think that is wrong. // probably a cut-n-paste from state 1 to 5 and that part not updated. break; @@ -3475,28 +3977,13 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) START_T3; select_t1_value(S); - // Resend I frame with N(S) equal to the N(R) in the SREJ. - // Note: X.25 says info part can contain additional sequence numbers. - // Here we stay with Ax.25 and don't consider the info part. - cdata_t *txdata = S->txdata_by_ns[nr]; - - if (txdata != NULL) { - - cmdres_t cr = cr_cmd; - int i_frame_ns = nr; - int i_frame_nr = S->vr; - int p = 0; - - packet_t pp = ax25_i_frame (S->addrs, S->num_addr, cr, S->modulo, i_frame_nr, i_frame_ns, p, txdata->pid, (unsigned char *)(txdata->data), txdata->len); - - // dw_printf ("calling lm_data_request for I frame, %s line %d\n", __func__, __LINE__); - - lm_data_request (S->chan, TQ_PRIO_1_LO, pp); + int num_resent = resend_for_srej (S, nr, info, info_len); + if (num_resent) { // my addition // Erratum: We sent I frame(s) and want to timeout if no ack comes back. -// We also sent N(R) so no need for extra RR at the end only for that. +// We also sent N(R), from V(R), so no need for extra RR at the end only for that. // We would sometimes end up in a situation where T1 was stopped on // both ends and everyone would wait for the other guy to timeout and do something. @@ -3506,10 +3993,6 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) START_T1; S->acknowledge_pending = 0; } - else { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Stream %d: INTERNAL ERROR for SREJ. I frame for N(S)=%d is not available.\n", S->stream_id, nr); - } // keep same state. } else { @@ -3521,6 +4004,11 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) case state_4_timer_recovery: + if (s_debug_timers) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("state 4 timer recovery, %s %d nr=%d, f=%d\n", __func__, __LINE__, nr, f); + } + S->peer_receiver_busy = 0; // Erratum: Original Flow chart has "check need for response here." @@ -3544,14 +4032,29 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) if (f) { // f=1 means ack up thru previous sequence. // Erratum: 2006 version tests "P". Original has "F." SET_VA(nr); + + if (s_debug_timers) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("state 4 timer recovery, %s %d set v(a)= %d\n", __func__, __LINE__, S->va); + } } if (S->vs == S->va) { // ACKs all caught up. Back to state 3. + // Erratum: I think this is unreachable. + // If the other side is asking for I frame with sequence X, it must have + // received X+1 or later. That means my V(S) must be X+2 or greater. + // So, I don't think we can ever have V(S) == V(A) here. + // If we were to remove the 'if' test and true part, case 4 would then + // be exactly the same as state 4. We need to rely on RR to get us + // back to state 3. + START_T3; - S->rc =0; // My enhancement. See Erratum note in select_t1_value. + SET_RC(0); // My enhancement. See Erratum note in select_t1_value. enter_new_state (S, state_3_connected, __func__, __LINE__); + // text_color_set(DW_COLOR_ERROR); + // dw_printf ("state 4 timer recovery, go to state 3 \n"); } else { @@ -3560,21 +4063,11 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) #if 1 // This is from the original protocol spec. // Resend I frame with N(S) equal to the N(R) in the SREJ. - cdata_t *txdata = S->txdata_by_ns[nr]; - - if (txdata != NULL) { - - cmdres_t cr = cr_cmd; - int i_frame_ns = nr; - int i_frame_nr = S->vr; - int p = 0; - - packet_t pp = ax25_i_frame (S->addrs, S->num_addr, cr, S->modulo, i_frame_nr, i_frame_ns, p, txdata->pid, (unsigned char *)(txdata->data), txdata->len); - - // dw_printf ("calling lm_data_request for I frame, %s line %d\n", __func__, __LINE__); - - lm_data_request (S->chan, TQ_PRIO_1_LO, pp); + //text_color_set(DW_COLOR_ERROR); + //dw_printf ("state 4 timer recovery, send requested frame(s) \n"); + int num_resent = resend_for_srej (S, nr, info, info_len); + if (num_resent) { // my addition // Erratum: We sent I frame(s) and want to timeout if no ack comes back. // We also sent N(R), from V(R), so no need for extra RR at the end only for that. @@ -3587,11 +4080,6 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) START_T1; S->acknowledge_pending = 0; } - else { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Stream %d: INTERNAL ERROR for SREJ. I frame for N(S)=%d is not available.\n", S->stream_id, nr); - } - #else // Erratum! This is from the 2006 revision. // We should resend only the single requested I frame. // I think there was a cut-n-paste from the REJ flow chart and this particular place did not get changed. @@ -3609,6 +4097,87 @@ static void srej_frame (ax25_dlsm_t *S, cmdres_t cr, int f, int nr) } /* end srej_frame */ +/*------------------------------------------------------------------------------ + * + * Name: resend_for_srej + * + * Purpose: Resend the I frame(s) specified in SREJ response. + * + * Inputs: S - Data Link State Machine. + * nr - N(R) from the frame. Peer has asked for a resend of I frame with this N(S). + * info - Information field, might contain additional sequence numbers for Multi-SREJ. + * info_len - Information field length, bytes. + * + * Returns: Number of frames sent. Should be at least one. + * + * Description: Simply resend requested frame(s). + * The calling context will worry about the F bit and other state stuff. + * + *------------------------------------------------------------------------------*/ + +static int resend_for_srej (ax25_dlsm_t *S, int nr, unsigned char *info, int info_len) +{ + cmdres_t cr = cr_cmd; + int i_frame_nr = S->vr; + int i_frame_ns = nr; + int p = 0; + int num_resent = 0; + + // Resend I frame with N(S) equal to the N(R) in the SREJ. + // Additional sequence numbers can be in optional information part. + + cdata_t *txdata = S->txdata_by_ns[i_frame_ns]; + + if (txdata != NULL) { + packet_t pp = ax25_i_frame (S->addrs, S->num_addr, cr, S->modulo, i_frame_nr, i_frame_ns, p, txdata->pid, (unsigned char *)(txdata->data), txdata->len); + // dw_printf ("calling lm_data_request for I frame, %s line %d\n", __func__, __LINE__); + lm_data_request (S->chan, TQ_PRIO_1_LO, pp); + num_resent++; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Stream %d: INTERNAL ERROR for SREJ. I frame for N(S)=%d is not available.\n", S->stream_id, i_frame_ns); + } + +// Multi-SREJ if there is an information part. + + int j; + for (j = 0; j < info_len; j++) { + + // We can have a single sequence number like this: + // xxx00000 (mod 8) + // xxxxxxx0 (mod 128) + // or we can have span (mod 128 only) like this, with the first and last: + // xxxxxxx1 + // xxxxxxx1 + // + // Note that the sequence number is shifted left by one + // and if the LSB is set, there should be two adjacent bytes + // with it set. + + if (S->modulo == 8) { + i_frame_ns = (info[j] >> 5) & 0x07; // no provision for span. + } + else { + i_frame_ns = (info[j] >> 1) & 0x7f; // TODO: test LSB and possible loop here. + } + + txdata = S->txdata_by_ns[i_frame_ns]; + if (txdata != NULL) { + packet_t pp = ax25_i_frame (S->addrs, S->num_addr, cr, S->modulo, i_frame_nr, i_frame_ns, p, txdata->pid, (unsigned char *)(txdata->data), txdata->len); + lm_data_request (S->chan, TQ_PRIO_1_LO, pp); + num_resent++; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Stream %d: INTERNAL ERROR for Multi-SREJ. I frame for N(S)=%d is not available.\n", S->stream_id, i_frame_ns); + } + } + return (num_resent); + +} /* end resend_for_srej */ + + /*------------------------------------------------------------------------------ @@ -3701,7 +4270,7 @@ static void sabm_e_frame (ax25_dlsm_t *S, int extended, int p) INIT_T1V_SRT; START_T3; - S->rc =0; // My enhancement. See Erratum note in select_t1_value. + SET_RC(0); // My enhancement. See Erratum note in select_t1_value. enter_new_state (S, state_3_connected, __func__, __LINE__); break; @@ -3809,7 +4378,7 @@ static void sabm_e_frame (ax25_dlsm_t *S, int extended, int p) SET_VS(0); SET_VA(0); SET_VR(0); - S->rc =0; // My enhancement. See Erratum note in select_t1_value. + SET_RC(0); // My enhancement. See Erratum note in select_t1_value. enter_new_state (S, state_3_connected, __func__, __LINE__); } break; @@ -3935,7 +4504,7 @@ static void disc_frame (ax25_dlsm_t *S, int p) * earliest opportunity. If the TNC is not capable of accepting a SABME command, it responds with a DM frame. * * A TNC that uses a version of AX.25 prior to v2.2 responds with a FRMR. - * ( I think the KPC-3+ has a bug - it replys with DM - WB2OSZ ) + * ( I think the KPC-3+ has a bug - it replies with DM - WB2OSZ ) * * 4.3.3.5. Disconnected Mode (DM) Response * @@ -4162,7 +4731,7 @@ static void ua_frame (ax25_dlsm_t *S, int f) if (f == 1) { if (S->layer_3_initiated) { text_color_set(DW_COLOR_INFO); - // TODO: add via if apppropriate. + // TODO: add via if appropriate. dw_printf ("Stream %d: Connected to %s. (%s)\n", S->stream_id, S->addrs[PEERCALL], S->state == state_5_awaiting_v22_connection ? "v2.2" : "v2.0"); // There is a subtle difference here between connect confirm and indication. // connect *confirm* means "has been made" @@ -4237,7 +4806,7 @@ static void ua_frame (ax25_dlsm_t *S, int f) mdl_negotiate_request (S); } - S->rc =0; // My enhancement. See Erratum note in select_t1_value. + SET_RC(0); // My enhancement. See Erratum note in select_t1_value. enter_new_state (S, state_3_connected, __func__, __LINE__); } else { @@ -4515,7 +5084,7 @@ static void ui_frame (ax25_dlsm_t *S, cmdres_t cr, int pf) static void xid_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, unsigned char *info_ptr, int info_len) { struct xid_param_s param; - char desc[120]; + char desc[150]; int ok; unsigned char xinfo[40]; int xlen; @@ -4542,7 +5111,7 @@ static void xid_frame (ax25_dlsm_t *S, cmdres_t cr, int pf, unsigned char *info_ if (ok) { negotiation_response (S, ¶m); - xlen = xid_encode (¶m, xinfo); + xlen = xid_encode (¶m, xinfo, res); pp = ax25_u_frame (S->addrs, S->num_addr, res, frame_type_U_XID, f, nopid, xinfo, xlen); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); @@ -4782,7 +5351,7 @@ static void t1_expiry (ax25_dlsm_t *S) packet_t pp; - S->rc++; + SET_RC(S->rc+1); if (S->rc > S->peak_rc_value) S->peak_rc_value = S->rc; // Keep statistics. pp = ax25_u_frame (S->addrs, S->num_addr, cmd, (S->state == state_5_awaiting_v22_connection) ? frame_type_U_SABME : frame_type_U_SABM, p, nopid, NULL, 0); @@ -4808,7 +5377,7 @@ static void t1_expiry (ax25_dlsm_t *S) packet_t pp; - S->rc++; + SET_RC(S->rc+1); if (S->rc > S->peak_rc_value) S->peak_rc_value = S->rc; pp = ax25_u_frame (S->addrs, S->num_addr, cmd, frame_type_U_DISC, p, nopid, NULL, 0); @@ -4821,7 +5390,7 @@ static void t1_expiry (ax25_dlsm_t *S) case state_3_connected: - S->rc = 1; + SET_RC(1); transmit_enquiry (S); enter_new_state (S, state_4_timer_recovery, __func__, __LINE__); break; @@ -4832,25 +5401,25 @@ static void t1_expiry (ax25_dlsm_t *S) // Erratum: 2006 version, page 103, is missing yes/no labels on decision blocks. - if (S->va != S->vr) { + if (S->va != S->vs) { if (s_debug_protocol_errors) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Stream %d: AX.25 Protocol Error I: N2 timeouts: unacknowledged data.\n", S->stream_id); + dw_printf ("Stream %d: AX.25 Protocol Error I: %d timeouts: unacknowledged sent data.\n", S->stream_id, S->n2_retry); } } else if (S->peer_receiver_busy) { if (s_debug_protocol_errors) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Stream %d: AX.25 Protocol Error U: N2 timeouts: extended peer busy condition.\n", S->stream_id); + dw_printf ("Stream %d: AX.25 Protocol Error U: %d timeouts: extended peer busy condition.\n", S->stream_id, S->n2_retry); } } else { if (s_debug_protocol_errors) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Stream %d: AX.25 Protocol Error T: N2 timeouts: no response to enquiry.\n", S->stream_id); + dw_printf ("Stream %d: AX.25 Protocol Error T: %d timeouts: no response to enquiry.\n", S->stream_id, S->n2_retry); } } @@ -4875,7 +5444,7 @@ static void t1_expiry (ax25_dlsm_t *S) enter_new_state (S, state_0_disconnected, __func__, __LINE__); } else { - S->rc++; + SET_RC(S->rc+1); if (S->rc > S->peak_rc_value) S->peak_rc_value = S->rc; // gather statistics. transmit_enquiry (S); @@ -4939,7 +5508,7 @@ static void t3_expiry (ax25_dlsm_t *S) // Erratum: Original sets RC to 0, 2006 revision sets RC to 1 which makes more sense. - S->rc = 1; + SET_RC(1); transmit_enquiry (S); enter_new_state (S, state_4_timer_recovery, __func__, __LINE__); break; @@ -5002,7 +5571,7 @@ static void tm201_expiry (ax25_dlsm_t *S) initiate_negotiation (S, ¶m); - xlen = xid_encode (¶m, xinfo); + xlen = xid_encode (¶m, xinfo, cmd); pp = ax25_u_frame (S->addrs, S->num_addr, cmd, frame_type_U_XID, p, nopid, xinfo, xlen); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); @@ -5076,7 +5645,7 @@ static void establish_data_link (ax25_dlsm_t *S) // Flow chart shows setting RC to 0 and we end up sending SAMB(e) 11 times when N2 (RETRY) is 10. // It should be 1 rather than 0. - S->rc = 1; + SET_RC(1); pp = ax25_u_frame (S->addrs, S->num_addr, cmd, (S->modulo == 128) ? frame_type_U_SABME : frame_type_U_SABM, p, nopid, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); STOP_T3; @@ -5141,7 +5710,7 @@ static void clear_exception_conditions (ax25_dlsm_t *S) * * Other guy gets RR/RNR command P=1. * Same action for either state 3 or 4. - * Whether he has outstanding un-ack'ed sent I frames is irrelevent. + * Whether he has outstanding un-ack'ed sent I frames is irrelevant. * He calls "enquiry response" which sends RR/RNR response F=1. * (Read about detour 1 below and in enquiry_response.) * @@ -5171,14 +5740,14 @@ static void transmit_enquiry (ax25_dlsm_t *S) if (s_debug_retry) { text_color_set(DW_COLOR_ERROR); - dw_printf ("\n****** TRANSMIT ENQUIRY ******\n\n"); + dw_printf ("\n****** TRANSMIT ENQUIRY RR/RNR cmd P=1 ****** state=%d, rc=%d\n\n", S->state, S->rc); } // This is the ONLY place that we send RR/RNR *command* with P=1. // Everywhere else should be response. // I don't think we ever use RR/RNR command P=0 but need to check on that. - pp = ax25_s_frame (S->addrs, S->num_addr, cmd, S->own_receiver_busy ? frame_type_S_RNR : frame_type_S_RR, S->modulo, nr, p); + pp = ax25_s_frame (S->addrs, S->num_addr, cmd, S->own_receiver_busy ? frame_type_S_RNR : frame_type_S_RR, S->modulo, nr, p, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); @@ -5222,7 +5791,7 @@ static void transmit_enquiry (ax25_dlsm_t *S) * The next response frame returned to a supervisory command frame with the P bit set to "1", received during * the information transfer state, is an RR, RNR or REJ response frame with the F bit set to "1". * - * Erattum! The flow chart says RR/RNR *command* but I'm confident should be response. + * Erattum! The flow chart says RR/RNR *command* but I'm confident it should be response. * * Erratum: Ax.25 spec has nothing here for SREJ. See X.25 2.4.6.11 for explanation. * @@ -5252,13 +5821,14 @@ static void enquiry_response (ax25_dlsm_t *S, ax25_frame_type_t frame_type, int // I'm busy. - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RNR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RNR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); - S->acknowledge_pending = 0; + S->acknowledge_pending = 0; // because we sent N(R) from V(R). } - else if (S->srej_enabled) { + else if (S->srej_enable == srej_single || S->srej_enable == srej_multi) { + // SREJ is enabled. This is based on X.25 2.4.6.11. @@ -5305,7 +5875,7 @@ static void enquiry_response (ax25_dlsm_t *S, ax25_frame_type_t frame_type, int // Not waiting for fill in of missing frames. X.25 2.4.6.11 c) - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; @@ -5319,7 +5889,12 @@ static void enquiry_response (ax25_dlsm_t *S, ax25_frame_type_t frame_type, int // And when we look at what happens when RR response, F=1 is received in state 4, it is // effectively REJ when N(R) is not the same as V(S). - pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f); + if (s_debug_retry) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\n****** ENQUIRY RESPONSE srej not enbled, sending RR resp F=%d ******\n\n", f); + } + + pp = ax25_s_frame (S->addrs, S->num_addr, cr, frame_type_S_RR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; @@ -5331,7 +5906,7 @@ static void enquiry_response (ax25_dlsm_t *S, ax25_frame_type_t frame_type, int // For cases other than (RR, RNR, I) command, P=1. - pp = ax25_s_frame (S->addrs, S->num_addr, cr, S->own_receiver_busy ? frame_type_S_RNR : frame_type_S_RR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, S->own_receiver_busy ? frame_type_S_RNR : frame_type_S_RR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; @@ -5343,7 +5918,7 @@ static void enquiry_response (ax25_dlsm_t *S, ax25_frame_type_t frame_type, int // Erratum: This is woefully inadequate when SREJ is enabled. // Erratum: Flow chart says RR/RNR command but I'm confident it should be response. - pp = ax25_s_frame (S->addrs, S->num_addr, cr, S->own_receiver_busy ? frame_type_S_RNR : frame_type_S_RR, S->modulo, nr, f); + pp = ax25_s_frame (S->addrs, S->num_addr, cr, S->own_receiver_busy ? frame_type_S_RNR : frame_type_S_RR, S->modulo, nr, f, NULL, 0); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); S->acknowledge_pending = 0; @@ -5363,9 +5938,14 @@ static void enquiry_response (ax25_dlsm_t *S, ax25_frame_type_t frame_type, int * Continue will all up to and including current V(S) value. * * Description: Resend one or more frames that have already been sent. + * Should always send at least one. * * This is probably the result of getting REJ asking for a resend. * + * Context: I would expect the caller to clear 'acknowledge_pending' after calling this + * because we sent N(R), from V(R), to ack what was received from other guy. + * I would also expect Stop T3 & Start T1 at the same place. + * *------------------------------------------------------------------------------*/ static void invoke_retransmission (ax25_dlsm_t *S, int nr_input) @@ -5379,6 +5959,19 @@ static void invoke_retransmission (ax25_dlsm_t *S, int nr_input) int local_vs; int sent_count = 0; + + if (s_debug_misc) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("invoke_retransmission(): starting with %d, state=%d, rc=%d, \n", nr_input, S->state, S->rc); + } + +// I don't think we should be here if SREJ is enabled. +// TODO: Figure out why this happens occasionally. + +// if (S->srej_enable != srej_none) { +// text_color_set(DW_COLOR_ERROR); +// dw_printf ("Internal Error, Did not expect to be here when SREJ enabled. %s %s %d\n", __FILE__, __func__, __LINE__); +// } if (S->txdata_by_ns[nr_input] == NULL) { text_color_set(DW_COLOR_ERROR); @@ -5399,7 +5992,7 @@ static void invoke_retransmission (ax25_dlsm_t *S, int nr_input) if (s_debug_misc) { text_color_set(DW_COLOR_INFO); - dw_printf ("invoke_retransmission(): state=%d, Resending N(S) = %d, probably as result of REJ N(R) = %d\n", S->state, ns, nr_input); + dw_printf ("invoke_retransmission(): Resending N(S) = %d\n", ns); } packet_t pp = ax25_i_frame (S->addrs, S->num_addr, cr, S->modulo, nr, ns, p, @@ -5438,7 +6031,9 @@ static void invoke_retransmission (ax25_dlsm_t *S, int nr_input) * * Outputs: S->va - updated from nr. * - * Description: TBD... Document when this is used. + * Description: This is called for: + * - 'I' frame received and N(R) is in expected range, states 3 & 4. + * - RR/RNR command with p=1 received and N(R) is in expected range, state 3 only. * *------------------------------------------------------------------------------*/ @@ -5535,7 +6130,7 @@ static void check_need_for_response (ax25_dlsm_t *S, ax25_frame_type_t frame_typ * * Outputs: S->srt New smoothed roundtrip time. * - * S->t1v How long to wait for an acknowlegement before resending. + * S->t1v How long to wait for an acknowledgement before resending. * Value used when starting timer T1, in seconds. * Here it is dynamically adjusted. * @@ -5634,7 +6229,7 @@ static void select_t1_value (ax25_dlsm_t *S) // This goes up exponentially if implemented as documented! // For example, if we were trying to connect to a station which is not there, we - // would retry after 3, the 8, 16, 32, ... and not time out for over an hour. + // would retry after 3, then 8, 16, 32, ... and not time out for over an hour. // That's ridiculous. Let's try increasing it by a quarter second each time. // We now give up after about a minute. @@ -5651,12 +6246,30 @@ static void select_t1_value (ax25_dlsm_t *S) } +// See https://groups.io/g/direwolf/topic/100782658#8542 +// Perhaps the demands of file transfer lead to this problem. + +// "Temporary" hack. +// Automatic fine tuning of t1v generally works well, but on very rare occasions, it gets wildly out of control. +// Until I have more time to properly diagnose this, add some guardrails so it does not go flying off a cliff. + +// The initial value of t1v is frack + frack * 2 (number of digipeateers in path) +// If anything, it should automatically be adjusted down. +// Let's say, something smells fishy if it exceeds twice that initial value. + +// TODO: Add some instrumentation to record where this was called from and all the values in the printf below. + +#if 1 + if (S->t1v < 0.25 || S->t1v > 2 * (g_misc_config_p->frack * (2 * (S->num_addr - 2) + 1)) ) { + INIT_T1V_SRT; + } +#else if (S->t1v < 0.99 || S->t1v > 30) { text_color_set(DW_COLOR_ERROR); dw_printf ("INTERNAL ERROR? Stream %d: select_t1_value, rc = %d, t1 remaining = %.3f, old srt = %.3f, new srt = %.3f, Extreme new t1v = %.3f\n", S->stream_id, S->rc, S->t1_remaining_when_last_stopped, old_srt, S->srt, S->t1v); } - +#endif } /* end select_t1_value */ @@ -5670,7 +6283,7 @@ static void select_t1_value (ax25_dlsm_t *S) static void set_version_2_0 (ax25_dlsm_t *S) { - S->srej_enabled = 0; + S->srej_enable = srej_none; S->modulo = 8; S->n1_paclen = g_misc_config_p->paclen; S->k_maxframe = g_misc_config_p->maxframe_basic; @@ -5687,8 +6300,8 @@ static void set_version_2_0 (ax25_dlsm_t *S) static void set_version_2_2 (ax25_dlsm_t *S) { - S->srej_enabled = 1; - //S->srej_enabled = 0; // temporarily disable for testing of REJ only with modulo 128 + S->srej_enable = srej_single; // Start with single. + // Can be increased to multi with XID exchange. S->modulo = 128; S->n1_paclen = g_misc_config_p->paclen; S->k_maxframe = g_misc_config_p->maxframe_extended; @@ -5725,7 +6338,7 @@ static void set_version_2_2 (ax25_dlsm_t *S) * * Pattern noticed: Anytime we have "is_good_nr" returning true, we should always * - set V(A) from N(R) or - * - call "check_i_frame_acked" which does the same and some timer stuff. + * - call "check_i_frame_ackd" which does the same and some timer stuff. * *------------------------------------------------------------------------------*/ @@ -5772,7 +6385,7 @@ static int is_good_nr (ax25_dlsm_t *S, int nr) * k * * Outputs: v(s) is incremented for each processed. - * ack_pending = 0 + * acknowledge_pending = 0 * *------------------------------------------------------------------------------*/ @@ -5843,7 +6456,7 @@ static void i_frame_pop_off_queue (ax25_dlsm_t *S) while ( ( ! S->peer_receiver_busy ) && S->i_frame_queue != NULL && - S->vs != AX25MODULO(S->va + S->k_maxframe, S->modulo, __FILE__, __func__, __LINE__) ) { + WITHIN_WINDOW_SIZE(S) ) { cdata_t *txdata; @@ -5943,7 +6556,7 @@ static void discard_i_queue (ax25_dlsm_t *S) * *------------------------------------------------------------------------------*/ -// TODO: requeuing... +// TODO: requeuing??? static void enter_new_state (ax25_dlsm_t *S, enum dlsm_state_e new_state, const char *from_func, int from_line) { @@ -5999,7 +6612,18 @@ static void mdl_negotiate_request (ax25_dlsm_t *S) int p = 1; int nopid = 0; packet_t pp; + int n; + +// At least one known [partial] v2.2 implementation understands SABME but not XID. +// Rather than wasting time, sending XID repeatedly until giving up, we have a workaround. +// The configuration file can contain a list of stations known not to respond to XID. +// Obviously this applies only to v2.2 because XID was not part of v2.0. + for (n = 0; n < g_misc_config_p->noxid_count; n++) { + if (strcmp(S->addrs[PEERCALL],g_misc_config_p->noxid_addrs[n]) == 0) { + return; + } + } switch (S->mdl_state) { @@ -6007,7 +6631,7 @@ static void mdl_negotiate_request (ax25_dlsm_t *S) initiate_negotiation (S, ¶m); - xlen = xid_encode (¶m, xinfo); + xlen = xid_encode (¶m, xinfo, cmd); pp = ax25_u_frame (S->addrs, S->num_addr, cmd, frame_type_U_XID, p, nopid, xinfo, xlen); lm_data_request (S->chan, TQ_PRIO_1_LO, pp); @@ -6040,7 +6664,17 @@ static void mdl_negotiate_request (ax25_dlsm_t *S) static void initiate_negotiation (ax25_dlsm_t *S, struct xid_param_s *param) { param->full_duplex = 0; - param->rej = S->srej_enabled ? selective_reject : implicit_reject; + switch (S->srej_enable) { + case srej_single: + case srej_multi: + param->srej = srej_multi; // see if other end reconizes it. + break; + case srej_none: + default: + param->srej = srej_none; + break; + } + param->modulo = S->modulo; param->i_field_length_rx = S->n1_paclen; // Hmmmm. Should we ask for what the user // specified for PACLEN or offer the maximum @@ -6074,9 +6708,7 @@ static void initiate_negotiation (ax25_dlsm_t *S, struct xid_param_s *param) static void negotiation_response (ax25_dlsm_t *S, struct xid_param_s *param) { -// Full duplex would not be that difficult. -// Just ignore the Carrier Detect when transmitting. -// But we haven't done that yet. +// TODO: Integrate with new full duplex capability in v1.5. param->full_duplex = 0; @@ -6094,11 +6726,8 @@ static void negotiation_response (ax25_dlsm_t *S, struct xid_param_s *param) // Erratum: 2006 version, section, 4.3.3.7 says default selective reject - reject. // We can't do that. - if (param->rej == unknown_reject) { - param->rej = (param->modulo == 128) ? selective_reject : implicit_reject; // not specified, set default - } - else { - param->rej = MIN(param->rej, selective_reject); + if (param->srej == srej_not_specified) { + param->srej = (param->modulo == 128) ? srej_single : srej_none; // not specified, set default } // We can currently do up to 2k. @@ -6166,8 +6795,8 @@ static void negotiation_response (ax25_dlsm_t *S, struct xid_param_s *param) static void complete_negotiation (ax25_dlsm_t *S, struct xid_param_s *param) { - if (param->rej != unknown_reject) { - S->srej_enabled = param->rej >= selective_reject; + if (param->srej != srej_not_specified) { + S->srej_enable = param->srej; } if (param->modulo != modulo_unknown) { diff --git a/ax25_link.h b/src/ax25_link.h similarity index 88% rename from ax25_link.h rename to src/ax25_link.h index 060b6ab4..40fa401b 100644 --- a/ax25_link.h +++ b/src/ax25_link.h @@ -52,6 +52,8 @@ void ax25_link_init (struct misc_config_s *pconfig); // These functions must be called on a single thread, one at a time. // The Data Link Queue (DLQ) is used to serialize events from multiple sources. +// Maybe the dispatch switch should be moved to ax25_link.c so they can all +// be made static and they can't be called from the wrong place accidentally. void dl_connect_request (dlq_item_t *E); @@ -63,11 +65,15 @@ void dl_register_callsign (dlq_item_t *E); void dl_unregister_callsign (dlq_item_t *E); +void dl_outstanding_frames_request (dlq_item_t *E); + void dl_client_cleanup (dlq_item_t *E); void lm_data_indication (dlq_item_t *E); +void lm_seize_confirm (dlq_item_t *E); + void lm_channel_busy (dlq_item_t *E); diff --git a/ax25_pad.c b/src/ax25_pad.c similarity index 89% rename from ax25_pad.c rename to src/ax25_pad.c index 37bc3166..0f075808 100644 --- a/ax25_pad.c +++ b/src/ax25_pad.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011 , 2013, 2014, 2015 John Langner, WB2OSZ +// Copyright (C) 2011 , 2013, 2014, 2015, 2019 John Langner, WB2OSZ // // 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 @@ -49,7 +49,7 @@ * * * APRS uses only UI frames. - * Each starts with 2-10 addressses (14-70 octets): + * Each starts with 2-10 addresses (14-70 octets): * * * Destination Address (note: opposite order in printed format) * @@ -87,18 +87,18 @@ * http://www.aprs.org/aprs12/preemptive-digipeating.txt * http://www.aprs.org/aprs12/RR-bits.txt * - * I don't recall why I originally intended to set the source/destination C bits both to 1. + * I don't recall why I originally set the source & destination C bits both to 1. * Reviewing this 5 years later, after spending more time delving into the * AX.25 spec, I think it should be 1 for destination and 0 for source. * In practice you see all four combinations being used by APRS stations - * and no one really cares about these two bits. + * and everyone apparently ignores them for APRS. They do make a big + * difference for connected mode. * * The final octet of the Source has the form: * * C R R SSID 0, where, * - * C = command/response = 1 (originally, now I think it should be 0 for source.) - * (Haven't gone back to check to see what code actually does.) + * C = command/response = 0 * R R = Reserved = 1 1 * SSID = substation ID * 0 = zero (or 1 if no repeaters) @@ -350,8 +350,9 @@ void ax25_delete (packet_t this_p) * strict - True to enforce rules for packets sent over the air. * False to be more lenient for packets from IGate server. * - * Messages from an IGate server can have longer - * addresses after qAC. Up to 9 observed so far. + * Packets from an IGate server can have longer + * addresses after qAC. Up to 9 observed so far. + * The SSID can be 2 alphanumeric characters, not just 1 to 15. * * We can just truncate the name because we will only * end up discarding it. TODO: check on this. @@ -372,11 +373,8 @@ packet_t ax25_from_text (char *monitor, int strict) /* * Tearing it apart is destructive so make our own copy first. */ - char stuff[512]; - + char stuff[AX25_MAX_PACKET_LEN+1]; char *pinfo; - char *pa; - char *saveptr; /* Used with strtok_r because strtok is not thread safe. */ int ssid_temp, heard_temp; char atemp[AX25_MAX_ADDR_LEN]; @@ -384,6 +382,10 @@ packet_t ax25_from_text (char *monitor, int strict) char info_part[AX25_MAX_INFO_LEN+1]; int info_len; + // text_color_set(DW_COLOR_DEBUG); + // dw_printf ("DEBUG: ax25_from_text ('%s', %d)\n", monitor, strict); + // fflush(stdout); sleep(1); + packet_t this_p = ax25_new (); #if AX25MEMDEBUG @@ -410,14 +412,15 @@ packet_t ax25_from_text (char *monitor, int strict) this_p->frame_data[AX25_DESTINATION*7+6] = SSID_H_MASK | SSID_RR_MASK; memset (this_p->frame_data + AX25_SOURCE*7, ' ' << 1, 6); - this_p->frame_data[AX25_SOURCE*7+6] = SSID_H_MASK | SSID_RR_MASK | SSID_LAST_MASK; + this_p->frame_data[AX25_SOURCE*7+6] = SSID_RR_MASK | SSID_LAST_MASK; this_p->frame_data[14] = AX25_UI_FRAME; this_p->frame_data[15] = AX25_PID_NO_LAYER_3; this_p->frame_len = 7 + 7 + 1 + 1; this_p->num_addr = (-1); - assert (ax25_get_num_addr(this_p) == 2); + (void) ax25_get_num_addr(this_p); // when num_addr is -1, this sets it properly. + assert (this_p->num_addr == 2); /* @@ -440,9 +443,10 @@ packet_t ax25_from_text (char *monitor, int strict) /* * Source address. - * Don't use traditional strtok because it is not thread safe. */ - pa = strtok_r (stuff, ">", &saveptr); + + char *pnxt = stuff; + char *pa = strsep (&pnxt, ">"); if (pa == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Failed to create packet from text. No source address\n"); @@ -465,7 +469,7 @@ packet_t ax25_from_text (char *monitor, int strict) * Destination address. */ - pa = strtok_r (NULL, ",", &saveptr); + pa = strsep (&pnxt, ","); if (pa == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Failed to create packet from text. No destination address\n"); @@ -487,11 +491,33 @@ packet_t ax25_from_text (char *monitor, int strict) /* * VIA path. */ - while (( pa = strtok_r (NULL, ",", &saveptr)) != NULL && this_p->num_addr < AX25_MAX_ADDRS ) { - int k; +// Originally this used strtok_r. +// strtok considers all adjacent delimiters to be a single delimiter. +// This is handy for varying amounts of whitespace. +// It will never return a zero length string. +// All was good until this bizarre case came along: + +// AISAT-1>CQ,,::CQ-0 :From AMSAT INDIA & Exseed Space |114304|48|45|42{962 + +// Apparently there are two digipeater fields but they are empty. +// When we parsed this text representation, the extra commas were ignored rather +// than pointed out as being invalid. + +// Use strsep instead. This does not collapse adjacent delimiters. + + while (( pa = strsep (&pnxt, ",")) != NULL && this_p->num_addr < AX25_MAX_ADDRS ) { - k = this_p->num_addr; + int k = this_p->num_addr; + + // printf ("DEBUG: get digi loop, num addr = %d, address = '%s'\n", k, pa);// FIXME + + // Hack for q construct, from APRS-IS, so it does not cause panic later. + + if ( ! strict && pa[0] == 'q' && pa[1] == 'A') { + pa[0] = 'Q'; + pa[2] = toupper(pa[2]); + } if ( ! ax25_parse_addr (k, pa, strict, atemp, &ssid_temp, &heard_temp)) { text_color_set(DW_COLOR_ERROR); @@ -707,7 +733,7 @@ packet_t ax25_dup (packet_t copy_from) * * in_addr - Input such as "WB2OSZ-15*" * - * strict - TRUE for strict checking (6 characters, no lower case, + * strict - 1 (true) for strict checking (6 characters, no lower case, * SSID must be in range of 0 to 15). * Strict is appropriate for packets sent * over the radio. Communication with IGate @@ -715,6 +741,9 @@ packet_t ax25_dup (packet_t copy_from) * alphanumeric characters for the SSID. * We also get messages like this from a server. * KB1POR>APU25N,TCPIP*,qAC,T2NUENGLD:... + * K1BOS-B>APOSB,TCPIP,WR2X-2*:... + * + * 2 (extra true) will complain if * is found at end. * * Outputs: out_addr - Address without any SSID. * Must be at least AX25_MAX_ADDR_LEN bytes. @@ -724,6 +753,7 @@ packet_t ax25_dup (packet_t copy_from) * out_heard - True if "*" found. * * Returns: True (1) if OK, false (0) if any error. + * When 0, out_addr, out_ssid, and out_heard are undefined. * * *------------------------------------------------------------------------------*/ @@ -744,35 +774,59 @@ int ax25_parse_addr (int position, char *in_addr, int strict, char *out_addr, in *out_ssid = 0; *out_heard = 0; - if (strict && strlen(in_addr) >= 2 && strncmp(in_addr, "qA", 2) == 0) { + // dw_printf ("ax25_parse_addr in: position=%d, '%s', strict=%d\n", position, in_addr, strict); + if (position < -1) position = -1; + if (position > AX25_REPEATER_8) position = AX25_REPEATER_8; + position++; /* Adjust for position_name above. */ + + if (strlen(in_addr) == 0) { text_color_set(DW_COLOR_ERROR); - dw_printf ("%sAddress \"%s\" is a \"q-construct\" used for communicating\n", position_name[position], in_addr); - dw_printf ("with APRS Internet Servers. It was not expected here.\n"); + dw_printf ("%sAddress \"%s\" is empty.\n", position_name[position], in_addr); + return 0; } - //dw_printf ("ax25_parse_addr in: %s\n", in_addr); + if (strict && strlen(in_addr) >= 2 && strncmp(in_addr, "qA", 2) == 0) { - if (position < -1) position = -1; - if (position > AX25_REPEATER_8) position = AX25_REPEATER_8; - position++; /* Adjust for position_name above. */ + text_color_set(DW_COLOR_ERROR); + dw_printf ("%sAddress \"%s\" is a \"q-construct\" used for communicating with\n", position_name[position], in_addr); + dw_printf ("APRS Internet Servers. It should never appear when going over the radio.\n"); + } + + // dw_printf ("ax25_parse_addr in: %s\n", in_addr); maxlen = strict ? 6 : (AX25_MAX_ADDR_LEN-1); p = in_addr; i = 0; - for (p = in_addr; isalnum(*p); p++) { + for (p = in_addr; *p != '\0' && *p != '-' && *p != '*'; p++) { if (i >= maxlen) { text_color_set(DW_COLOR_ERROR); dw_printf ("%sAddress is too long. \"%s\" has more than %d characters.\n", position_name[position], in_addr, maxlen); return 0; } + if ( ! isalnum(*p)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("%sAddress, \"%s\" contains character other than letter or digit in character position %d.\n", position_name[position], in_addr, (int)(long)(p-in_addr)+1); + return 0; + } + out_addr[i++] = *p; out_addr[i] = '\0'; + +#if DECAMAIN // Hack when running in decode_aprs utility. + // Exempt the "qA..." case because it was already mentioned. + + if (strict && islower(*p) && strncmp(in_addr, "qA", 2) != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("%sAddress has lower case letters. \"%s\" must be all upper case.\n", position_name[position], in_addr); + } +#else if (strict && islower(*p)) { text_color_set(DW_COLOR_ERROR); dw_printf ("%sAddress has lower case letters. \"%s\" must be all upper case.\n", position_name[position], in_addr); return 0; } +#endif } j = 0; @@ -804,15 +858,20 @@ int ax25_parse_addr (int position, char *in_addr, int strict, char *out_addr, in if (*p == '*') { *out_heard = 1; p++; + if (strict == 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\"*\" is not allowed at end of address \"%s\" here.\n", in_addr); + return 0; + } } if (*p != '\0') { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Invalid character \"%c\" found in %saddress \"%s\".\n", *p, position_name[position], in_addr); + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid character \"%c\" found in %saddress \"%s\".\n", *p, position_name[position], in_addr); return 0; } - //dw_printf ("ax25_parse_addr out: %s %d %d\n", out_addr, *out_ssid, *out_heard); + // dw_printf ("ax25_parse_addr out: '%s' %d %d\n", out_addr, *out_ssid, *out_heard); return (1); @@ -890,7 +949,7 @@ int ax25_check_addresses (packet_t pp) * * Name: ax25_unwrap_third_party * - * Purpose: Unwrap a third party messge from the header. + * Purpose: Unwrap a third party message from the header. * * Inputs: copy_from - Existing packet object. * @@ -958,6 +1017,11 @@ void ax25_set_addr (packet_t this_p, int n, char *ad) //dw_printf ("ax25_set_addr (%d, %s) num_addr=%d\n", n, ad, this_p->num_addr); + if (strlen(ad) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Set address error! Station address for position %d is empty!\n", n); + } + if (n >= 0 && n < this_p->num_addr) { //dw_printf ("ax25_set_addr , existing case\n"); @@ -1039,6 +1103,11 @@ void ax25_insert_addr (packet_t this_p, int n, char *ad) //dw_printf ("ax25_insert_addr (%d, %s)\n", n, ad); + if (strlen(ad) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Set address error! Station address for position %d is empty!\n", n); + } + /* Don't do it if we already have the maximum number. */ /* Should probably return success/fail code but currently the caller doesn't care. */ @@ -1257,13 +1326,31 @@ void ax25_get_addr_with_ssid (packet_t this_p, int n, char *station) return; } - memset (station, 0, 7); + // At one time this would stop at the first space, on the assumption we would have only trailing spaces. + // Then there was a forum discussion where someone encountered the address " WIDE2" with a leading space. + // In that case, we would have returned a zero length string here. + // Now we return exactly what is in the address field and trim trailing spaces. + // This will provide better information for troubleshooting. + for (i=0; i<6; i++) { - unsigned char ch; + station[i] = (this_p->frame_data[n*7+i] >> 1) & 0x7f; + } + station[6] = '\0'; - ch = (this_p->frame_data[n*7+i] >> 1) & 0x7f; - if (ch <= ' ') break; - station[i] = ch; + for (i=5; i>=0; i--) { + if (station[i] == '\0') { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Station address \"%s\" contains nul character. AX.25 protocol requires trailing ASCII spaces when less than 6 characters.\n", station); + } + else if (station[i] == ' ') + station[i] = '\0'; + else + break; + } + + if (strlen(station) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Station address, in position %d, is empty! This is not a valid AX.25 frame.\n", n); } ssid = ax25_get_ssid (this_p, n); @@ -1322,13 +1409,27 @@ void ax25_get_addr_no_ssid (packet_t this_p, int n, char *station) return; } - memset (station, 0, 7); + // At one time this would stop at the first space, on the assumption we would have only trailing spaces. + // Then there was a forum discussion where someone encountered the address " WIDE2" with a leading space. + // In that case, we would have returned a zero length string here. + // Now we return exactly what is in the address field and trim trailing spaces. + // This will provide better information for troubleshooting. + for (i=0; i<6; i++) { - unsigned char ch; + station[i] = (this_p->frame_data[n*7+i] >> 1) & 0x7f; + } + station[6] = '\0'; - ch = (this_p->frame_data[n*7+i] >> 1) & 0x7f; - if (ch <= ' ') break; - station[i] = ch; + for (i=5; i>=0; i--) { + if (station[i] == ' ') + station[i] = '\0'; + else + break; + } + + if (strlen(station) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Station address, in position %d, is empty! This is not a valid AX.25 frame.\n", n); } } /* end ax25_get_addr_no_ssid */ @@ -1625,6 +1726,19 @@ int ax25_get_info (packet_t this_p, unsigned char **paddr) } /* end ax25_get_info */ +void ax25_set_info (packet_t this_p, unsigned char *new_info_ptr, int new_info_len) +{ + unsigned char *old_info_ptr; + int old_info_len = ax25_get_info (this_p, &old_info_ptr); + this_p->frame_len -= old_info_len; + + if (new_info_len < 0) new_info_len = 0; + if (new_info_len > AX25_MAX_INFO_LEN) new_info_len = AX25_MAX_INFO_LEN; + memcpy (old_info_ptr, new_info_ptr, new_info_len); + this_p->frame_len += new_info_len; +} + + /*------------------------------------------------------------------------------ * * Name: ax25_cut_at_crlf @@ -1752,7 +1866,7 @@ packet_t ax25_get_nextp (packet_t this_p) * * Inputs: this_p - Current packet object. * - * release_time - Time as returned by dtime_now(). + * release_time - Time as returned by dtime_monotonic(). * *------------------------------------------------------------------------------*/ @@ -1800,6 +1914,25 @@ void ax25_set_modulo (packet_t this_p, int modulo) } +/*------------------------------------------------------------------------------ + * + * Name: ax25_get_modulo + * + * Purpose: Get modulo value for I and S frame sequence numbers. + * + * Returns: 8 or 128 if known. + * 0 if unknown. + * + *------------------------------------------------------------------------------*/ + +int ax25_get_modulo (packet_t this_p) +{ + assert (this_p->magic1 == MAGIC); + assert (this_p->magic2 == MAGIC); + + return (this_p->modulo); +} + @@ -1852,6 +1985,7 @@ void ax25_format_addrs (packet_t this_p, char *result) } ax25_get_addr_with_ssid (this_p, AX25_SOURCE, stemp); + // FIXME: For ALL strcat: Pass in sizeof result and use strlcat. strcat (result, stemp); strcat (result, ">"); @@ -1870,6 +2004,8 @@ void ax25_format_addrs (packet_t this_p, char *result) } strcat (result, ":"); + + // dw_printf ("DEBUG ax25_format_addrs, num_addr = %d, result = '%s'\n", this_p->num_addr, result); } @@ -1956,7 +2092,7 @@ int ax25_pack (packet_t this_p, unsigned char result[AX25_MAX_PACKET_LEN]) assert (this_p->magic1 == MAGIC); assert (this_p->magic2 == MAGIC); - assert (this_p->frame_len > 0 && this_p->frame_len <= AX25_MAX_PACKET_LEN); + assert (this_p->frame_len >= 0 && this_p->frame_len <= AX25_MAX_PACKET_LEN); memcpy (result, this_p->frame_data, this_p->frame_len); @@ -1977,7 +2113,7 @@ int ax25_pack (packet_t this_p, unsigned char result[AX25_MAX_PACKET_LEN]) * * Outputs: desc - Text description such as "I frame" or * "U frame SABME". - * Supply 40 bytes to be safe. + * Supply 56 bytes to be safe. * * cr - Command or response? * @@ -1993,7 +2129,7 @@ int ax25_pack (packet_t this_p, unsigned char result[AX25_MAX_PACKET_LEN]) // TODO: need someway to ensure caller allocated enough space. // Should pass in as parameter. -#define DESC_SIZ 40 +#define DESC_SIZ 56 ax25_frame_type_t ax25_frame_type (packet_t this_p, cmdres_t *cr, char *desc, int *pf, int *nr, int *ns) @@ -2264,16 +2400,30 @@ void ax25_hex_dump (packet_t this_p) dw_printf ("%s\n", cp_text); } + // Address fields must be only upper case letters and digits. + // If less than 6 characters, trailing positions are filled with ASCII space. + // Using all zero bits in one of these 6 positions is wrong. + // Any non printable characters will be printed as "." here. - dw_printf (" dest %c%c%c%c%c%c %2d c/r=%d res=%d last=%d\n", - fptr[0]>>1, fptr[1]>>1, fptr[2]>>1, fptr[3]>>1, fptr[4]>>1, fptr[5]>>1, + dw_printf (" dest %c%c%c%c%c%c %2d c/r=%d res=%d last=%d\n", + isprint(fptr[0]>>1) ? fptr[0]>>1 : '.', + isprint(fptr[1]>>1) ? fptr[1]>>1 : '.', + isprint(fptr[2]>>1) ? fptr[2]>>1 : '.', + isprint(fptr[3]>>1) ? fptr[3]>>1 : '.', + isprint(fptr[4]>>1) ? fptr[4]>>1 : '.', + isprint(fptr[5]>>1) ? fptr[5]>>1 : '.', (fptr[6]&SSID_SSID_MASK)>>SSID_SSID_SHIFT, (fptr[6]&SSID_H_MASK)>>SSID_H_SHIFT, (fptr[6]&SSID_RR_MASK)>>SSID_RR_SHIFT, fptr[6]&SSID_LAST_MASK); dw_printf (" source %c%c%c%c%c%c %2d c/r=%d res=%d last=%d\n", - fptr[7]>>1, fptr[8]>>1, fptr[9]>>1, fptr[10]>>1, fptr[11]>>1, fptr[12]>>1, + isprint(fptr[7]>>1) ? fptr[7]>>1 : '.', + isprint(fptr[8]>>1) ? fptr[8]>>1 : '.', + isprint(fptr[9]>>1) ? fptr[9]>>1 : '.', + isprint(fptr[10]>>1) ? fptr[10]>>1 : '.', + isprint(fptr[11]>>1) ? fptr[11]>>1 : '.', + isprint(fptr[12]>>1) ? fptr[12]>>1 : '.', (fptr[13]&SSID_SSID_MASK)>>SSID_SSID_SHIFT, (fptr[13]&SSID_H_MASK)>>SSID_H_SHIFT, (fptr[13]&SSID_RR_MASK)>>SSID_RR_SHIFT, @@ -2283,7 +2433,12 @@ void ax25_hex_dump (packet_t this_p) dw_printf (" digi %d %c%c%c%c%c%c %2d h=%d res=%d last=%d\n", n - 1, - fptr[n*7+0]>>1, fptr[n*7+1]>>1, fptr[n*7+2]>>1, fptr[n*7+3]>>1, fptr[n*7+4]>>1, fptr[n*7+5]>>1, + isprint(fptr[n*7+0]>>1) ? fptr[n*7+0]>>1 : '.', + isprint(fptr[n*7+1]>>1) ? fptr[n*7+1]>>1 : '.', + isprint(fptr[n*7+2]>>1) ? fptr[n*7+2]>>1 : '.', + isprint(fptr[n*7+3]>>1) ? fptr[n*7+3]>>1 : '.', + isprint(fptr[n*7+4]>>1) ? fptr[n*7+4]>>1 : '.', + isprint(fptr[n*7+5]>>1) ? fptr[n*7+5]>>1 : '.', (fptr[n*7+6]&SSID_SSID_MASK)>>SSID_SSID_SHIFT, (fptr[n*7+6]&SSID_H_MASK)>>SSID_H_SHIFT, (fptr[n*7+6]&SSID_RR_MASK)>>SSID_RR_SHIFT, @@ -2459,6 +2614,42 @@ int ax25_get_pid (packet_t this_p) } + +/*------------------------------------------------------------------ + * + * Function: ax25_get_frame_len + * + * Purpose: Get length of frame. + * + * Inputs: this_p - pointer to packet object. + * + * Returns: Number of octets in the frame buffer. + * Does NOT include the extra 2 for FCS. + * + *------------------------------------------------------------------*/ + +int ax25_get_frame_len (packet_t this_p) +{ + assert (this_p->magic1 == MAGIC); + assert (this_p->magic2 == MAGIC); + + assert (this_p->frame_len >= 0 && this_p->frame_len <= AX25_MAX_PACKET_LEN); + + return (this_p->frame_len); + +} /* end ax25_get_frame_len */ + + +unsigned char *ax25_get_frame_data_ptr (packet_t this_p) +{ + assert (this_p->magic1 == MAGIC); + assert (this_p->magic2 == MAGIC); + + return (this_p->frame_data); + +} /* end ax25_get_frame_data_ptr */ + + /*------------------------------------------------------------------------------ * * Name: ax25_dedupe_crc @@ -2570,6 +2761,7 @@ unsigned short ax25_m_m_crc (packet_t pp) unsigned char fbuf[AX25_MAX_PACKET_LEN]; int flen; + // TODO: I think this can be more efficient by getting the packet content pointer instead of copying. flen = ax25_pack (pp, fbuf); crc = 0xffff; @@ -2622,7 +2814,8 @@ unsigned short ax25_m_m_crc (packet_t pp) * *------------------------------------------------------------------*/ -#define MAXSAFE 500 +//#define MAXSAFE 500 +#define MAXSAFE AX25_MAX_INFO_LEN void ax25_safe_print (char *pstr, int len, int ascii_only) { @@ -2730,7 +2923,9 @@ int ax25_alevel_to_text (alevel_t alevel, char text[AX25_ALEVEL_TO_TEXT_SIZE]) snprintf (text, AX25_ALEVEL_TO_TEXT_SIZE, "%d(%+d/%+d)", alevel.rec, alevel.mark, alevel.space); } - else if (alevel.mark == -1 && alevel.space == -1) { /* PSK - single number. */ + else if ((alevel.mark == -1 && alevel.space == -1) || /* PSK */ + (alevel.mark == -99 && alevel.space == -99)) { /* v. 1.7 "B" FM demodulator. */ + // ?? Where does -99 come from? snprintf (text, AX25_ALEVEL_TO_TEXT_SIZE, "%d", alevel.rec); } diff --git a/ax25_pad.h b/src/ax25_pad.h similarity index 96% rename from ax25_pad.h rename to src/ax25_pad.h index 6e859b19..cdb84c65 100644 --- a/ax25_pad.h +++ b/src/ax25_pad.h @@ -11,7 +11,7 @@ #define AX25_MAX_REPEATERS 8 -#define AX25_MIN_ADDRS 2 /* Destinatin & Source. */ +#define AX25_MIN_ADDRS 2 /* Destination & Source. */ #define AX25_MAX_ADDRS 10 /* Destination, Source, 8 digipeaters. */ #define AX25_DESTINATION 0 /* Address positions in frame. */ @@ -98,7 +98,7 @@ struct packet_s { * * Bits: H R R SSID 0 * - * H for digipeaters set to 0 intially. + * H for digipeaters set to 0 initially. * Changed to 1 when position has been used. * * for source & destination it is called @@ -397,6 +397,7 @@ extern int ax25_get_first_not_repeated(packet_t pp); extern int ax25_get_rr (packet_t this_p, int n); extern int ax25_get_info (packet_t pp, unsigned char **paddr); +extern void ax25_set_info (packet_t pp, unsigned char *info_ptr, int info_len); extern int ax25_cut_at_crlf (packet_t this_p); extern void ax25_set_nextp (packet_t this_p, packet_t next_p); @@ -409,6 +410,7 @@ extern void ax25_set_release_time (packet_t this_p, double release_time); extern double ax25_get_release_time (packet_t this_p); extern void ax25_set_modulo (packet_t this_p, int modulo); +extern int ax25_get_modulo (packet_t this_p); extern void ax25_format_addrs (packet_t pp, char *); extern void ax25_format_via_path (packet_t this_p, char *result, size_t result_size); @@ -427,13 +429,16 @@ extern int ax25_get_c2 (packet_t this_p); extern int ax25_get_pid (packet_t this_p); +extern int ax25_get_frame_len (packet_t this_p); +extern unsigned char *ax25_get_frame_data_ptr (packet_t this_p); + extern unsigned short ax25_dedupe_crc (packet_t pp); extern unsigned short ax25_m_m_crc (packet_t pp); extern void ax25_safe_print (char *, int, int ascii_only); -#define AX25_ALEVEL_TO_TEXT_SIZE 32 // overkill but safe. +#define AX25_ALEVEL_TO_TEXT_SIZE 40 // overkill but safe. extern int ax25_alevel_to_text (alevel_t alevel, char text[AX25_ALEVEL_TO_TEXT_SIZE]); diff --git a/ax25_pad2.c b/src/ax25_pad2.c similarity index 92% rename from ax25_pad2.c rename to src/ax25_pad2.c index 7c37dbf5..347df4b1 100644 --- a/ax25_pad2.c +++ b/src/ax25_pad2.c @@ -102,7 +102,9 @@ * * RR note: It seems that some implementations put a hint * in the "RR" reserved bits. - * http://www.tapr.org/pipermail/ax25-layer2/2005-October/000297.html + * http://www.tapr.org/pipermail/ax25-layer2/2005-October/000297.html (now broken) + * https://elixir.bootlin.com/linux/latest/source/net/ax25/ax25_addr.c#L237 + * * The RR bits can also be used for "DAMA" which is * some sort of channel access coordination scheme. * http://internet.freepage.de/cgi-bin/feets/freepage_ext/41030x030A/rewrite/hennig/afu/afudoc/afudama.html @@ -355,15 +357,19 @@ packet_t ax25_u_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_ad * * pf - Poll/Final flag. * + * pinfo - Pointer to data for Info field. Allowed only for SREJ. + * + * info_len - Length for Info field. + * * * Returns: Pointer to new packet object. * *------------------------------------------------------------------------------*/ #if AX25MEMDEBUG -packet_t ax25_s_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf, char *src_file, int src_line) +packet_t ax25_s_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf, unsigned char *pinfo, int info_len, char *src_file, int src_line) #else -packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf) +packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf, unsigned char *pinfo, int info_len) #endif { packet_t this_p; @@ -383,7 +389,7 @@ packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_ad if ( ! set_addrs (this_p, addrs, num_addr, cr)) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Internal error in %s: Could not set addresses for U frame.\n", __func__); + dw_printf ("Internal error in %s: Could not set addresses for S frame.\n", __func__); ax25_delete (this_p); return (NULL); } @@ -401,6 +407,14 @@ packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_ad nr &= (modulo - 1); } + // Erratum: The AX.25 spec is not clear about whether SREJ should be command, response, or both. + // The underlying X.25 spec clearly says it is response only. Let's go with that. + + if (ftype == frame_type_S_SREJ && cr != cr_res) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Internal error in %s: SREJ must be response.\n", __func__); + } + switch (ftype) { case frame_type_S_RR: ctrl = 0x01; break; @@ -434,6 +448,24 @@ packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_ad this_p->frame_len++; } + if (ftype == frame_type_S_SREJ) { + if (pinfo != NULL && info_len > 0) { + if (info_len > AX25_MAX_INFO_LEN) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Internal error in %s: SREJ frame, Invalid information field length %d.\n", __func__, info_len); + info_len = AX25_MAX_INFO_LEN; + } + memcpy (p, pinfo, info_len); + p += info_len; + this_p->frame_len += info_len; + } + } + else { + if (pinfo != NULL || info_len != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Internal error in %s: Info part not allowed for RR, RNR, REJ frame.\n", __func__); + } + } *p = '\0'; assert (p == this_p->frame_data + this_p->frame_len); @@ -766,7 +798,7 @@ int main () for (pf = 0; pf <= 1; pf++) { - int cmin, cmax; + int cmin = 0, cmax = 1; switch (ftype) { // 0 = response, 1 = command @@ -813,7 +845,7 @@ int main () text_color_set(DW_COLOR_INFO); dw_printf ("\nConstruct S frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); - pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf); + pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, NULL, 0); ax25_hex_dump (pp); ax25_delete (pp); @@ -827,7 +859,7 @@ int main () text_color_set(DW_COLOR_INFO); dw_printf ("\nConstruct S frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); - pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf); + pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, NULL, 0); ax25_hex_dump (pp); ax25_delete (pp); @@ -835,6 +867,26 @@ int main () } } +/* SREJ is only S frame which can have information part. */ + + static unsigned char srej_info[] = { 1<<1, 2<<1, 3<<1, 4<<1 }; + + ftype = frame_type_S_SREJ; + for (pf = 0; pf <= 1; pf++) { + + modulo = 128; + nr = 127; + cr = cr_res; + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct Multi-SREJ S frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, srej_info, (int)(sizeof(srej_info))); + + ax25_hex_dump (pp); + ax25_delete (pp); + } + dw_printf ("\n----------\n\n"); /* I frame */ diff --git a/ax25_pad2.h b/src/ax25_pad2.h similarity index 87% rename from ax25_pad2.h rename to src/ax25_pad2.h index 48609417..c6dc17a2 100644 --- a/ax25_pad2.h +++ b/src/ax25_pad2.h @@ -22,14 +22,14 @@ packet_t ax25_u_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int pf, int pid, unsigned char *pinfo, int info_len, char *src_file, int src_line); -packet_t ax25_s_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf, char *src_file, int src_line); +packet_t ax25_s_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf, unsigned char *pinfo, int info_len, char *src_file, int src_line); packet_t ax25_i_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, int modulo, int nr, int ns, int pf, int pid, unsigned char *pinfo, int info_len, char *src_file, int src_line); #define ax25_u_frame(a,n,c,f,p,q,i,l) ax25_u_frame_debug(a,n,c,f,p,q,i,l,__FILE__,__LINE__) -#define ax25_s_frame(a,n,c,f,m,r,p) ax25_s_frame_debug(a,n,c,f,m,r,p,__FILE__,__LINE__) +#define ax25_s_frame(a,n,c,f,m,r,p,i,l) ax25_s_frame_debug(a,n,c,f,m,r,p,i,l,__FILE__,__LINE__) #define ax25_i_frame(a,n,c,m,r,s,p,q,i,l) ax25_i_frame_debug(a,n,c,m,r,s,p,q,i,l,__FILE__,__LINE__) @@ -38,7 +38,7 @@ packet_t ax25_i_frame_debug (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int packet_t ax25_u_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int pf, int pid, unsigned char *pinfo, int info_len); -packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf); +packet_t ax25_s_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, ax25_frame_type_t ftype, int modulo, int nr, int pf, unsigned char *pinfo, int info_len); packet_t ax25_i_frame (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, cmdres_t cr, int modulo, int nr, int ns, int pf, int pid, unsigned char *pinfo, int info_len); diff --git a/beacon.c b/src/beacon.c similarity index 73% rename from beacon.c rename to src/beacon.c index a5a2bade..69a72701 100644 --- a/beacon.c +++ b/src/beacon.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2013, 2014, 2015, 2016, 2017 John Langner, WB2OSZ // // 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 @@ -59,26 +59,6 @@ #include "mheard.h" -#if __WIN32__ - -/* - * Windows doesn't have localtime_r. - * It should have the equivalent localtime_s, with opposite parameter - * order, but I get undefined reference when trying to use it. - */ - -struct tm *localtime_r(time_t *clock, struct tm *res) -{ - struct tm *tm; - - tm = localtime (clock); - memcpy (res, tm, sizeof(struct tm)); - return (res); -} - -#endif - - /* * Save pointers to configuration settings. */ @@ -141,6 +121,7 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo); void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct igate_config_s *pigate) { time_t now; + struct tm tm; int j; int count; #if __WIN32__ @@ -171,12 +152,20 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct * If a serious error is found, set type to BEACON_IGNORE and that * table entry should be ignored later on. */ + +// TODO: Better checking. +// We should really have a table for which keywords are are required, +// optional, or not allowed for each beacon type. Options which +// are not applicable are often silently ignored, causing confusion. + for (j=0; jnum_beacons; j++) { int chan = g_misc_config_p->beacon[j].sendto_chan; if (chan < 0) chan = 0; /* For IGate, use channel 0 call. */ + if (chan >= MAX_CHANS) chan = 0; // For ICHANNEL, use channel 0 call. - if (g_modem_config_p->achan[chan].valid) { + if (g_modem_config_p->chan_medium[chan] == MEDIUM_RADIO || + g_modem_config_p->chan_medium[chan] == MEDIUM_NETTNC) { if (strlen(g_modem_config_p->achan[chan].mycall) > 0 && strcasecmp(g_modem_config_p->achan[chan].mycall, "N0CALL") != 0 && @@ -206,6 +195,18 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct g_misc_config_p->beacon[j].btype = BEACON_IGNORE; continue; } + + /* INFO and INFOCMD are only for Custom Beacon. */ + + if (g_misc_config_p->beacon[j].custom_info != NULL || g_misc_config_p->beacon[j].custom_infocmd != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: INFO or INFOCMD are allowed only for custom beacon.\n", g_misc_config_p->beacon[j].lineno); + dw_printf ("INFO and INFOCMD allow you to specify contents of the Information field so it\n"); + dw_printf ("so it would not make sense to use these with other beacon types which construct\n"); + dw_printf ("the Information field. Perhaps you want to use COMMENT or COMMENTCMD option.\n"); + //g_misc_config_p->beacon[j].btype = BEACON_IGNORE; + continue; + } break; case BEACON_TRACKER: @@ -220,8 +221,29 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: GPS must be configured to use TBEACON.\n", g_misc_config_p->beacon[j].lineno); g_misc_config_p->beacon[j].btype = BEACON_IGNORE; +#if __WIN32__ + dw_printf ("You must specify the GPSNMEA command in your configuration file.\n"); + dw_printf ("This contains the name of the serial port where the receiver is connected.\n"); +#else + dw_printf ("You must specify the source of the GPS data in your configuration file.\n"); + dw_printf ("It can be either GPSD, meaning the gpsd daemon, or GPSNMEA for\n"); + dw_printf ("for a serial port connection with exclusive use.\n"); +#endif + } } + + /* INFO and INFOCMD are only for Custom Beacon. */ + + if (g_misc_config_p->beacon[j].custom_info != NULL || g_misc_config_p->beacon[j].custom_infocmd != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: INFO or INFOCMD are allowed only for custom beacon.\n", g_misc_config_p->beacon[j].lineno); + dw_printf ("INFO and INFOCMD allow you to specify contents of the Information field so it\n"); + dw_printf ("so it would not make sense to use these with other beacon types which construct\n"); + dw_printf ("the Information field. Perhaps you want to use COMMENT or COMMENTCMD option.\n"); + //g_misc_config_p->beacon[j].btype = BEACON_IGNORE; + continue; + } break; case BEACON_CUSTOM: @@ -270,21 +292,71 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct } /* - * Calculate first time for each beacon from the 'delay' value. + * Calculate first time for each beacon from the 'slot' or 'delay' value. */ now = time(NULL); + localtime_r (&now, &tm); for (j=0; jnum_beacons; j++) { + struct beacon_s *bp = & (g_misc_config_p->beacon[j]); #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("beacon[%d] chan=%d, delay=%d, every=%d\n", + dw_printf ("beacon[%d] chan=%d, delay=%d, slot=%d, every=%d\n", j, - g_misc_config_p->beacon[j].sendto_chan, - g_misc_config_p->beacon[j].delay, - g_misc_config_p->beacon[j].every); + bp->sendto_chan, + bp->delay, + bp->slot, + bp->every); #endif + +/* + * If timeslots, there must be a full number of beacon intervals per hour. + */ +#define IS_GOOD(x) ((3600/(x))*(x) == 3600) + + if (bp->slot != G_UNKNOWN) { + + if ( ! IS_GOOD(bp->every)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: When using timeslots, there must be a whole number of beacon intervals per hour.\n", bp->lineno); + + // Try to make it valid by adjusting up or down. + + int n; + for (n=1; ; n++) { + int e; + e = bp->every + n; + if (e > 3600) { + bp->every = 3600; + break; + } + if (IS_GOOD(e)) { + bp->every = e; + break; + } + e = bp->every - n; + if (e < 1) { + bp->every = 1; // Impose a larger minimum? + break; + } + if (IS_GOOD(e)) { + bp->every = e; + break; + } + } + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Time between slotted beacons has been adjusted to %d seconds.\n", bp->lineno, bp->every); + } +/* + * Determine when next slot time will arrive. + */ + bp->delay = bp->slot - (tm.tm_min * 60 + tm.tm_sec); + while (bp->delay > bp->every) bp->delay -= bp->every; + while (bp->delay < 5) bp->delay += bp->every; + } + g_misc_config_p->beacon[j].next = now + g_misc_config_p->beacon[j].delay; } @@ -312,7 +384,7 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct #else int e; - e = pthread_create (&beacon_tid, NULL, beacon_thread, (void *)0); + e = pthread_create (&beacon_tid, NULL, beacon_thread, NULL); if (e != 0) { text_color_set(DW_COLOR_ERROR); perror("Could not create beacon thread"); @@ -491,10 +563,12 @@ static void * beacon_thread (void *arg) */ for (j=0; jnum_beacons; j++) { - if (g_misc_config_p->beacon[j].btype == BEACON_IGNORE) + struct beacon_s *bp = & (g_misc_config_p->beacon[j]); + + if (bp->btype == BEACON_IGNORE) continue; - if (g_misc_config_p->beacon[j].next <= now) { + if (bp->next <= now) { /* Send the beacon. */ @@ -503,13 +577,20 @@ static void * beacon_thread (void *arg) /* Calculate when the next one should be sent. */ /* Easy for fixed interval. SmartBeaconing takes more effort. */ - if (g_misc_config_p->beacon[j].btype == BEACON_TRACKER) { + if (bp->btype == BEACON_TRACKER) { if (gpsinfo.fix < DWFIX_2D) { /* Fix not available so beacon was not sent. */ - /* Try again in a couple seconds. */ - g_misc_config_p->beacon[j].next = now + 2; + if (g_misc_config_p->sb_configured) { + /* Try again in a couple seconds. */ + bp->next = now + 2; + } + else { + /* Stay with the schedule. */ + /* Important for slotted. Might reconsider otherwise. */ + bp->next += bp->every; + } } else if (g_misc_config_p->sb_configured) { @@ -519,18 +600,37 @@ static void * beacon_thread (void *arg) sb_prev_time = now; sb_prev_course = gpsinfo.track; - g_misc_config_p->beacon[j].next = sb_calculate_next_time (now, + bp->next = sb_calculate_next_time (now, DW_KNOTS_TO_MPH(gpsinfo.speed_knots), gpsinfo.track, sb_prev_time, sb_prev_course); } else { - g_misc_config_p->beacon[j].next = now + g_misc_config_p->beacon[j].every; + /* Tracker beacon, fixed spacing. */ + bp->next += bp->every; } } else { - /* non-tracker beacons are at fixed spacing. */ - - g_misc_config_p->beacon[j].next = now + g_misc_config_p->beacon[j].every; + /* Non-tracker beacon, fixed spacing. */ + /* Increment by 'every' so slotted times come out right. */ + /* i.e. Don't take relative to now in case there was some delay. */ + + bp->next += bp->every; + + // https://github.com/wb2osz/direwolf/pull/301 + // https://github.com/wb2osz/direwolf/pull/301 + // This happens with a portable system with no Internet connection. + // On reboot, the time is in the past. + // After time gets set from GPS, all beacons from that interval are sent. + // FIXME: This will surely break time slotted scheduling. + // TODO: The correct fix will be using monotonic, rather than clock, time. + + /* craigerl: if next beacon is scheduled in the past, then set next beacon relative to now (happens when NTP pushes clock AHEAD) */ + /* fixme: if NTP sets clock BACK an hour, this thread will sleep for that hour */ + if ( bp->next < now ) { + bp->next = now + bp->every; + text_color_set(DW_COLOR_INFO); + dw_printf("\nSystem clock appears to have jumped forward. Beacon schedule updated.\n\n"); + } } } /* if time to send it */ @@ -682,6 +782,7 @@ static time_t sb_calculate_next_time (time_t now, static void beacon_send (int j, dwgps_info_t *gpsinfo) { + struct beacon_s *bp = & (g_misc_config_p->beacon[j]); int strict = 1; /* Strict packet checking because they will go over air. */ char stemp[20]; @@ -704,13 +805,19 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) */ strlcpy (mycall, "NOCALL", sizeof(mycall)); - assert (g_misc_config_p->beacon[j].sendto_chan >= 0); + assert (bp->sendto_chan >= 0); - strlcpy (mycall, g_modem_config_p->achan[g_misc_config_p->beacon[j].sendto_chan].mycall, sizeof(mycall)); + if (g_modem_config_p->chan_medium[bp->sendto_chan] == MEDIUM_IGATE) { // ICHANNEL uses chan 0 mycall. + // TODO: Maybe it should be allowed to have own. + strlcpy (mycall, g_modem_config_p->achan[0].mycall, sizeof(mycall)); + } + else { + strlcpy (mycall, g_modem_config_p->achan[bp->sendto_chan].mycall, sizeof(mycall)); + } if (strlen(mycall) == 0 || strcmp(mycall, "NOCALL") == 0) { text_color_set(DW_COLOR_ERROR); - dw_printf ("MYCALL not set for beacon in config file line %d.\n", g_misc_config_p->beacon[j].lineno); + dw_printf ("MYCALL not set for beacon to chan %d in config file line %d.\n", bp->sendto_chan, bp->lineno); return; } @@ -720,20 +827,25 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) * src > dest [ , via ] */ - strlcpy (beacon_text, mycall, sizeof(beacon_text)); + if (bp->source != NULL) { + strlcpy (beacon_text, bp->source, sizeof(beacon_text)); + } + else { + strlcpy (beacon_text, mycall, sizeof(beacon_text)); + } strlcat (beacon_text, ">", sizeof(beacon_text)); - if (g_misc_config_p->beacon[j].dest != NULL) { - strlcat (beacon_text, g_misc_config_p->beacon[j].dest, sizeof(beacon_text)); + if (bp->dest != NULL) { + strlcat (beacon_text, bp->dest, sizeof(beacon_text)); } else { snprintf (stemp, sizeof(stemp), "%s%1d%1d", APP_TOCALL, MAJOR_VERSION, MINOR_VERSION); strlcat (beacon_text, stemp, sizeof(beacon_text)); } - if (g_misc_config_p->beacon[j].via != NULL) { + if (bp->via != NULL) { strlcat (beacon_text, ",", sizeof(beacon_text)); - strlcat (beacon_text, g_misc_config_p->beacon[j].via, sizeof(beacon_text)); + strlcat (beacon_text, bp->via, sizeof(beacon_text)); } strlcat (beacon_text, ":", sizeof(beacon_text)); @@ -746,23 +858,23 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) // TODO: test & document. strlcpy (super_comment, "", sizeof(super_comment)); - if (g_misc_config_p->beacon[j].comment != NULL) { - strlcpy (super_comment, g_misc_config_p->beacon[j].comment, sizeof(super_comment)); + if (bp->comment != NULL) { + strlcpy (super_comment, bp->comment, sizeof(super_comment)); } - if (g_misc_config_p->beacon[j].commentcmd != NULL) { + if (bp->commentcmd != NULL) { char var_comment[AX25_MAX_INFO_LEN]; int k; /* Run given command to get variable part of comment. */ - k = dw_run_cmd (g_misc_config_p->beacon[j].commentcmd, 2, var_comment, sizeof(var_comment)); + k = dw_run_cmd (bp->commentcmd, 2, var_comment, sizeof(var_comment)); if (k > 0) { strlcat (super_comment, var_comment, sizeof(super_comment)); } else { text_color_set(DW_COLOR_ERROR); - dw_printf ("xBEACON, config file line %d, COMMENTCMD failure.\n", g_misc_config_p->beacon[j].lineno); + dw_printf ("xBEACON, config file line %d, COMMENTCMD failure.\n", bp->lineno); } } @@ -770,17 +882,17 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) /* * Add the info part depending on beacon type. */ - switch (g_misc_config_p->beacon[j].btype) { + switch (bp->btype) { case BEACON_POSITION: - encode_position (g_misc_config_p->beacon[j].messaging, g_misc_config_p->beacon[j].compress, - g_misc_config_p->beacon[j].lat, g_misc_config_p->beacon[j].lon, 0, - (int)roundf(DW_METERS_TO_FEET(g_misc_config_p->beacon[j].alt_m)), - g_misc_config_p->beacon[j].symtab, g_misc_config_p->beacon[j].symbol, - g_misc_config_p->beacon[j].power, g_misc_config_p->beacon[j].height, g_misc_config_p->beacon[j].gain, g_misc_config_p->beacon[j].dir, + encode_position (bp->messaging, bp->compress, + bp->lat, bp->lon, bp->ambiguity, + (int)roundf(DW_METERS_TO_FEET(bp->alt_m)), + bp->symtab, bp->symbol, + bp->power, bp->height, bp->gain, bp->dir, G_UNKNOWN, G_UNKNOWN, /* course, speed */ - g_misc_config_p->beacon[j].freq, g_misc_config_p->beacon[j].tone, g_misc_config_p->beacon[j].offset, + bp->freq, bp->tone, bp->offset, super_comment, info, sizeof(info)); strlcat (beacon_text, info, sizeof(beacon_text)); @@ -788,11 +900,11 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) case BEACON_OBJECT: - encode_object (g_misc_config_p->beacon[j].objname, g_misc_config_p->beacon[j].compress, 0, g_misc_config_p->beacon[j].lat, g_misc_config_p->beacon[j].lon, 0, - g_misc_config_p->beacon[j].symtab, g_misc_config_p->beacon[j].symbol, - g_misc_config_p->beacon[j].power, g_misc_config_p->beacon[j].height, g_misc_config_p->beacon[j].gain, g_misc_config_p->beacon[j].dir, + encode_object (bp->objname, bp->compress, 0, bp->lat, bp->lon, bp->ambiguity, + bp->symtab, bp->symbol, + bp->power, bp->height, bp->gain, bp->dir, G_UNKNOWN, G_UNKNOWN, /* course, speed */ - g_misc_config_p->beacon[j].freq, g_misc_config_p->beacon[j].tone, g_misc_config_p->beacon[j].offset, super_comment, + bp->freq, bp->tone, bp->offset, super_comment, info, sizeof(info)); strlcat (beacon_text, info, sizeof(beacon_text)); break; @@ -809,7 +921,7 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) /* transmission of altitude from GPS. */ my_alt_ft = G_UNKNOWN; - if (gpsinfo->fix >= 3 && gpsinfo->altitude != G_UNKNOWN && g_misc_config_p->beacon[j].alt_m > 0) { + if (gpsinfo->fix >= 3 && gpsinfo->altitude != G_UNKNOWN && bp->alt_m > 0) { my_alt_ft = (int)roundf(DW_METERS_TO_FEET(gpsinfo->altitude)); } @@ -818,12 +930,12 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) coarse = (int)roundf(gpsinfo->track); } - encode_position (g_misc_config_p->beacon[j].messaging, g_misc_config_p->beacon[j].compress, - gpsinfo->dlat, gpsinfo->dlon, 0, my_alt_ft, - g_misc_config_p->beacon[j].symtab, g_misc_config_p->beacon[j].symbol, - g_misc_config_p->beacon[j].power, g_misc_config_p->beacon[j].height, g_misc_config_p->beacon[j].gain, g_misc_config_p->beacon[j].dir, + encode_position (bp->messaging, bp->compress, + gpsinfo->dlat, gpsinfo->dlon, bp->ambiguity, my_alt_ft, + bp->symtab, bp->symbol, + bp->power, bp->height, bp->gain, bp->dir, coarse, (int)roundf(gpsinfo->speed_knots), - g_misc_config_p->beacon[j].freq, g_misc_config_p->beacon[j].tone, g_misc_config_p->beacon[j].offset, + bp->freq, bp->tone, bp->offset, super_comment, info, sizeof(info)); strlcat (beacon_text, info, sizeof(beacon_text)); @@ -845,8 +957,8 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) A.g_dcs = G_UNKNOWN; strlcpy (A.g_src, mycall, sizeof(A.g_src)); - A.g_symbol_table = g_misc_config_p->beacon[j].symtab; - A.g_symbol_code = g_misc_config_p->beacon[j].symbol; + A.g_symbol_table = bp->symtab; + A.g_symbol_code = bp->symbol; A.g_lat = gpsinfo->dlat; A.g_lon = gpsinfo->dlon; A.g_speed_mph = DW_KNOTS_TO_MPH(gpsinfo->speed_knots); @@ -865,25 +977,25 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) case BEACON_CUSTOM: - if (g_misc_config_p->beacon[j].custom_info != NULL) { + if (bp->custom_info != NULL) { /* Fixed handcrafted text. */ - strlcat (beacon_text, g_misc_config_p->beacon[j].custom_info, sizeof(beacon_text)); + strlcat (beacon_text, bp->custom_info, sizeof(beacon_text)); } - else if (g_misc_config_p->beacon[j].custom_infocmd != NULL) { + else if (bp->custom_infocmd != NULL) { char info_part[AX25_MAX_INFO_LEN]; int k; /* Run given command to obtain the info part for packet. */ - k = dw_run_cmd (g_misc_config_p->beacon[j].custom_infocmd, 2, info_part, sizeof(info_part)); + k = dw_run_cmd (bp->custom_infocmd, 2, info_part, sizeof(info_part)); if (k > 0) { strlcat (beacon_text, info_part, sizeof(beacon_text)); } else { text_color_set(DW_COLOR_ERROR); - dw_printf ("CBEACON, config file line %d, INFOCMD failure.\n", g_misc_config_p->beacon[j].lineno); + dw_printf ("CBEACON, config file line %d, INFOCMD failure.\n", bp->lineno); strlcpy (beacon_text, "", sizeof(beacon_text)); // abort! } } @@ -935,21 +1047,21 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) alevel_t alevel; - switch (g_misc_config_p->beacon[j].sendto_type) { + switch (bp->sendto_type) { case SENDTO_IGATE: text_color_set(DW_COLOR_XMIT); dw_printf ("[ig] %s\n", beacon_text); - igate_send_rec_packet (0, pp); + igate_send_rec_packet (-1, pp); // Channel -1 to avoid RF>IS filtering. ax25_delete (pp); break; case SENDTO_XMIT: default: - tq_append (g_misc_config_p->beacon[j].sendto_chan, TQ_PRIO_1_LO, pp); + tq_append (bp->sendto_chan, TQ_PRIO_1_LO, pp); break; case SENDTO_RECV: @@ -957,13 +1069,13 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo) /* Simulated reception from radio. */ memset (&alevel, 0xff, sizeof(alevel)); - dlq_rec_frame (g_misc_config_p->beacon[j].sendto_chan, 0, 0, pp, alevel, 0, ""); + dlq_rec_frame (bp->sendto_chan, 0, 0, pp, alevel, 0, 0, ""); break; } } else { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file: Failed to parse packet constructed from line %d.\n", g_misc_config_p->beacon[j].lineno); + dw_printf ("Config file: Failed to parse packet constructed from line %d.\n", bp->lineno); dw_printf ("%s\n", beacon_text); } diff --git a/beacon.h b/src/beacon.h similarity index 100% rename from beacon.h rename to src/beacon.h diff --git a/cdigipeater.c b/src/cdigipeater.c similarity index 74% rename from cdigipeater.c rename to src/cdigipeater.c index e531efd7..06128b20 100644 --- a/cdigipeater.c +++ b/src/cdigipeater.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2016 John Langner, WB2OSZ +// Copyright (C) 2016, 2017 John Langner, WB2OSZ // // 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 @@ -34,7 +34,7 @@ * are significantly different and I thought it would be * too confusing to munge them together. * - * References: The Ax.25 protcol barely mentions digipeaters and + * References: The Ax.25 protocol barely mentions digipeaters and * and doesn't describe how they should work. * *------------------------------------------------------------------*/ @@ -49,7 +49,7 @@ #include #include /* for isdigit, isupper */ #include "regex.h" -#include +#include #include "ax25_pad.h" #include "cdigipeater.h" @@ -59,7 +59,7 @@ static packet_t cdigipeat_match (int from_chan, packet_t pp, char *mycall_rec, char *mycall_xmit, - regex_t *alias, int to_chan, char *filter_str); + int has_alias, regex_t *alias, int to_chan, char *cfilter_str); /* @@ -129,8 +129,10 @@ void cdigipeater (int from_chan, packet_t pp) { int to_chan; + // Connected mode is allowed only for channels with internal modem. + // It probably wouldn't matter for digipeating but let's keep that rule simple and consistent. - if ( from_chan < 0 || from_chan >= MAX_CHANS || ( ! save_audio_config_p->achan[from_chan].valid) ) { + if ( from_chan < 0 || from_chan >= MAX_CHANS || save_audio_config_p->chan_medium[from_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("cdigipeater: Did not expect to receive on invalid channel %d.\n", from_chan); return; @@ -149,9 +151,10 @@ void cdigipeater (int from_chan, packet_t pp) packet_t result; result = cdigipeat_match (from_chan, pp, save_audio_config_p->achan[from_chan].mycall, - save_audio_config_p->achan[to_chan].mycall, - &save_cdigi_config_p->alias[from_chan][to_chan], to_chan, - save_cdigi_config_p->filter_str[from_chan][to_chan]); + save_audio_config_p->achan[to_chan].mycall, + save_cdigi_config_p->has_alias[from_chan][to_chan], + &(save_cdigi_config_p->alias[from_chan][to_chan]), to_chan, + save_cdigi_config_p->cfilter_str[from_chan][to_chan]); if (result != NULL) { tq_append (to_chan, TQ_PRIO_0_HI, result); cdigi_count[from_chan][to_chan]++; @@ -171,9 +174,10 @@ void cdigipeater (int from_chan, packet_t pp) packet_t result; result = cdigipeat_match (from_chan, pp, save_audio_config_p->achan[from_chan].mycall, - save_audio_config_p->achan[to_chan].mycall, - &save_cdigi_config_p->alias[from_chan][to_chan], to_chan, - save_cdigi_config_p->filter_str[from_chan][to_chan]); + save_audio_config_p->achan[to_chan].mycall, + save_cdigi_config_p->has_alias[from_chan][to_chan], + &(save_cdigi_config_p->alias[from_chan][to_chan]), to_chan, + save_cdigi_config_p->cfilter_str[from_chan][to_chan]); if (result != NULL) { tq_append (to_chan, TQ_PRIO_0_HI, result); cdigi_count[from_chan][to_chan]++; @@ -203,12 +207,15 @@ void cdigipeater (int from_chan, packet_t pp) * packet is to be transmitted. Could be the same as * mycall_rec or different. * - * alias - Compiled pattern for my station aliases. - * Could be NULL if no aliases. + * has_alias - True if we have an alias. + * + * alias - Optional compiled pattern for my station aliases. + * Do NOT attempt to use this if 'has_alias' is false. * * to_chan - Channel number that we are transmitting to. * - * filter_str - Filter expression string or NULL. + * cfilter_str - Filter expression string for the from/to channel pair or NULL. + * Note that only a subset of the APRS filters are applicable here. * * Returns: Packet object for transmission or NULL. * The original packet is not modified. The caller is responsible for freeing it. @@ -227,20 +234,38 @@ void cdigipeater (int from_chan, packet_t pp) static packet_t cdigipeat_match (int from_chan, packet_t pp, char *mycall_rec, char *mycall_xmit, - regex_t *alias, int to_chan, char *filter_str) + int has_alias, regex_t *alias, int to_chan, char *cfilter_str) { int r; char repeater[AX25_MAX_ADDR_LEN]; int err; char err_msg[100]; +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("cdigipeat_match (from_chan=%d, pp=%p, mycall_rec=%s, mycall_xmit=%s, has_alias=%d, alias=%p, to_chan=%d, cfilter_str=%s\n", + from_chan, pp, mycall_rec, mycall_xmit, has_alias, alias, to_chan, cfilter_str); +#endif + /* * First check if filtering has been configured. + * Note that we have three different config file filter commands: + * + * FILTER - APRS digipeating and IGate client side. + * Originally this was the only one. + * Should we change it to AFILTER to make it clearer? + * CFILTER - Similar for connected moded digipeater. + * IGFILTER - APRS-IS (IGate) server side - completely different. + * Confusing with similar name but much different idea. + * Maybe this should be renamed to SUBSCRIBE or something like that. + * + * Logically this should come later, after an address/alias match. + * But here we only have to do it once. */ - if (filter_str != NULL) { + if (cfilter_str != NULL) { - if (pfilter(from_chan, to_chan, filter_str, pp, 0) != 1) { + if (pfilter(from_chan, to_chan, cfilter_str, pp, 0) != 1) { return(NULL); } } @@ -286,7 +311,11 @@ static packet_t cdigipeat_match (int from_chan, packet_t pp, char *mycall_rec, c /* * If we have an alias match, substitute MYCALL. */ - if (alias != NULL) { + if (has_alias) { +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Checking %s for alias match.\n", repeater); +#endif err = regexec(alias,repeater,0,NULL,0); if (err == 0) { packet_t result; @@ -304,6 +333,12 @@ static packet_t cdigipeat_match (int from_chan, packet_t pp, char *mycall_rec, c dw_printf ("%s\n", err_msg); } } + else { +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("No alias was specified.\n"); +#endif + } /* * Don't repeat it if we get here. diff --git a/cdigipeater.h b/src/cdigipeater.h similarity index 71% rename from cdigipeater.h rename to src/cdigipeater.h index 5a50b872..69a4b8c7 100644 --- a/cdigipeater.h +++ b/src/cdigipeater.h @@ -23,15 +23,18 @@ struct cdigi_config_s { /* * Rules for each of the [from_chan][to_chan] combinations. */ - + int enabled[MAX_CHANS][MAX_CHANS]; // Is it enabled for from/to pair? + + int has_alias[MAX_CHANS][MAX_CHANS]; // If there was no alias in the config file, + // the structure below will not be set up + // properly and an attempt to use it could + // result in a crash. (fixed v1.5) + // Not needed for [APRS] DIGIPEAT because + // the alias is mandatory there. regex_t alias[MAX_CHANS][MAX_CHANS]; - int enabled[MAX_CHANS][MAX_CHANS]; - - char *filter_str[MAX_CHANS+1][MAX_CHANS+1]; + char *cfilter_str[MAX_CHANS][MAX_CHANS]; // NULL or optional Packet Filter strings such as "t/m". - // Notice the size of arrays is one larger than normal. - // That extra position is for the IGate. }; /* diff --git a/src/cm108.c b/src/cm108.c new file mode 100644 index 00000000..ff3ff792 --- /dev/null +++ b/src/cm108.c @@ -0,0 +1,1080 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2017,2019,2021 John Langner, WB2OSZ +// +// Parts of this were adapted from "hamlib" which contains the notice: +// +// * Copyright (c) 2000-2012 by Stephane Fillod +// * Copyright (c) 2011 by Andrew Errington +// * CM108 detection code Copyright (c) Thomas Sailer used with permission +// +// 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, see . + +/*------------------------------------------------------------------ + * + * Module: cm108.c + * + * Purpose: Use the CM108/CM119 (or compatible) GPIO pins for the Push To Talk (PTT) Control. + * + * Description: + * + * There is an increasing demand for using the GPIO pins of USB audio devices for PTT. + * We have a few commercial products: + * + * DINAH https://hamprojects.info/dinah/ + * PAUL https://hamprojects.info/paul/ + * DMK URI http://www.dmkeng.com/URI_Order_Page.htm + * RB-USB RIM http://www.repeater-builder.com/products/usb-rim-lite.html + * RA-35 http://www.masterscommunications.com/products/radio-adapter/ra35.html + * + * and homebrew projects which are all very similar. + * + * http://www.qsl.net/kb9mwr/projects/voip/usbfob-119.pdf + * http://rtpdir.weebly.com/uploads/1/6/8/7/1687703/usbfob.pdf + * http://www.repeater-builder.com/projects/fob/USB-Fob-Construction.pdf + * https://irongarment.wordpress.com/2011/03/29/cm108-compatible-chips-with-gpio/ + * + * Homebrew plans all use GPIO 3 because it is easier to tack solder a wire to a pin on the end. + * All of the products, that I have seen, also use the same pin so this is the default. + * + * Soundmodem and hamlib paved the way but didn't get too far. + * Dire Wolf 1.3 added HAMLIB support (Linux only) which theoretically allows this in a + * painful roundabout way. This is documented in the User Guide, section called, + * "Hamlib PTT Example 2: Use GPIO of USB audio adapter. (e.g. DMK URI)" + * + * It's rather involved and the explanation doesn't cover the case of multiple + * USB-Audio adapters. It is not as straightforward as you might expect. Here we have + * an example of 3 C-Media USB adapters, a SignaLink USB, a keyboard, and a mouse. + * + * + * VID PID Product Sound ADEVICE HID [ptt] + * --- --- ------- ----- ------- --------- + * ** 0d8c 000c C-Media USB Headphone Set /dev/snd/pcmC1D0c plughw:1,0 /dev/hidraw0 + * ** 0d8c 000c C-Media USB Headphone Set /dev/snd/pcmC1D0p plughw:1,0 /dev/hidraw0 + * ** 0d8c 000c C-Media USB Headphone Set /dev/snd/controlC1 /dev/hidraw0 + * 08bb 2904 USB Audio CODEC /dev/snd/pcmC2D0c plughw:2,0 /dev/hidraw2 + * 08bb 2904 USB Audio CODEC /dev/snd/pcmC2D0p plughw:2,0 /dev/hidraw2 + * 08bb 2904 USB Audio CODEC /dev/snd/controlC2 /dev/hidraw2 + * ** 0d8c 000c C-Media USB Headphone Set /dev/snd/pcmC0D0c plughw:0,0 /dev/hidraw1 + * ** 0d8c 000c C-Media USB Headphone Set /dev/snd/pcmC0D0p plughw:0,0 /dev/hidraw1 + * ** 0d8c 000c C-Media USB Headphone Set /dev/snd/controlC0 /dev/hidraw1 + * ** 0d8c 0008 C-Media USB Audio Device /dev/snd/pcmC4D0c plughw:4,0 /dev/hidraw6 + * ** 0d8c 0008 C-Media USB Audio Device /dev/snd/pcmC4D0p plughw:4,0 /dev/hidraw6 + * ** 0d8c 0008 C-Media USB Audio Device /dev/snd/controlC4 /dev/hidraw6 + * 413c 2010 Dell USB Keyboard /dev/hidraw4 + * 0461 4d15 USB Optical Mouse /dev/hidraw5 + * + * + * The USB soundcards (/dev/snd/pcm...) have an associated Human Interface Device (HID) + * corresponding to the GPIO pins which are sometimes connected to pushbuttons. + * The mapping has no obvious pattern. + * + * Sound Card 0 HID 1 + * Sound Card 1 HID 0 + * Sound Card 2 HID 2 + * Sound Card 4 HID 6 + * + * That would be a real challenge if you had to figure that all out and configure manually. + * Dire Wolf version 1.5 makes this much more flexible and easier to use by supporting multiple + * sound devices and automatically determining the corresponding HID for the PTT signal. + * + * In version 1.7, we add a half-backed solution for Windows. It's fine for situations + * with a single USB Audio Adapter, but does not automatically handle the multiple device case. + * Manual configuration needs to be used in this case. + * + * Here is something new and interesting. The All in One cable (AIOC). + * https://github.com/skuep/AIOC/tree/master + * + * A microcontroller is used to emulate a CM108-compatible soundcard + * and a serial port. It fits right on the side of a Bao Feng or similar. + * + *---------------------------------------------------------------*/ + +#include "direwolf.h" + +#ifndef USE_CM108 + +#ifdef CM108_MAIN + + +#include "textcolor.h" + +int main (void) +{ + text_color_init (0); // Turn off text color. +#if defined(__OpenBSD__) || defined(__FreeBSD__) + dw_printf ("CM108 PTT support is not available for this operating system.\n"); +#else + dw_printf ("CM108 PTT support was excluded because /usr/include/libudev.h was missing.\n"); + dw_printf ("Install it with \"sudo apt-get install libudev-dev\" or\n"); + dw_printf ("\"sudo yum install libudev-devel\" then rebuild.\n"); +#endif + return (0); +} + +#endif + +#else // USE_CM108 is defined + +#include +#include +#include +#include +#include +#include + +#if __WIN32__ +#include +#include "hidapi.h" +#else +#include +#include +#include +#include // ioctl, _IOR +#include +#include +#include // for HIDIOCGRAWINFO +#endif + +#include "textcolor.h" +#include "cm108.h" + +static int cm108_write (char *name, int iomask, int iodata); + + +// The CM108, CM109, and CM119 datasheets all say that idProduct can be in the range +// of 0008 to 000f programmable by MSEL and MODE pin. How can we tell the difference? + +// CM108B is 0012. +// CM119B is 0013. +// CM108AH is 0139 programmable by MSEL and MODE pin. +// CM119A is 013A programmable by MSEL and MODE pin. + +// To make matters even more confusing, these can be overridden +// with an external EEPROM. Some have 8, rather than 4 GPIO. + +#define CMEDIA_VID 0xd8c // Vendor ID +#define CMEDIA_PID1_MIN 0x0008 // range for CM108, CM109, CM119 (no following letters) +#define CMEDIA_PID1_MAX 0x000f + +#define CMEDIA_PID_CM108AH 0x0139 // CM108AH +#define CMEDIA_PID_CM108AH_alt 0x013c // CM108AH? - see issue 210 +#define CMEDIA_PID_CM108B 0x0012 // CM108B +#define CMEDIA_PID_CM119A 0x013a // CM119A +#define CMEDIA_PID_CM119B 0x0013 // CM119B +#define CMEDIA_PID_HS100 0x013c // HS100 + +// The SSS chips seem to be pretty much compatible but they have only two GPIO. +// https://irongarment.wordpress.com/2011/03/29/cm108-compatible-chips-with-gpio/ +// Data sheet says VID/PID is from an EEPROM but mentions no default. + +#define SSS_VID 0x0c76 // SSS1621, SSS1623 +#define SSS_PID1 0x1605 +#define SSS_PID2 0x1607 +#define SSS_PID3 0x160b + +// https://github.com/skuep/AIOC/blob/master/stm32/aioc-fw/Src/usb_descriptors.h + +#define AIOC_VID 0x1209 +#define AIOC_PID 0x7388 + + +// Device VID PID Number of GPIO +// ------ --- --- -------------- +// CM108 0d8c 0008-000f * 4 +// CM108AH 0d8c 0139 * 3 Has GPIO 1,3,4 but not 2 +// CM108B 0d8c 0012 3 Has GPIO 1,3,4 but not 2 +// CM109 0d8c 0008-000f * 8 +// CM119 0d8c 0008-000f * 8 +// CM119A 0d8c 013a * 8 +// CM119B 0d8c 0013 8 +// HS100 0d8c 013c 0 (issue 210 reported 013c +// being seen for CM108AH) +// +// SSS1621 0c76 1605 2 per ZL3AME, Can't find data sheet +// SSS1623 0c76 1607,160b 2 per ZL3AME, Not in data sheet. +// +// * idProduct programmable by MSEL and MODE pin. +// + +// CMedia pin GPIO Notes +// ---------- ---- ----- +// 43 1 +// 11 2 N.C. for CM108AH, CM108B +// 13 3 Most popular for PTT because it is on the end. +// 15 4 +// 16 5 CM109, CM119, CM119A, CM119B only +// 17 6 " +// 20 7 " +// 22 8 " + +// Test for supported devices. + +#define GOOD_DEVICE(v,p) ( (v == CMEDIA_VID && ((p >= CMEDIA_PID1_MIN && p <= CMEDIA_PID1_MAX) \ + || p == CMEDIA_PID_CM108AH \ + || p == CMEDIA_PID_CM108AH_alt \ + || p == CMEDIA_PID_CM108B \ + || p == CMEDIA_PID_CM119A \ + || p == CMEDIA_PID_CM119B )) \ + || \ + (v == SSS_VID && (p == SSS_PID1 || p == SSS_PID2 || p == SSS_PID3)) \ + || \ + (v == AIOC_VID && p == AIOC_PID) ) + +// Look out for null source pointer, and avoid buffer overflow on destination. + +#define SAFE_STRCPY(to,from) { if (from != NULL) { strncpy(to,from,sizeof(to)); to[sizeof(to)-1] = '\0'; } } + + +// Used to process regular expression matching results. + +#ifndef __WIN32__ + +static void substr_se (char *dest, const char *src, int start, int endp1) +{ + int len = endp1 - start; + + if (start < 0 || endp1 < 0 || len <= 0) { + dest[0] = '\0'; + return; + } + memcpy (dest, src + start, len); + dest[len] = '\0'; + +} /* end substr_se */ + +#endif + +// Maximum length of name for PTT HID. +// For Linux, this was originally 17 to handle names like /dev/hidraw3. +// Windows has more complicated names. The longest I saw was 95 but longer have been reported. + +#define MAXX_HIDRAW_NAME_LEN 128 + +/* + * Result of taking inventory of USB soundcards and USB HIDs. + */ + +struct thing_s { + int vid; // vendor id, displayed as four hexadecimal digits. + int pid; // product id, displayed as four hexadecimal digits. + char card_number[8]; // "Card" Number. e.g. 2 for plughw:2,0 + char card_name[32]; // Audio Card Name, assigned by system (e.g. Device_1) or by udev rule. + char product[32]; // product name (e.g. manufacturer, model) + char devnode_sound[22]; // e.g. /dev/snd/pcmC0D0p + char plughw[72]; // Above in more familiar format e.g. plughw:0,0 + // Oversized to silence a compiler warning. + char plughw2[72]; // With name rather than number. + char devpath[128]; // Kernel dev path. Does not include /sys mount point. + char devnode_hidraw[MAXX_HIDRAW_NAME_LEN]; + // e.g. /dev/hidraw3 - for Linux - was length 17 + // The Windows path for a HID looks like this, lengths up to 95 seen. + // \\?\hid#vid_0d8c&pid_000c&mi_03#8&164d11c9&0&0000#{4d1e55b2-f16f-11cf-88cb-001111000030} + char devnode_usb[25]; // e.g. /dev/bus/usb/001/012 + // This is what we use to match up audio and HID. +}; + +int cm108_inventory (struct thing_s *things, int max_things); + + +/*------------------------------------------------------------------- + * + * Name: main + * + * Purpose: Useful utility to list USB audio and HID devices. + * + * Optional command line arguments: + * + * HID path + * GPIO number (default 3) + * + * When specified the pin will be set high and low until interrupted. + * + *------------------------------------------------------------------*/ + +//#define EXTRA 1 + +#define MAXX_THINGS 60 + +#ifdef CM108_MAIN + +static void usage(void) +{ + text_color_set(DW_COLOR_ERROR); + dw_printf ("\n"); + dw_printf ("Usage: cm108 [ device-path [ gpio-num ] ]\n"); + dw_printf ("\n"); + dw_printf ("With no command line arguments, this will produce a list of\n"); +#if __WIN32__ + dw_printf ("Human Interface Devices (HID) and indicate which ones can be\n"); + dw_printf ("used for GPIO PTT.\n"); +#else + dw_printf ("Audio devices and Human Interface Devices (HID) and indicate\n"); + dw_printf ("which ones can be used for GPIO PTT.\n"); +#endif + dw_printf ("\n"); + dw_printf ("Specify the HID device path to test the PTT function.\n"); + dw_printf ("Its state should change once per second.\n"); +#if __WIN32__ + dw_printf ("You might need to quote the path depending on the command processor.\n"); +#endif + dw_printf ("GPIO 3 is the default. A different number can be optionally specified.\n"); + exit (EXIT_FAILURE); +} + +int main (int argc, char **argv) +{ + struct thing_s things[MAXX_THINGS]; + int num_things; + int i; + + text_color_init (0); // Turn off text color. + text_color_set(DW_COLOR_INFO); + + if (argc >=2) { + char path[128]; + strlcpy(path, argv[1], sizeof(path)); + int gpio = 3; + if (argc >= 3) { + gpio = atoi(argv[2]); + } + if (gpio < 1 || gpio > 8) { + dw_printf ("GPIO number must be in range of 1 - 8.\n"); + usage(); + exit (EXIT_FAILURE); + } + int state = 0; + while (1) { + dw_printf ("%d", state); + fflush (stdout); + int err = cm108_set_gpio_pin (path, gpio, state); + if (err != 0) { + dw_printf ("\nWRITE ERROR for USB Audio Adapter GPIO!\n"); + usage(); + exit (EXIT_FAILURE); + } + SLEEP_SEC(1); + state = ! state; + } + } + + +// Take inventory of USB Audio adapters and other HID devices. + + num_things = cm108_inventory (things, MAXX_THINGS); + +#if __WIN32__ + +///////////////////////////////////////////////////// +// Windows - Remove the sound related columns for now. +///////////////////////////////////////////////////// + + dw_printf (" VID PID %-*s %-*s" + "\n", (int)sizeof(things[0].product), "Product", + 17, "HID [ptt]" + ); + + dw_printf (" --- --- %-*s %-*s" + + "\n", (int)sizeof(things[0].product), "-------", + 17, "---------" + ); + for (i = 0; i < num_things; i++) { + + dw_printf ("%2s %04x %04x %-*s %s" + + "\n", + GOOD_DEVICE(things[i].vid,things[i].pid) ? "**" : " ", + things[i].vid, things[i].pid, + (int)sizeof(things[i].product), things[i].product, + things[i].devnode_hidraw + ); + } + dw_printf ("\n"); + dw_printf ("** = Can use Audio Adapter GPIO for PTT.\n"); + dw_printf ("\n"); + + // T.B.D. - additional text ??? + +#else + +///////////////////////////////////////////// +// Linux +///////////////////////////////////////////// + + + dw_printf (" VID PID %-*s %-*s %-*s %-*s %-*s" +#if EXTRA + " %-*s" +#endif + "\n", (int)sizeof(things[0].product), "Product", + (int)sizeof(things[0].devnode_sound), "Sound", + (int)sizeof(things[0].plughw)/5, "ADEVICE", + (int)sizeof(things[0].plughw2)/4, "ADEVICE", + 17, "HID [ptt]" +#if EXTRA + , (int)sizeof(things[0].devnode_usb), "USB" +#endif + ); + + dw_printf (" --- --- %-*s %-*s %-*s %-*s %-*s" +#if EXTRA + " %-*s" +#endif + "\n", (int)sizeof(things[0].product), "-------", + (int)sizeof(things[0].devnode_sound), "-----", + (int)sizeof(things[0].plughw)/5, "-------", + (int)sizeof(things[0].plughw2)/4, "-------", + 17, "---------" +#if EXTRA + , (int)sizeof(things[0].devnode_usb), "---" +#endif + ); + for (i = 0; i < num_things; i++) { + + dw_printf ("%2s %04x %04x %-*s %-*s %-*s %-*s %s" +#if EXTRA + " %-*s" +#endif + "\n", + GOOD_DEVICE(things[i].vid,things[i].pid) ? "**" : " ", + things[i].vid, things[i].pid, + (int)sizeof(things[i].product), things[i].product, + (int)sizeof(things[i].devnode_sound), things[i].devnode_sound, + (int)sizeof(things[0].plughw)/5, things[i].plughw, + (int)sizeof(things[0].plughw2)/4, things[i].plughw2, + things[i].devnode_hidraw +#if EXTRA + , (int)sizeof(things[i].devnode_usb), things[i].devnode_usb +#endif + ); + //dw_printf (" %-*s\n", (int)sizeof(things[i].devpath), things[i].devpath); + } + dw_printf ("\n"); + dw_printf ("** = Can use Audio Adapter GPIO for PTT.\n"); + dw_printf ("\n"); + + static const char *suggested_names[] = {"Fred", "Wilma", "Pebbles", "Dino", "Barney", "Betty", "Bamm_Bamm", "Chip", "Roxy" }; + int iname = 0; + + // From example in https://alsa.opensrc.org/Udev + + dw_printf ("Notice that each USB Audio adapter is assigned a number and a name. These are not predictable so you could\n"); + dw_printf ("end up using the wrong adapter after adding or removing other USB devices or after rebooting. You can assign a\n"); + dw_printf ("name to each USB adapter so you can refer to the same one each time. This can be based on any characteristics\n"); + dw_printf ("that makes them unique such as product id or serial number. Unfortunately these devices don't have unique serial\n"); + dw_printf ("numbers so how can we tell them apart? A name can also be assigned based on the physical USB socket.\n"); + dw_printf ("Create a file like \"/etc/udev/rules.d/85-my-usb-audio.rules\" with the following contents and then reboot.\n"); + dw_printf ("\n"); + dw_printf ("SUBSYSTEM!=\"sound\", GOTO=\"my_usb_audio_end\"\n"); + dw_printf ("ACTION!=\"add\", GOTO=\"my_usb_audio_end\"\n"); + +// Consider only the 'devnode' paths that end with "card" and a number. +// Replace the number with a question mark. + + regex_t devpath_re; + char emsg[100]; + // Drop any "/sys" at the beginning. + int e = regcomp (&devpath_re, "(/devices/.+/card)[0-9]$", REG_EXTENDED); + if (e) { + regerror (e, &devpath_re, emsg, sizeof(emsg)); + text_color_set(DW_COLOR_ERROR); + dw_printf("INTERNAL ERROR: %s:%d: %s\n", __FILE__, __LINE__, emsg); + return (-1); + } + + for (i = 0; i < num_things; i++) { + if (i == 0 || strcmp(things[i].devpath,things[i-1].devpath) != 0) { + regmatch_t devpath_match[2]; + if (regexec (&devpath_re, things[i].devpath, 2, devpath_match, 0) == 0) { + char without_number[256]; + substr_se (without_number, things[i].devpath, devpath_match[1].rm_so, devpath_match[1].rm_eo); + dw_printf ("DEVPATH==\"%s?\", ATTR{id}=\"%s\"\n", without_number, suggested_names[iname]); + if (iname < 6) iname++; + } + } + } + dw_printf ("LABEL=\"my_usb_audio_end\"\n"); + dw_printf ("\n"); +#endif + return (0); +} + +#endif // CM108_MAIN + + + +/*------------------------------------------------------------------- + * + * Name: cm108_inventory + * + * Purpose: Take inventory of USB audio and HID. + * + * Inputs: max_things - Maximum number of items to collect. + * + * Outputs: things - Array of items collected. + * Corresponding sound device and HID are merged into one item. + * + * Returns: Number of items placed in things array. + * Should be in the range of 0 thru max_things. + * -1 for a bad unexpected error. + * + *------------------------------------------------------------------*/ + + +int cm108_inventory (struct thing_s *things, int max_things) +{ + int num_things = 0; + memset (things, 0, sizeof(struct thing_s) * max_things); + +#if __WIN32__ + + struct hid_device_info *devs, *cur_dev; + + if (hid_init()) { + text_color_set(DW_COLOR_ERROR); + dw_printf("cm108_inventory: hid_init() failed.\n"); + return (-1); + } + + devs = hid_enumerate(0x0, 0x0); + cur_dev = devs; + while (cur_dev) { +#if 0 + printf("Device Found\n type: %04hx %04hx\n path: %s\n serial_number: %ls", cur_dev->vendor_id, cur_dev->product_id, cur_dev->path, cur_dev->serial_number); + printf("\n"); + printf(" Manufacturer: %ls\n", cur_dev->manufacturer_string); + printf(" Product: %ls\n", cur_dev->product_string); + printf(" Release: %hx\n", cur_dev->release_number); + printf(" Interface: %d\n", cur_dev->interface_number); + printf(" Usage (page): 0x%hx (0x%hx)\n", cur_dev->usage, cur_dev->usage_page); + printf("\n"); +#endif + if (num_things < max_things && cur_dev->vendor_id != 0x051d) { // FIXME - remove exception + things[num_things].vid = cur_dev->vendor_id; + things[num_things].pid = cur_dev->product_id; + wcstombs (things[num_things].product, cur_dev->product_string, sizeof(things[num_things].product)); + things[num_things].product[sizeof(things[num_things].product) - 1] = '\0'; + strlcpy (things[num_things].devnode_hidraw, cur_dev->path, sizeof(things[num_things].devnode_hidraw)); + + num_things++; + } + cur_dev = cur_dev->next; + } + hid_free_enumeration(devs); + +#else // Linux, with udev + + struct udev *udev; + struct udev_enumerate *enumerate; + struct udev_list_entry *devices, *dev_list_entry; + struct udev_device *dev; + struct udev_device *parentdev; + + char const *pattrs_id = NULL; + char const *pattrs_number = NULL; + char card_devpath[128] = ""; + + +/* + * First get a list of the USB audio devices. + * This is based on the example in http://www.signal11.us/oss/udev/ + */ + udev = udev_new(); + if (!udev) { + text_color_set(DW_COLOR_ERROR); + dw_printf("INTERNAL ERROR: Can't create udev.\n"); + return (-1); + } + + enumerate = udev_enumerate_new(udev); + udev_enumerate_add_match_subsystem(enumerate, "sound"); + udev_enumerate_scan_devices(enumerate); + devices = udev_enumerate_get_list_entry(enumerate); + udev_list_entry_foreach(dev_list_entry, devices) { + const char *path = udev_list_entry_get_name(dev_list_entry); + dev = udev_device_new_from_syspath(udev, path); + char const *devnode = udev_device_get_devnode(dev); + + if (devnode == NULL ) { + // I'm not happy with this but couldn't figure out how + // to get attributes from one level up from the pcmC?D?? node. + strlcpy (card_devpath, path, sizeof(card_devpath)); + pattrs_id = udev_device_get_sysattr_value(dev,"id"); + pattrs_number = udev_device_get_sysattr_value(dev,"number"); + //dw_printf (" >card_devpath = %s\n", card_devpath); + //dw_printf (" >>pattrs_id = %s\n", pattrs_id); + //dw_printf (" >>pattrs_number = %s\n", pattrs_number); + } + else { + parentdev = udev_device_get_parent_with_subsystem_devtype( dev, "usb", "usb_device"); + if (parentdev != NULL) { + char const *p; + int vid = 0; + int pid = 0; + + p = udev_device_get_sysattr_value(parentdev,"idVendor"); + if (p != NULL) vid = strtol(p, NULL, 16); + p = udev_device_get_sysattr_value(parentdev,"idProduct"); + if (p != NULL) pid = strtol(p, NULL, 16); + + if (num_things < max_things) { + things[num_things].vid = vid; + things[num_things].pid = pid; + SAFE_STRCPY (things[num_things].card_name, pattrs_id); + SAFE_STRCPY (things[num_things].card_number, pattrs_number); + SAFE_STRCPY (things[num_things].product, udev_device_get_sysattr_value(parentdev,"product")); + SAFE_STRCPY (things[num_things].devnode_sound, devnode); + SAFE_STRCPY (things[num_things].devnode_usb, udev_device_get_devnode(parentdev)); + strlcpy (things[num_things].devpath, card_devpath, sizeof(things[num_things].devpath)); + num_things++; + } + udev_device_unref(parentdev); + } + } + } + udev_enumerate_unref(enumerate); + udev_unref(udev); + +/* + * Now merge in all of the USB HID. + */ + udev = udev_new(); + if (!udev) { + text_color_set(DW_COLOR_ERROR); + dw_printf("INTERNAL ERROR: Can't create udev.\n"); + return (-1); + } + + enumerate = udev_enumerate_new(udev); + udev_enumerate_add_match_subsystem(enumerate, "hidraw"); + udev_enumerate_scan_devices(enumerate); + devices = udev_enumerate_get_list_entry(enumerate); + udev_list_entry_foreach(dev_list_entry, devices) { + const char *path = udev_list_entry_get_name(dev_list_entry); + dev = udev_device_new_from_syspath(udev, path); + char const *devnode = udev_device_get_devnode(dev); + if (devnode != NULL) { + parentdev = udev_device_get_parent_with_subsystem_devtype( dev, "usb", "usb_device"); + if (parentdev != NULL) { + char const *p; + int vid = 0; + int pid = 0; + + p = udev_device_get_sysattr_value(parentdev,"idVendor"); + if (p != NULL) vid = strtol(p, NULL, 16); + p = udev_device_get_sysattr_value(parentdev,"idProduct"); + if (p != NULL) pid = strtol(p, NULL, 16); + + int j, matched = 0; + char const *usb = udev_device_get_devnode(parentdev); + + // Add hidraw name to any matching existing. + for (j = 0; j < num_things; j++) { + if (things[j].vid == vid && things[j].pid == pid && usb != NULL && strcmp(things[j].devnode_usb,usb) == 0) { + matched = 1; + SAFE_STRCPY (things[j].devnode_hidraw, devnode); + } + } + + // If it did not match to existing, add new entry. + if (matched == 0 && num_things < max_things) { + things[num_things].vid = vid; + things[num_things].pid = pid; + SAFE_STRCPY (things[num_things].product, udev_device_get_sysattr_value(parentdev,"product")); + SAFE_STRCPY (things[num_things].devnode_hidraw, devnode); + SAFE_STRCPY (things[num_things].devnode_usb, usb); + SAFE_STRCPY (things[num_things].devpath, udev_device_get_devpath(dev)); + num_things++; + } + udev_device_unref(parentdev); + } + } + } + udev_enumerate_unref(enumerate); + udev_unref(udev); + +/* + * Seeing the form /dev/snd/pcmC4D0p will be confusing to many because we + * would generally something like plughw:4,0 for in the direwolf configuration file. + * Construct the more familiar form. + * Previously we only used the numeric form. In version 1.6, the name is listed as well + * and we describe how to assign names based on the physical USB socket for repeatability. + */ + int i; + regex_t pcm_re; + char emsg[100]; + int e = regcomp (&pcm_re, "pcmC([0-9]+)D([0-9]+)[cp]", REG_EXTENDED); + if (e) { + regerror (e, &pcm_re, emsg, sizeof(emsg)); + text_color_set(DW_COLOR_ERROR); + dw_printf("INTERNAL ERROR: %s:%d: %s\n", __FILE__, __LINE__, emsg); + return (-1); + } + + for (i = 0; i < num_things; i++) { + regmatch_t match[3]; + + if (regexec (&pcm_re, things[i].devnode_sound, 3, match, 0) == 0) { + char c[32], d[32]; + substr_se (c, things[i].devnode_sound, match[1].rm_so, match[1].rm_eo); + substr_se (d, things[i].devnode_sound, match[2].rm_so, match[2].rm_eo); + snprintf (things[i].plughw, sizeof(things[i].plughw), "plughw:%s,%s", c, d); + snprintf (things[i].plughw2, sizeof(things[i].plughw), "plughw:%s,%s", things[i].card_name, d); + } + } + +#endif // end Linux + + return (num_things); + +} /* end cm108_inventory */ + + +/*------------------------------------------------------------------- + * + * Name: cm108_find_ptt + * + * Purpose: Try to find /dev/hidraw corresponding to a USB audio "card." + * + * Inputs: output_audio_device + * - Used in the ADEVICE configuration. + * This can take many forms such as: + * surround41:CARD=Fred,DEV=0 + * surround41:Fred,0 + * surround41:Fred + * plughw:2,3 + * In our case we just need to extract the card number or name. + * + * ptt_device_size - Size of result area to avoid buffer overflow. + * + * Outputs: ptt_device - Device name, something like /dev/hidraw2. + * Will be empty string if no match found. + * + * Returns: none + * + *------------------------------------------------------------------*/ + +void cm108_find_ptt (char *output_audio_device, char *ptt_device, int ptt_device_size) +{ + struct thing_s things[MAXX_THINGS]; + int num_things; + + //dw_printf ("DEBUG: cm108_find_ptt('%s')\n", output_audio_device); + + strlcpy (ptt_device, "", ptt_device_size); + + // Possible improvement: Skip if inventory already taken. + num_things = cm108_inventory (things, MAXX_THINGS); + +#if __WIN32__ + // FIXME - This is just a half baked implementation. + // I have not been able to figure out how to find the connection + // between the audio device and HID in the same package. + // This is fine for a single USB Audio Adapter, good enough for most people. + // Those with multiple devices will need to manually configure PTT device path. + + // Count how many good devices we have. + + int good_devices = 0; + + for (int i = 0; i < num_things; i++) { + if (GOOD_DEVICE(things[i].vid,things[i].pid) ) { + good_devices++; + //dw_printf ("DEBUG: success! returning '%s'\n", things[i].devnode_hidraw); + strlcpy (ptt_device, things[i].devnode_hidraw, ptt_device_size); + } + } + + if (good_devices == 0) return; // None found - caller will print a message. + + if (good_devices == 1) return; // Success - Only one candidate device. + + text_color_set(DW_COLOR_ERROR); + dw_printf ("There are multiple USB Audio Devices with GPIO capability.\n"); + dw_printf ("Explicitly specify one of them for more predictable results:\n"); + for (int i = 0; i < num_things; i++) { + if (GOOD_DEVICE(things[i].vid,things[i].pid) ) { + dw_printf (" \"%s\"\n", things[i].devnode_hidraw); + } + } + dw_printf ("Run the \"cm108\" utility for more details.\n"); + text_color_set(DW_COLOR_INFO); +#else + regex_t sound_re; + char emsg[100]; + int e = regcomp (&sound_re, ".+:(CARD=)?([A-Za-z0-9_]+)(,.*)?", REG_EXTENDED); + if (e) { + regerror (e, &sound_re, emsg, sizeof(emsg)); + text_color_set(DW_COLOR_ERROR); + dw_printf("INTERNAL ERROR: %s:%d: %s\n", __FILE__, __LINE__, emsg); + return; + } + + char num_or_name[64]; + strcpy (num_or_name, ""); + regmatch_t sound_match[4]; + if (regexec (&sound_re, output_audio_device, 4, sound_match, 0) == 0) { + substr_se (num_or_name, output_audio_device, sound_match[2].rm_so, sound_match[2].rm_eo); + //dw_printf ("DEBUG: Got '%s' from '%s'\n", num_or_name, output_audio_device); + } + if (strlen(num_or_name) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not extract card number or name from %s\n", output_audio_device); + dw_printf ("Can't automatically find matching HID for PTT.\n"); + return; + } + + for (int i = 0; i < num_things; i++) { + //dw_printf ("DEBUG: i=%d, card_name='%s', card_number='%s'\n", i, things[i].card_name, things[i].card_number); + if (strcmp(num_or_name,things[i].card_name) == 0 || strcmp(num_or_name,things[i].card_number) == 0) { + //dw_printf ("DEBUG: success! returning '%s'\n", things[i].devnode_hidraw); + strlcpy (ptt_device, things[i].devnode_hidraw, ptt_device_size); + if ( ! GOOD_DEVICE(things[i].vid,things[i].pid) ) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Warning: USB audio card %s (%s) is not a device known to work with GPIO PTT.\n", + things[i].card_number, things[i].card_name); + } + return; + } + } +#endif + +} /* end cm108_find_ptt */ + + + +/*------------------------------------------------------------------- + * + * Name: cm108_set_gpio_pin + * + * Purpose: Set one GPIO pin of the CM108 or similar. + * + * Inputs: name - Name of device such as /dev/hidraw2 or + * \\?\hid#vid_0d8c&pid_0008&mi_03#8&39d3555&0&0000#{4d1e55b2-f16f-11cf-88cb-001111000030} + * + * num - GPIO number, range 1 thru 8. + * + * state - 1 for on, 0 for off. + * + * Returns: 0 for success. -1 for error. + * + * Errors: A descriptive error message will be printed for any problem. + * + * Shortcut: For our initial implementation we are making the simplifying + * restriction of using only one GPIO pin per device and limit + * configuration to PTT only. + * Longer term, we might want to have DCD, and maybe other + * controls thru the same chip. + * In this case, we would need to retain bit masks for each + * device so new data can be merged with old before sending it out. + * + *------------------------------------------------------------------*/ + + +int cm108_set_gpio_pin (char *name, int num, int state) +{ + int iomask; + int iodata; + + if (num < 1 || num > 8) { + text_color_set(DW_COLOR_ERROR); + dw_printf("%s CM108 GPIO number %d must be in range of 1 thru 8.\n", name, num); + return (-1); + } + + if (state != 0 && state != 1) { + text_color_set(DW_COLOR_ERROR); + dw_printf("%s CM108 GPIO state %d must be 0 or 1.\n", name, state); + return (-1); + } + + iomask = 1 << (num - 1); // 0=input, 1=output + iodata = state << (num - 1); // 0=low, 1=high + return (cm108_write (name, iomask, iodata)); + +} /* end cm108_set_gpio_pin */ + + +/*------------------------------------------------------------------- + * + * Name: cm108_write + * + * Purpose: Set the GPIO pins of the CM108 or similar. + * + * Inputs: name - Name of device such as /dev/hidraw2. + * + * iomask - Bit mask for I/O direction. + * LSB is GPIO1, bit 1 is GPIO2, etc. + * 1 for output, 0 for input. + * + * iodata - Output data, same bit order as iomask. + * + * Returns: 0 for success. -1 for error. + * + * Errors: A descriptive error message will be printed for any problem. + * + * Description: This is the lowest level function. + * An application probably wants to use cm108_set_gpio_pin. + * + *------------------------------------------------------------------*/ + +static int cm108_write (char *name, int iomask, int iodata) +{ + +#if __WIN32__ + + //text_color_set(DW_COLOR_DEBUG); + //dw_printf ("TEMP DEBUG cm108_write: %s %d %d\n", name, iomask, iodata); + + hid_device *handle = hid_open_path(name); + if (handle == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not open %s for write\n", name); + return (-1); + } + + unsigned char io[5]; + io[0] = 0; + io[1] = 0; + io[2] = iodata; + io[3] = iomask; + io[4] = 0; + + int res = hid_write(handle, io, sizeof(io)); + if (res < 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Write failed to %s\n", name); + return (-1); + } + + hid_close(handle); + +#else + int fd; + struct hidraw_devinfo info; + char io[5]; + int n; + + //text_color_set(DW_COLOR_DEBUG); + //dw_printf ("TEMP DEBUG cm108_write: %s %d %d\n", name, iomask, iodata); + +/* + * By default, the USB HID are accessible only by root: + * + * crw------- 1 root root 249, 1 ... /dev/hidraw1 + * + * How should we handle this? + * Manually changing it will revert back on the next reboot or + * when the device is removed and reinserted. + * + * According to various articles on the Internet, we should be able to + * add a file to /etc/udev/rules.d. "99-direwolf-cmedia.rules" would be a + * suitable name. The leading number is the order. We want this to be + * near the end. I think the file extension must be ".rules." + * + * We could completely open it up to everyone like this: + * + * # Allow ordinary user to access CMedia GPIO for PTT. + * SUBSYSTEM=="hidraw", ATTRS{idVendor}=="0d8c", MODE="0666" + * + * Whenever we have CMedia USB audio adapter, it should be accessible by everyone. + * This would not apply to other /dev/hidraw* corresponding to keyboard, mouse, etc. + * + * Notice the == (double =) for testing and := for setting a property. + * + * If you are concerned about security, you could restrict access to + * a particular group, something like this: + * + * SUBSYSTEM=="hidraw", ATTRS{idVendor}=="0d8c", GROUP="audio", MODE="0660" + * + * I figure "audio" makes more sense than "gpio" because we need to be part of + * audio group to use the USB Audio adapter for sound. + */ + + fd = open (name, O_WRONLY); + if (fd == -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not open %s for write, errno=%d\n", name, errno); + if (errno == EACCES) { // 13 + dw_printf ("Type \"ls -l %s\" and verify that it has audio group rw similar to this:\n", name); + dw_printf (" crw-rw---- 1 root audio 247, 0 Oct 6 19:24 %s\n", name); + dw_printf ("rather than root-only access like this:\n"); + dw_printf (" crw------- 1 root root 247, 0 Sep 24 09:40 %s\n", name); + } + return (-1); + } + + // Just for fun, let's get the device information. + +#if 1 + n = ioctl(fd, HIDIOCGRAWINFO, &info); + if (n == 0) { + if ( ! GOOD_DEVICE(info.vendor, info.product)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ioctl HIDIOCGRAWINFO failed for %s. errno = %d.\n", name, errno); + } + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("%s is not a supported device type. Proceed at your own risk. vid=%04x pid=%04x\n", name, info.vendor, info.product); + } +#endif + // To make a long story short, I think we need 0 for the first two bytes. + + io[0] = 0; + io[1] = 0; +// Issue 210 - These were reversed. Fixed in 1.6. + io[2] = iodata; + io[3] = iomask; + io[4] = 0; + + // Writing 4 bytes fails with errno 32, EPIPE, "broken pipe." + // Hamlib writes 5 bytes which I don't understand. + // Writing 5 bytes works. + // I have no idea why. From the CMedia datasheet it looks like we need 4. + + n = write (fd, io, sizeof(io)); + if (n != sizeof(io)) { + // Errors observed during development. + // as pi EACCES 13 /* Permission denied */ + // as root EPIPE 32 /* Broken pipe - Happens if we send 4 bytes */ + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Write to %s failed, n=%d, errno=%d\n", name, n, errno); + + if (errno == EACCES) { + dw_printf ("Type \"ls -l %s\" and verify that it has audio group rw similar to this:\n", name); + dw_printf (" crw-rw---- 1 root audio 247, 0 Oct 6 19:24 %s\n", name); + dw_printf ("rather than root-only access like this:\n"); + dw_printf (" crw------- 1 root root 247, 0 Sep 24 09:40 %s\n", name); + } + + close (fd); + return (-1); + } + + close (fd); + +#endif + return (0); + +} /* end cm108_write */ + +#endif // ifdef USE_CM108 + +/* end cm108.c */ + + diff --git a/src/cm108.h b/src/cm108.h new file mode 100644 index 00000000..2def77a8 --- /dev/null +++ b/src/cm108.h @@ -0,0 +1,5 @@ +/* Dire Wolf cm108.h */ + +extern void cm108_find_ptt (char *output_audio_device, char *ptt_device, int ptt_device_size); + +extern int cm108_set_gpio_pin (char *name, int num, int state); \ No newline at end of file diff --git a/config.c b/src/config.c similarity index 75% rename from config.c rename to src/config.c index 455a0d31..1ad6c081 100644 --- a/config.c +++ b/src/config.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2021 John Langner, WB2OSZ // // 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 @@ -63,6 +63,10 @@ #include "tt_text.h" #include "ax25_link.h" +#if USE_CM108 // Current Linux or Windows only +#include "cm108.h" +#endif + // geotranz #include "utm.h" @@ -75,8 +79,6 @@ -//#include "tq.h" - /* * Conversions from various units to meters. * There is some disagreement about the exact values for some of these. @@ -315,7 +317,7 @@ static double parse_ll (char *str, enum parse_ll_which_e which, int line) * - Negative zone for south. * - Separate North or South. * - * I'm using the first alternatve. + * I'm using the first alternative. * GEOTRANS uses the third. * We will also recognize the second one but I'm not sure if I want to document it. * @@ -515,7 +517,7 @@ static int check_via_path (char *via_path) r = stemp; while (( a = strsep(&r,",")) != NULL) { - int strict = 1; + int strict = 2; int ok; char addr[AX25_MAX_ADDR_LEN]; int ssid; @@ -583,14 +585,17 @@ static int check_via_path (char *via_path) * *--------------------------------------------------------------------*/ -#define MAXCMDLEN 256 +#define MAXCMDLEN 1200 static char *split (char *string, int rest_of_line) { static char cmd[MAXCMDLEN]; static char token[MAXCMDLEN]; - static char *c; // current position in cmd. + static char shutup[] = " "; // Shut up static analysis which gets upset + // over the case where this could be called with + // string NULL and c was not yet initialized. + static char *c = shutup; // Current position in command line. char *s, *t; int in_quotes; @@ -704,6 +709,14 @@ static char *split (char *string, int rest_of_line) * *--------------------------------------------------------------------*/ +static void rtfm() +{ + text_color_set(DW_COLOR_ERROR); + dw_printf ("See online documentation:\n"); + dw_printf (" stable release: https://github.com/wb2osz/direwolf/tree/master/doc\n"); + dw_printf (" development version: https://github.com/wb2osz/direwolf/tree/dev/doc\n"); + dw_printf (" additional topics: https://github.com/wb2osz/direwolf-doc\n"); +} void config_init (char *fname, struct audio_s *p_audio_config, struct digi_config_s *p_digi_config, @@ -731,6 +744,8 @@ void config_init (char *fname, struct audio_s *p_audio_config, memset (p_audio_config, 0, sizeof(struct audio_s)); + p_audio_config->igate_vchannel = -1; // none. + /* First audio device is always available with defaults. */ /* Others must be explicitly defined before use. */ @@ -750,10 +765,11 @@ void config_init (char *fname, struct audio_s *p_audio_config, for (channel=0; channelachan[channel].valid = 0; /* One or both channels will be */ - /* set to valid when corresponding */ + p_audio_config->chan_medium[channel] = MEDIUM_NONE; /* One or both channels will be */ + /* set to radio when corresponding */ /* audio device is defined. */ p_audio_config->achan[channel].modem_type = MODEM_AFSK; + p_audio_config->achan[channel].v26_alternative = V26_UNSPECIFIED; p_audio_config->achan[channel].mark_freq = DEFAULT_MARK_FREQ; /* -m option */ p_audio_config->achan[channel].space_freq = DEFAULT_SPACE_FREQ; /* -s option */ p_audio_config->achan[channel].baud = DEFAULT_BAUD; /* -b option */ @@ -764,6 +780,10 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->achan[channel].num_freq = 1; p_audio_config->achan[channel].offset = 0; + p_audio_config->achan[channel].layer2_xmit = LAYER2_AX25; + p_audio_config->achan[channel].il2p_max_fec = 1; + p_audio_config->achan[channel].il2p_invert_polarity = 0; + p_audio_config->achan[channel].fix_bits = DEFAULT_FIX_BITS; p_audio_config->achan[channel].sanity_test = SANITY_APRS; p_audio_config->achan[channel].passall = 0; @@ -790,17 +810,19 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->achan[channel].persist = DEFAULT_PERSIST; p_audio_config->achan[channel].txdelay = DEFAULT_TXDELAY; p_audio_config->achan[channel].txtail = DEFAULT_TXTAIL; + p_audio_config->achan[channel].fulldup = DEFAULT_FULLDUP; } + p_audio_config->fx25_auto_enable = AX25_N2_RETRY_DEFAULT / 2; + /* First channel should always be valid. */ /* If there is no ADEVICE, it uses default device in mono. */ - p_audio_config->achan[0].valid = 1; + p_audio_config->chan_medium[0] = MEDIUM_RADIO; - - memset (p_digi_config, 0, sizeof(struct digi_config_s)); + memset (p_digi_config, 0, sizeof(struct digi_config_s)); // APRS digipeater p_digi_config->dedupe_time = DEFAULT_DEDUPE; - memset (p_cdigi_config, 0, sizeof(struct cdigi_config_s)); + memset (p_cdigi_config, 0, sizeof(struct cdigi_config_s)); // Connected mode digipeater memset (p_tt_config, 0, sizeof(struct tt_config_s)); p_tt_config->gateway_enabled = 0; @@ -844,8 +866,18 @@ void config_init (char *fname, struct audio_s *p_audio_config, memset (p_misc_config, 0, sizeof(struct misc_config_s)); p_misc_config->agwpe_port = DEFAULT_AGWPE_PORT; - p_misc_config->kiss_port = DEFAULT_KISS_PORT; + + for (int i=0; ikiss_port[i] = 0; // entry not used. + p_misc_config->kiss_chan[i] = -1; + } + p_misc_config->kiss_port[0] = DEFAULT_KISS_PORT; + p_misc_config->kiss_chan[0] = -1; // all channels. + p_misc_config->enable_kiss_pt = 0; /* -p option */ + p_misc_config->kiss_copy = 0; + + p_misc_config->dns_sd_enabled = 1; /* Defaults from http://info.aprs.net/index.php?title=SmartBeaconing */ @@ -864,19 +896,22 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_igate_config->tx_limit_1 = IGATE_TX_LIMIT_1_DEFAULT; p_igate_config->tx_limit_5 = IGATE_TX_LIMIT_5_DEFAULT; p_igate_config->igmsp = 1; + p_igate_config->rx2ig_dedupe_time = IGATE_RX2IG_DEDUPE_TIME; /* People find this confusing. */ /* Ideally we'd like to figure out if com0com is installed */ /* and automatically enable this. */ - //strlcpy (p_misc_config->nullmodem, DEFAULT_NULLMODEM, sizeof(p_misc_config->nullmodem)); - strlcpy (p_misc_config->nullmodem, "", sizeof(p_misc_config->nullmodem)); + strlcpy (p_misc_config->kiss_serial_port, "", sizeof(p_misc_config->kiss_serial_port)); + p_misc_config->kiss_serial_speed = 0; + p_misc_config->kiss_serial_poll = 0; strlcpy (p_misc_config->gpsnmea_port, "", sizeof(p_misc_config->gpsnmea_port)); - strlcpy (p_misc_config->waypoint_port, "", sizeof(p_misc_config->waypoint_port)); + strlcpy (p_misc_config->waypoint_serial_port, "", sizeof(p_misc_config->waypoint_serial_port)); - strlcpy (p_misc_config->logdir, "", sizeof(p_misc_config->logdir)); + p_misc_config->log_daily_names = 0; + strlcpy (p_misc_config->log_path, "", sizeof(p_misc_config->log_path)); /* connected mode. */ @@ -890,7 +925,11 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_misc_config->maxframe_extended = AX25_K_MAXFRAME_EXTENDED_DEFAULT; /* Max frames to send before ACK. mod 128 "Window" size. */ - p_misc_config->maxv22 = AX25_N2_RETRY_DEFAULT / 2; /* Max SABME before falling back to SABM. */ + p_misc_config->maxv22 = AX25_N2_RETRY_DEFAULT / 3; /* Max SABME before falling back to SABM. */ + p_misc_config->v20_addrs = NULL; /* Go directly to v2.0 for stations listed. */ + p_misc_config->v20_count = 0; + p_misc_config->noxid_addrs = NULL; /* Don't send XID to these stations. */ + p_misc_config->noxid_count = 0; /* * Try to extract options from a file. @@ -939,7 +978,8 @@ void config_init (char *fname, struct audio_s *p_audio_config, text_color_set(DW_COLOR_ERROR); dw_printf ("ERROR - Could not open config file %s\n", filepath); dw_printf ("Try using -c command line option for alternate location.\n"); - return; + rtfm(); + exit(EXIT_FAILURE); } dw_printf ("\nReading config file %s\n", filepath); @@ -971,20 +1011,23 @@ void config_init (char *fname, struct audio_s *p_audio_config, * * ADEVICE plughw:1,0 -- same for in and out. * ADEVICE plughw:2,0 plughw:3,0 -- different in/out for a channel or channel pair. + * ADEVICE1 udp:7355 default -- from Software defined radio (SDR) via UDP. * */ /* Note that ALSA name can contain comma such as hw:1,0 */ if (strncasecmp(t, "ADEVICE", 7) == 0) { + /* "ADEVICE" is equivalent to "ADEVICE0". */ adevice = 0; - if (isdigit(t[7])) { - adevice = t[7] - '0'; + if (strlen(t) >= 8) { + adevice = atoi(t+7); } if (adevice < 0 || adevice >= MAX_ADEVS) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file: Device number %d out of range for ADEVICE command on line %d.\n", adevice, line); + dw_printf ("If you really need more than %d audio devices, increase MAX_ADEVS and recompile.\n", MAX_ADEVS); adevice = 0; continue; } @@ -993,13 +1036,14 @@ void config_init (char *fname, struct audio_s *p_audio_config, if (t == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file: Missing name of audio device for ADEVICE command on line %d.\n", line); - continue; + rtfm(); + exit(EXIT_FAILURE); } p_audio_config->adev[adevice].defined = 1; /* First channel of device is valid. */ - p_audio_config->achan[ADEVFIRSTCHAN(adevice)].valid = 1; + p_audio_config->chan_medium[ADEVFIRSTCHAN(adevice)] = MEDIUM_RADIO; strlcpy (p_audio_config->adev[adevice].adevice_in, t, sizeof(p_audio_config->adev[adevice].adevice_in)); strlcpy (p_audio_config->adev[adevice].adevice_out, t, sizeof(p_audio_config->adev[adevice].adevice_out)); @@ -1054,7 +1098,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->adev[adevice].defined = 1; /* First channel of device is valid. */ - p_audio_config->achan[ADEVFIRSTCHAN(adevice)].valid = 1; + p_audio_config->chan_medium[ADEVFIRSTCHAN(adevice)] = MEDIUM_RADIO; strlcpy (p_audio_config->adev[adevice].adevice_in, t, sizeof(p_audio_config->adev[adevice].adevice_in)); } @@ -1081,7 +1125,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->adev[adevice].defined = 1; /* First channel of device is valid. */ - p_audio_config->achan[ADEVFIRSTCHAN(adevice)].valid = 1; + p_audio_config->chan_medium[ADEVFIRSTCHAN(adevice)] = MEDIUM_RADIO; strlcpy (p_audio_config->adev[adevice].adevice_out, t, sizeof(p_audio_config->adev[adevice].adevice_out)); } @@ -1128,9 +1172,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, /* Set valid channels depending on mono or stereo. */ - p_audio_config->achan[ADEVFIRSTCHAN(adevice)].valid = 1; + p_audio_config->chan_medium[ADEVFIRSTCHAN(adevice)] = MEDIUM_RADIO; if (n == 2) { - p_audio_config->achan[ADEVFIRSTCHAN(adevice) + 1].valid = 1; + p_audio_config->chan_medium[ADEVFIRSTCHAN(adevice) + 1] = MEDIUM_RADIO; } } else { @@ -1144,7 +1188,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, */ /* - * CHANNEL - Set channel for following commands. + * CHANNEL n - Set channel for channel-specific commands. */ else if (strcasecmp(t, "CHANNEL") == 0) { @@ -1160,7 +1204,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, channel = n; - if ( ! p_audio_config->achan[n].valid) { + if (p_audio_config->chan_medium[n] != MEDIUM_RADIO) { if ( ! p_audio_config->adev[ACHAN2ADEV(n)].defined) { text_color_set(DW_COLOR_ERROR); @@ -1180,6 +1224,44 @@ void config_init (char *fname, struct audio_s *p_audio_config, } } +/* + * ICHANNEL n - Define IGate virtual channel. + * + * This allows a client application to talk to to APRS-IS + * by using a channel number outside the normal range for modems. + * In the future there might be other typs of virtual channels. + * This does not change the current channel number used by MODEM, PTT, etc. + */ + + else if (strcasecmp(t, "ICHANNEL") == 0) { + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing virtual channel number for ICHANNEL command.\n", line); + continue; + } + int ichan = atoi(t); + if (ichan >= MAX_CHANS && ichan < MAX_TOTAL_CHANS) { + + if (p_audio_config->chan_medium[ichan] == MEDIUM_NONE) { + + p_audio_config->chan_medium[ichan] = MEDIUM_IGATE; + + // This is redundant but saves the time of searching through all + // the channels for each packet. + p_audio_config->igate_vchannel = ichan; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: ICHANNEL can't use %d because it is already in use.\n", line, ichan); + } + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: ICHANNEL number must in range of %d to %d.\n", line, MAX_CHANS, MAX_TOTAL_CHANS-1); + } + } + /* * MYCALL station */ @@ -1192,6 +1274,24 @@ void config_init (char *fname, struct audio_s *p_audio_config, } else { + char *p; + int const strict = 2; + char call_no_ssid[AX25_MAX_ADDR_LEN]; + int ssid, heard; + + for (p = t; *p != '\0'; p++) { + if (islower(*p)) { + *p = toupper(*p); /* Silently force upper case. */ + /* Might change to warning someday. */ + } + } + + if ( ! ax25_parse_addr (-1, t, strict, call_no_ssid, &ssid, &heard)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Invalid value for MYCALL command on line %d.\n", line); + continue; + } + // Definitely set for current channel. // Set for other channels which have not been set yet. @@ -1204,17 +1304,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, strcasecmp(p_audio_config->achan[c].mycall, "NOCALL") == 0 || strcasecmp(p_audio_config->achan[c].mycall, "N0CALL") == 0) { - char *p; - strlcpy (p_audio_config->achan[c].mycall, t, sizeof(p_audio_config->achan[c].mycall)); - - for (p = p_audio_config->achan[c].mycall; *p != '\0'; p++) { - if (islower(*p)) { - *p = toupper(*p); /* silently force upper case. */ - } - } - // TODO: additional checks if valid. - // Should have a function to check for valid callsign[-ssid] } } } @@ -1234,7 +1324,11 @@ void config_init (char *fname, struct audio_s *p_audio_config, * Options: * mark:space - AFSK tones. Defaults based on speed. * num@offset - Multiple decoders on different frequencies. - * + * /9 - Divide sample rate by specified number. + * *9 - Upsample ratio for G3RUH. + * [A-Z+-]+ - Letters, plus, minus for the demodulator "profile." + * g3ruh - This modem type regardless of default for speed. + * v26a or v26b - V.26 alternative. a=original, b=MFJ compatible */ else if (strcasecmp(t, "MODEM") == 0) { @@ -1245,10 +1339,18 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Line %d: Missing data transmission speed for MODEM command.\n", line); continue; } - n = atoi(t); + if (strcasecmp(t,"AIS") == 0) { + n = MAX_BAUD-1; // Hack - See special case later. + } + else if (strcasecmp(t,"EAS") == 0) { + n = MAX_BAUD-2; // Hack - See special case later. + } + else { + n = atoi(t); + } if (n >= MIN_BAUD && n <= MAX_BAUD) { p_audio_config->achan[channel].baud = n; - if (n != 300 && n != 1200 && n != 2400 && n != 4800 && n != 9600 && n != 19200) { + if (n != 300 && n != 1200 && n != 2400 && n != 4800 && n != 9600 && n != 19200 && n != MAX_BAUD-1 && n != MAX_BAUD-2) { text_color_set(DW_COLOR_ERROR); dw_printf ("Line %d: Warning: Non-standard data rate of %d bits per second. Are you sure?\n", line, n); } @@ -1287,13 +1389,26 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->achan[channel].mark_freq = 0; p_audio_config->achan[channel].space_freq = 0; } + else if (p_audio_config->achan[channel].baud == MAX_BAUD-1) { + p_audio_config->achan[channel].modem_type = MODEM_AIS; + p_audio_config->achan[channel].mark_freq = 0; + p_audio_config->achan[channel].space_freq = 0; + } + else if (p_audio_config->achan[channel].baud == MAX_BAUD-2) { + p_audio_config->achan[channel].modem_type = MODEM_EAS; + p_audio_config->achan[channel].baud = 521; // Actually 520.83 but we have an integer field here. + // Will make more precise in afsk demod init. + p_audio_config->achan[channel].mark_freq = 2083; // Actually 2083.3 - logic 1. + p_audio_config->achan[channel].space_freq = 1563; // Actually 1562.5 - logic 0. + // ? strlcpy (p_audio_config->achan[channel].profiles, "A", sizeof(p_audio_config->achan[channel].profiles)); + } else { p_audio_config->achan[channel].modem_type = MODEM_SCRAMBLE; p_audio_config->achan[channel].mark_freq = 0; p_audio_config->achan[channel].space_freq = 0; } - /* Get mark frequency. */ + /* Get any options. */ t = split(NULL,0); if (t == NULL) { @@ -1305,6 +1420,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, /* old style */ + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Old style (pre version 1.2) format will no longer be supported in next version.\n", line); + n = atoi(t); /* Originally the upper limit was 3000. */ /* Version 1.0 increased to 5000 because someone */ @@ -1489,6 +1607,38 @@ void config_init (char *fname, struct audio_s *p_audio_config, } } + else if (*t == '*') { /* *upsample */ + int n = atoi(t+1); + + if (n >= 1 && n <= 4) { + p_audio_config->achan[channel].upsample = n; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Ignoring unreasonable upsample ratio of %d.\n", line, n); + } + } + + else if (strcasecmp(t, "G3RUH") == 0) { /* Force G3RUH modem regardless of default for speed. New in 1.6. */ + + p_audio_config->achan[channel].modem_type = MODEM_SCRAMBLE; + p_audio_config->achan[channel].mark_freq = 0; + p_audio_config->achan[channel].space_freq = 0; + } + + else if (strcasecmp(t, "V26A") == 0 || /* Compatible with direwolf versions <= 1.5. New in 1.6. */ + strcasecmp(t, "V26B") == 0) { /* Compatible with MFJ-2400. New in 1.6. */ + + if (p_audio_config->achan[channel].modem_type != MODEM_QPSK || + p_audio_config->achan[channel].baud != 2400) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: %s option can only be used with 2400 bps PSK.\n", line, t); + continue; + } + p_audio_config->achan[channel].v26_alternative = (strcasecmp(t, "V26A") == 0) ? V26_A : V26_B; + } + else { text_color_set(DW_COLOR_ERROR); dw_printf ("Line %d: Unrecognized option for MODEM: %s\n", line, t); @@ -1526,7 +1676,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, /* * FIX_BITS n [ APRS | AX25 | NONE ] [ PASSALL ] * - * - Attempt to fix frames with bad FCS. + * - Attempt to fix frames with bad FCS. + * - n is maximum number of bits to attempt fixing. + * - Optional sanity check & allow everything even with bad FCS. */ else if (strcasecmp(t, "FIX_BITS") == 0) { @@ -1548,16 +1700,13 @@ void config_init (char *fname, struct audio_s *p_audio_config, line, n, p_audio_config->achan[channel].fix_bits); } - if (p_audio_config->achan[channel].fix_bits > RETRY_INVERT_SINGLE) { + if (p_audio_config->achan[channel].fix_bits > DEFAULT_FIX_BITS) { text_color_set(DW_COLOR_INFO); dw_printf ("Line %d: Using a FIX_BITS value greater than %d is not recommended for normal operation.\n", - line, RETRY_INVERT_SINGLE); - } - - if (p_audio_config->achan[channel].fix_bits >= RETRY_INVERT_TWO_SEP) { - text_color_set(DW_COLOR_INFO); - dw_printf ("Line %d: Using a FIX_BITS value of %d will waste a lot of CPU power and produce wrong results.\n", - line, RETRY_INVERT_TWO_SEP); + line, DEFAULT_FIX_BITS); + dw_printf ("FIX_BITS > 1 was an interesting experiment but turned out to be a bad idea.\n"); + dw_printf ("Don't be surprised if it takes 100%% CPU, direwolf can't keep up with the audio stream,\n"); + dw_printf ("and you see messages like \"Audio input device 0 error code -32: Broken pipe\"\n"); } t = split(NULL,0); @@ -1599,8 +1748,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, * xxx serial-port [-]rts-or-dtr [ [-]rts-or-dtr ] * xxx GPIO [-]gpio-num * xxx LPT [-]bit-num - * PTT RIG model port - * PTT RIG AUTO port + * PTT RIG model port [ rate ] + * PTT RIG AUTO port [ rate ] + * PTT CM108 [ [-]bit-num ] [ hid-device ] * * When model is 2, port would host:port like 127.0.0.1:4532 * Otherwise, port would be a serial port like /dev/ttyS0 @@ -1629,7 +1779,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file line %d: Missing serial port name for %s command.\n", + dw_printf ("Config file line %d: Missing output control device for %s command.\n", line, otname); continue; } @@ -1700,6 +1850,13 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->achan[channel].octrl[ot].ptt_model = -1; } else { + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: A rig number, not a name, is required here.\n", line); + dw_printf ("For example, if you have a Yaesu FT-847, specify 101.\n"); + dw_printf ("See https://github.com/Hamlib/Hamlib/wiki/Supported-Radios for more details.\n"); + continue; + } int n = atoi(t); if (n < 1 || n > 9999) { text_color_set(DW_COLOR_ERROR); @@ -1717,6 +1874,19 @@ void config_init (char *fname, struct audio_s *p_audio_config, } strlcpy (p_audio_config->achan[channel].octrl[ot].ptt_device, t, sizeof(p_audio_config->achan[channel].octrl[ot].ptt_device)); + // Optional serial port rate for CAT control PTT. + + t = split(NULL,0); + if (t != NULL) { + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: An optional number is required here for CAT serial port speed: %s\n", line, t); + continue; + } + int n = atoi(t); + p_audio_config->achan[channel].octrl[ot].ptt_rate = n; + } + t = split(NULL,0); if (t != NULL) { text_color_set(DW_COLOR_ERROR); @@ -1725,9 +1895,109 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_audio_config->achan[channel].octrl[ot].ptt_method = PTT_METHOD_HAMLIB; +#else +#if __WIN32__ + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: Windows version of direwolf does not support HAMLIB.\n", line); + exit (EXIT_FAILURE); #else text_color_set(DW_COLOR_ERROR); dw_printf ("Config file line %d: %s with RIG is only available when hamlib support is enabled.\n", line, otname); + dw_printf ("You must rebuild direwolf with hamlib support.\n"); + dw_printf ("See User Guide for details.\n"); +#endif + +#endif + } + else if (strcasecmp(t, "CM108") == 0) { + +/* CM108 - GPIO of USB sound card. case, Linux and Windows only. */ + +#if USE_CM108 + + if (ot != OCTYPE_PTT) { + // Future project: Allow DCD and CON via the same device. + // This gets more complicated because we can't selectively change a single GPIO bit. + // We would need to keep track of what is currently there, change one bit, in our local + // copy of the status and then write out the byte for all of the pins. + // Let's keep it simple with just PTT for the first stab at this. + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: PTT CM108 option is only valid for PTT, not %s.\n", line, otname); + continue; + } + + p_audio_config->achan[channel].octrl[ot].out_gpio_num = 3; // All known designs use GPIO 3. + // User can override for special cases. + p_audio_config->achan[channel].octrl[ot].ptt_invert = 0; // High for transmit. + strcpy (p_audio_config->achan[channel].octrl[ot].ptt_device, ""); + + // Try to find PTT device for audio output device. + // Simplifiying assumption is that we have one radio per USB Audio Adapter. + // Failure at this point is not an error. + // See if config file sets it explicitly before complaining. + + cm108_find_ptt (p_audio_config->adev[ACHAN2ADEV(channel)].adevice_out, + p_audio_config->achan[channel].octrl[ot].ptt_device, + (int)sizeof(p_audio_config->achan[channel].octrl[ot].ptt_device)); + + while ((t = split(NULL,0)) != NULL) { + if (*t == '-') { + p_audio_config->achan[channel].octrl[ot].out_gpio_num = atoi(t+1); + p_audio_config->achan[channel].octrl[ot].ptt_invert = 1; + } + else if (isdigit(*t)) { + p_audio_config->achan[channel].octrl[ot].out_gpio_num = atoi(t); + p_audio_config->achan[channel].octrl[ot].ptt_invert = 0; + } +#if __WIN32__ + else if (*t == '\\') { + strlcpy (p_audio_config->achan[channel].octrl[ot].ptt_device, t, sizeof(p_audio_config->achan[channel].octrl[ot].ptt_device)); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: Found \"%s\" when expecting GPIO number or device name like \\\\?\\hid#vid_0d8c&... .\n", line, t); + continue; + } +#else + else if (*t == '/') { + strlcpy (p_audio_config->achan[channel].octrl[ot].ptt_device, t, sizeof(p_audio_config->achan[channel].octrl[ot].ptt_device)); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: Found \"%s\" when expecting GPIO number or device name like /dev/hidraw1.\n", line, t); + continue; + } +#endif + } + if (p_audio_config->achan[channel].octrl[ot].out_gpio_num < 1 || p_audio_config->achan[channel].octrl[ot].out_gpio_num > 8) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: CM108 GPIO number %d is not in range of 1 thru 8.\n", line, + p_audio_config->achan[channel].octrl[ot].out_gpio_num); + continue; + } + if (strlen(p_audio_config->achan[channel].octrl[ot].ptt_device) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: Could not determine USB Audio GPIO PTT device for audio output %s.\n", line, + p_audio_config->adev[ACHAN2ADEV(channel)].adevice_out); +#if __WIN32__ + dw_printf ("You must explicitly mention a HID path.\n"); +#else + dw_printf ("You must explicitly mention a device name such as /dev/hidraw1.\n"); +#endif + dw_printf ("Run \"cm108\" utility to get a list.\n"); + dw_printf ("See Interface Guide for details.\n"); + continue; + } + p_audio_config->achan[channel].octrl[ot].ptt_method = PTT_METHOD_CM108; + +#else + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file line %d: %s with CM108 is only available when USB Audio GPIO support is enabled.\n", line, otname); + dw_printf ("You must rebuild direwolf with CM108 Audio Adapter GPIO PTT support.\n"); + dw_printf ("See Interface Guide for details.\n"); + rtfm(); + exit (EXIT_FAILURE); #endif } else { @@ -1818,7 +2088,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, * * TXINH - TX holdoff input * - * xxx GPIO [-]gpio-num (only type supported yet) + * TXINH GPIO [-]gpio-num (only type supported so far) */ else if (strcasecmp(t, "TXINH") == 0) { @@ -1861,7 +2131,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, /* - * DWAIT - Extra delay for receiver squelch. + * DWAIT n - Extra delay for receiver squelch. n = 10 mS units. */ else if (strcasecmp(t, "DWAIT") == 0) { @@ -1885,7 +2155,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * SLOTTIME - For non-digipeat transmit delay timing. + * SLOTTIME n - For non-digipeat transmit delay timing. n = 10 mS units. */ else if (strcasecmp(t, "SLOTTIME") == 0) { @@ -1933,7 +2203,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * TXDELAY - For transmit delay timing. + * TXDELAY n - For transmit delay timing. n = 10 mS units. */ else if (strcasecmp(t, "TXDELAY") == 0) { @@ -1957,7 +2227,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * TXTAIL - For transmit timing. + * TXTAIL n - For transmit timing. n = 10 mS units. */ else if (strcasecmp(t, "TXTAIL") == 0) { @@ -1980,6 +2250,30 @@ void config_init (char *fname, struct audio_s *p_audio_config, } } +/* + * FULLDUP {on|off} - Full Duplex + */ + else if (strcasecmp(t, "FULLDUP") == 0) { + + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing parameter for FULLDUP command. Expecting ON or OFF.\n", line); + continue; + } + if (strcasecmp(t, "ON") == 0) { + p_audio_config->achan[channel].fulldup = 1; + } + else if (strcasecmp(t, "OFF") == 0) { + p_audio_config->achan[channel].fulldup = 0; + } + else { + p_audio_config->achan[channel].fulldup = 0; + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Expected ON or OFF for FULLDUP.\n", line); + } + } + /* * SPEECH script * @@ -2010,12 +2304,116 @@ void config_init (char *fname, struct audio_s *p_audio_config, } } +/* + * FX25TX n - Enable FX.25 transmission. Default off. + * 0 = off, 1 = auto mode, others are suggestions for testing + * or special cases. 16, 32, 64 is number of parity bytes to add. + * Also set by "-X n" command line option. + * V1.7 changed from global to per-channel setting. + */ + + else if (strcasecmp(t, "FX25TX") == 0) { + int n; + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing FEC mode for FX25TX command.\n", line); + continue; + } + n = atoi(t); + if (n >= 0 && n < 200) { + p_audio_config->achan[channel].fx25_strength = n; + p_audio_config->achan[channel].layer2_xmit = LAYER2_FX25; + } + else { + p_audio_config->achan[channel].fx25_strength = 1; + p_audio_config->achan[channel].layer2_xmit = LAYER2_FX25; + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Unreasonable value for FX.25 transmission mode. Using %d.\n", + line, p_audio_config->achan[channel].fx25_strength); + } + } + +/* + * FX25AUTO n - Enable Automatic use of FX.25 for connected mode. + * Automatically enable, for that session only, when an identical + * frame is sent more than this number of times. + * Default 5 based on half of default RETRY. + * 0 to disable feature. + * Current a global setting. Could be per channel someday. + */ + + else if (strcasecmp(t, "FX25AUTO") == 0) { + int n; + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing count for FX25AUTO command.\n", line); + continue; + } + n = atoi(t); + if (n >= 0 && n < 20) { + p_audio_config->fx25_auto_enable = n; + } + else { + p_audio_config->fx25_auto_enable = AX25_N2_RETRY_DEFAULT / 2; + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Unreasonable count for connected mode automatic FX.25. Using %d.\n", + line, p_audio_config->fx25_auto_enable); + } + } + +/* + * IL2PTX [ + - ] [ 0 1 ] - Enable IL2P transmission. Default off. + * "+" means normal polarity. Redundant since it is the default. + * (command line -I for first channel) + * "-" means inverted polarity. Do not use for 1200 bps. + * (command line -i for first channel) + * "0" means weak FEC. Not recommended. + * "1" means stronger FEC. "Max FEC." Default if not specified. + */ + + else if (strcasecmp(t, "IL2PTX") == 0) { + + p_audio_config->achan[channel].layer2_xmit = LAYER2_IL2P; + p_audio_config->achan[channel].il2p_max_fec = 1; + p_audio_config->achan[channel].il2p_invert_polarity = 0; + + while ((t = split(NULL,0)) != NULL) { + for (char *c = t; *c != '\0'; c++) { + switch (*c) { + case '+': + p_audio_config->achan[channel].il2p_invert_polarity = 0; + break; + case '-': + p_audio_config->achan[channel].il2p_invert_polarity = 1; + break; + case '0': + p_audio_config->achan[channel].il2p_max_fec = 0; + break; + case '1': + p_audio_config->achan[channel].il2p_max_fec = 1; + break; + default: + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Invalid parameter '%c' fol IL2PTX command.\n", line, *c); + continue; + break; + } + } + } + } + + /* * ==================== APRS Digipeater parameters ==================== */ /* - * DIGIPEAT from-chan to-chan alias-pattern wide-pattern [ OFF|DROP|MARK|TRACE ] + * DIGIPEAT from-chan to-chan alias-pattern wide-pattern [ OFF|DROP|MARK|TRACE | ATGP=alias ] + * + * ATGP is an ugly hack for the specific need of ATGP which needs more that 8 digipeaters. + * DO NOT put this in the User Guide. On a need to know basis. */ else if (strcasecmp(t, "DIGIPEAT") == 0 || strcasecmp(t, "DIGIPEATER") == 0) { @@ -2030,6 +2428,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Missing FROM-channel on line %d.\n", line); continue; } + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: '%s' is not allowed for FROM-channel. It must be a number.\n", + line, t); + continue; + } from_chan = atoi(t); if (from_chan < 0 || from_chan >= MAX_CHANS) { text_color_set(DW_COLOR_ERROR); @@ -2037,7 +2441,11 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[from_chan].valid) { + + // Channels specified must be radio channels or network TNCs. + + if (p_audio_config->chan_medium[from_chan] != MEDIUM_RADIO && + p_audio_config->chan_medium[from_chan] != MEDIUM_NETTNC) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: FROM-channel %d is not valid.\n", line, from_chan); @@ -2050,6 +2458,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Missing TO-channel on line %d.\n", line); continue; } + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: '%s' is not allowed for TO-channel. It must be a number.\n", + line, t); + continue; + } to_chan = atoi(t); if (to_chan < 0 || to_chan >= MAX_CHANS) { text_color_set(DW_COLOR_ERROR); @@ -2057,7 +2471,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[to_chan].valid) { + + if (p_audio_config->chan_medium[to_chan] != MEDIUM_RADIO && + p_audio_config->chan_medium[to_chan] != MEDIUM_NETTNC) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TO-channel %d is not valid.\n", line, to_chan); @@ -2104,17 +2520,29 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,0); } else if (strcasecmp(t, "DROP") == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Preemptive digipeating DROP option is discouraged.\n", line); + dw_printf ("It can create a via path which is misleading about the actual path taken.\n"); + dw_printf ("PREEMPT is the best choice for this feature.\n"); p_digi_config->preempt[from_chan][to_chan] = PREEMPT_DROP; t = split(NULL,0); } else if (strcasecmp(t, "MARK") == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Preemptive digipeating MARK option is discouraged.\n", line); + dw_printf ("It can create a via path which is misleading about the actual path taken.\n"); + dw_printf ("PREEMPT is the best choice for this feature.\n"); p_digi_config->preempt[from_chan][to_chan] = PREEMPT_MARK; t = split(NULL,0); } - else if (strcasecmp(t, "TRACE") == 0) { + else if ((strcasecmp(t, "TRACE") == 0) || (strncasecmp(t, "PREEMPT", 7) == 0)){ p_digi_config->preempt[from_chan][to_chan] = PREEMPT_TRACE; t = split(NULL,0); } + else if (strncasecmp(t, "ATGP=", 5) == 0) { + strlcpy (p_digi_config->atgp[from_chan][to_chan], t+5, sizeof(p_digi_config->atgp[from_chan][to_chan]));; + t = split(NULL,0); + } } if (t != NULL) { @@ -2124,7 +2552,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * DEDUPE - Time to suppress digipeating of duplicate packets. + * DEDUPE - Time to suppress digipeating of duplicate APRS packets. */ else if (strcasecmp(t, "DEDUPE") == 0) { @@ -2161,6 +2589,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Missing FROM-channel on line %d.\n", line); continue; } + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: '%s' is not allowed for FROM-channel. It must be a number.\n", + line, t); + continue; + } from_chan = atoi(t); if (from_chan < 0 || from_chan >= MAX_CHANS) { text_color_set(DW_COLOR_ERROR); @@ -2168,7 +2602,10 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[from_chan].valid) { + + // Only radio channels are valid for regenerate. + + if (p_audio_config->chan_medium[from_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: FROM-channel %d is not valid.\n", line, from_chan); @@ -2181,6 +2618,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Missing TO-channel on line %d.\n", line); continue; } + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: '%s' is not allowed for TO-channel. It must be a number.\n", + line, t); + continue; + } to_chan = atoi(t); if (to_chan < 0 || to_chan >= MAX_CHANS) { text_color_set(DW_COLOR_ERROR); @@ -2188,7 +2631,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[to_chan].valid) { + if (p_audio_config->chan_medium[to_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TO-channel %d is not valid.\n", line, to_chan); @@ -2220,6 +2663,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Missing FROM-channel on line %d.\n", line); continue; } + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: '%s' is not allowed for FROM-channel. It must be a number.\n", + line, t); + continue; + } from_chan = atoi(t); if (from_chan < 0 || from_chan >= MAX_CHANS) { text_color_set(DW_COLOR_ERROR); @@ -2227,10 +2676,17 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[from_chan].valid) { + + // For connected mode Link layer, only internal modems should be allowed. + // A network TNC probably would not provide information about channel status. + // There is discussion about this in the document called + // Why-is-9600-only-twice-as-fast-as-1200.pdf + + if (p_audio_config->chan_medium[from_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: FROM-channel %d is not valid.\n", line, from_chan); + dw_printf ("Only internal modems can be used for connected mode packet.\n"); continue; } @@ -2240,6 +2696,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Missing TO-channel on line %d.\n", line); continue; } + if ( ! alldigits(t)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: '%s' is not allowed for TO-channel. It must be a number.\n", + line, t); + continue; + } to_chan = atoi(t); if (to_chan < 0 || to_chan >= MAX_CHANS) { text_color_set(DW_COLOR_ERROR); @@ -2247,17 +2709,21 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[to_chan].valid) { + if (p_audio_config->chan_medium[to_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TO-channel %d is not valid.\n", line, to_chan); + dw_printf ("Only internal modems can be used for connected mode packet.\n"); continue; } t = split(NULL,0); if (t != NULL) { e = regcomp (&(p_cdigi_config->alias[from_chan][to_chan]), t, REG_EXTENDED|REG_NOSUB); - if (e != 0) { + if (e == 0) { + p_cdigi_config->has_alias[from_chan][to_chan] = 1; + } + else { regerror (e, &(p_cdigi_config->alias[from_chan][to_chan]), message, sizeof(message)); text_color_set(DW_COLOR_ERROR); dw_printf ("Config file: Invalid alias matching pattern on line %d:\n%s\n", @@ -2284,6 +2750,32 @@ void config_init (char *fname, struct audio_s *p_audio_config, * FILTER from-chan to-chan filter_specification_expression * FILTER from-chan IG filter_specification_expression * FILTER IG to-chan filter_specification_expression + * + * + * Note that we have three different config file filter commands: + * + * FILTER - Originally for APRS digipeating but later enhanced + * to include IGate client side. Maybe it should be + * renamed AFILTER to make it clearer after adding CFILTER. + * + * Both internal modem and NET TNC channels allowed here. + * "IG" should be used for the IGate, NOT a virtual channel + * assigned to it. + * + * CFILTER - Similar for connected moded digipeater. + * + * Only internal modems can be used because they provide + * information about radio channel status. + * A remote network TNC might not provide the necessary + * status for correct operation. + * There is discussion about this in the document called + * Why-is-9600-only-twice-as-fast-as-1200.pdf + * + * IGFILTER - APRS-IS (IGate) server side - completely different. + * I'm not happy with this name because IG sounds like IGate + * which is really the client side. More comments later. + * Maybe it should be called subscribe or something like that + * because the subscriptions are cumulative. */ else if (strcasecmp(t, "FILTER") == 0) { @@ -2298,6 +2790,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, } if (*t == 'i' || *t == 'I') { from_chan = MAX_CHANS; + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: FILTER IG ... on line %d.\n", line); + dw_printf ("Warning! Don't mess with IS>RF filtering unless you are an expert and have an unusual situation.\n"); + dw_printf ("Warning! The default is fine for nearly all situations.\n"); + dw_printf ("Warning! Be sure to read carefully and understand Successful-APRS-Gateway-Operation.pdf .\n"); + dw_printf ("Warning! If you insist, be sure to add \" | i/180 \" so you don't break messaging.\n"); } else { from_chan = isdigit(*t) ? atoi(t) : -999; @@ -2308,12 +2806,19 @@ void config_init (char *fname, struct audio_s *p_audio_config, continue; } - if ( ! p_audio_config->achan[from_chan].valid) { + if (p_audio_config->chan_medium[from_chan] != MEDIUM_RADIO && + p_audio_config->chan_medium[from_chan] != MEDIUM_NETTNC) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: FROM-channel %d is not valid.\n", line, from_chan); continue; } + if (p_audio_config->chan_medium[from_chan] == MEDIUM_IGATE) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Use 'IG' rather than %d for FROM-channel.\n", + line, from_chan); + continue; + } } t = split(NULL,0); @@ -2324,6 +2829,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, } if (*t == 'i' || *t == 'I') { to_chan = MAX_CHANS; + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: FILTER ... IG ... on line %d.\n", line); + dw_printf ("Warning! Don't mess with RF>IS filtering unless you are an expert and have an unusual situation.\n"); + dw_printf ("Warning! Expected behavior is for everything to go from RF to IS.\n"); + dw_printf ("Warning! The default is fine for nearly all situations.\n"); + dw_printf ("Warning! Be sure to read carefully and understand Successful-APRS-Gateway-Operation.pdf .\n"); } else { to_chan = isdigit(*t) ? atoi(t) : -999; @@ -2333,12 +2844,19 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[to_chan].valid) { + if (p_audio_config->chan_medium[to_chan] != MEDIUM_RADIO && + p_audio_config->chan_medium[to_chan] != MEDIUM_NETTNC) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TO-channel %d is not valid.\n", line, to_chan); continue; } + if (p_audio_config->chan_medium[to_chan] == MEDIUM_IGATE) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Use 'IG' rather than %d for TO-channel.\n", + line, to_chan); + continue; + } } t = split(NULL,1); /* Take rest of line including spaces. */ @@ -2368,6 +2886,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, /* * CFILTER from-chan to-chan filter_specification_expression + * + * Why did I put this here? + * What would be a useful use case? Perhaps block by source or destination? */ else if (strcasecmp(t, "CFILTER") == 0) { @@ -2389,7 +2910,10 @@ void config_init (char *fname, struct audio_s *p_audio_config, continue; } - if ( ! p_audio_config->achan[from_chan].valid) { + // DO NOT allow a network TNC here. + // Must be internal modem to have necessary knowledge about channel status. + + if (p_audio_config->chan_medium[from_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: FROM-channel %d is not valid.\n", line, from_chan); @@ -2410,7 +2934,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[to_chan].valid) { + if (p_audio_config->chan_medium[to_chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TO-channel %d is not valid.\n", line, to_chan); @@ -2423,7 +2947,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = " "; /* Empty means permit nothing. */ } - p_cdigi_config->filter_str[from_chan][to_chan] = strdup(t); + p_cdigi_config->cfilter_str[from_chan][to_chan] = strdup(t); //TODO1.2: Do a test run to see errors now instead of waiting. @@ -2724,7 +3248,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Line %d: Missing latitude for TTGRID command.\n", line); + dw_printf ("Line %d: Missing minimum latitude for TTGRID command.\n", line); continue; } tl->grid.lat0 = parse_ll(t,LAT,line); @@ -2734,7 +3258,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Line %d: Missing longitude for TTGRID command.\n", line); + dw_printf ("Line %d: Missing minimum longitude for TTGRID command.\n", line); continue; } tl->grid.lon0 = parse_ll(t,LON,line); @@ -2744,7 +3268,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Line %d: Missing latitude for TTGRID command.\n", line); + dw_printf ("Line %d: Missing maximum latitude for TTGRID command.\n", line); continue; } tl->grid.lat9 = parse_ll(t,LAT,line); @@ -2754,13 +3278,16 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Line %d: Missing longitude for TTGRID command.\n", line); + dw_printf ("Line %d: Missing maximum longitude for TTGRID command.\n", line); continue; } - tl->grid.lon0 = parse_ll(t,LON,line); + tl->grid.lon9 = parse_ll(t,LON,line); /* temp debugging */ + // dw_printf ("CONFIG TTGRID min %f %f\n", tl->grid.lat0, tl->grid.lon0); + // dw_printf ("CONFIG TTGRID max %f %f\n", tl->grid.lat9, tl->grid.lon9); + //for (j=0; jttloc_len; j++) { // dw_printf ("debug ttloc %d/%d %s\n", j, p_tt_config->ttloc_size, // p_tt_config->ttloc_ptr[j].pattern); @@ -3297,7 +3824,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, strlcpy(tl->pattern, "", sizeof(tl->pattern)); /* Pattern: Any combination of digits, x, y, and z. */ - /* Also make note of which letters are used in pattern and defintition. */ + /* Also make note of which letters are used in pattern and definition. */ /* Version 1.2: also allow A,B,C,D in the pattern. */ t = split(NULL,0); @@ -3567,7 +4094,11 @@ void config_init (char *fname, struct audio_s *p_audio_config, MAX_CHANS-1, line); continue; } - if ( ! p_audio_config->achan[r].valid) { + + // I suppose we need internal modem channel here. + // otherwise a DTMF decoder would not be available. + + if (p_audio_config->chan_medium[r] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TTOBJ DTMF receive channel %d is not valid.\n", line, r); @@ -3593,7 +4124,8 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file: Transmit channel must be in range of 0 to %d on line %d.\n", MAX_CHANS-1, line); x = -1; } - else if ( ! p_audio_config->achan[x].valid) { + else if (p_audio_config->chan_medium[x] != MEDIUM_RADIO && + p_audio_config->chan_medium[x] != MEDIUM_NETTNC) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: TTOBJ transmit channel %d is not valid.\n", line, x); x = -1; @@ -3919,6 +4451,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, /* * IGFILTER - IGate Server side filters. + * Is this name too confusing. Too similar to FILTER IG 0 ... + * Maybe SSFILTER suggesting Server Side. + * SUBSCRIBE might be better because it's not a filter that limits. * * IGFILTER filter-spec ... */ @@ -3927,6 +4462,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, t = split(NULL,1); /* Take rest of line as one string. */ + if (p_igate_config->t2_filter != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Warning - Earlier IGFILTER value will be replaced by this one.\n", line); + continue; + } + if (t != NULL && strlen(t) > 0) { p_igate_config->t2_filter = strdup (t); } @@ -4005,13 +4546,13 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_igate_config->igmsp = n; } else { - p_igate_config->satgate_delay = 1; + p_igate_config->igmsp = 1; text_color_set(DW_COLOR_ERROR); dw_printf ("Line %d: Unreasonable number of times for message sender position. Using default 1.\n", line); } } else { - p_igate_config->satgate_delay = 1; + p_igate_config->igmsp = 1; text_color_set(DW_COLOR_ERROR); dw_printf ("Line %d: Missing number of times for message sender position. Using default 1.\n", line); } @@ -4027,6 +4568,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, else if (strcasecmp(t, "SATGATE") == 0) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Line %d: SATGATE is pretty useless and will be removed in a future version.\n", line); + t = split(NULL,0); if (t != NULL) { @@ -4066,6 +4610,13 @@ void config_init (char *fname, struct audio_s *p_audio_config, continue; } n = atoi(t); + t = split(NULL,0); + if (t != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Unexpected \"%s\" after the port number.\n", line, t); + dw_printf ("Perhaps you were trying to use feature available only with KISSPORT.\n"); + continue; + } if ((n >= MIN_IP_PORT_NUMBER && n <= MAX_IP_PORT_NUMBER) || n == 0) { p_misc_config->agwpe_port = n; } @@ -4078,46 +4629,197 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * KISSPORT - Port number for KISS over IP. + * KISSPORT port [ chan ] - Port number for KISS over IP. */ + // Previously we allowed only a single TCP port for KISS. + // An increasing number of people want to run multiple radios. + // Unfortunately, most applications don't know how to deal with multi-radio TNCs. + // They ignore the channel on receive and always transmit to channel 0. + // Running multiple instances of direwolf is a work-around but this leads to + // more complex configuration and we lose the cross-channel digipeating capability. + // In release 1.7 we add a new feature to assign a single radio channel to a TCP port. + // e.g. + // KISSPORT 8001 # default, all channels. Radio channel = KISS channel. + // + // KISSPORT 7000 0 # Only radio channel 0 for receive. + // # Transmit to radio channel 0, ignoring KISS channel. + // + // KISSPORT 7001 1 # Only radio channel 1 for receive. KISS channel set to 0. + // # Transmit to radio channel 1, ignoring KISS channel. + +// FIXME else if (strcasecmp(t, "KISSPORT") == 0) { int n; + int tcp_port = 0; + int chan = -1; // optional. default to all if not specified. t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Line %d: Missing port number for KISSPORT command.\n", line); + dw_printf ("Line %d: Missing TCP port number for KISSPORT command.\n", line); continue; } n = atoi(t); if ((n >= MIN_IP_PORT_NUMBER && n <= MAX_IP_PORT_NUMBER) || n == 0) { - p_misc_config->kiss_port = n; + tcp_port = n; } else { - p_misc_config->kiss_port = DEFAULT_KISS_PORT; text_color_set(DW_COLOR_ERROR); - dw_printf ("Line %d: Invalid port number for KISS TCPIP Socket Interface. Using %d.\n", - line, p_misc_config->kiss_port); + dw_printf ("Line %d: Invalid TCP port number for KISS TCPIP Socket Interface.\n", line); + dw_printf ("Use something in the range of %d to %d.\n", MIN_IP_PORT_NUMBER, MAX_IP_PORT_NUMBER); + continue; } + + t = split(NULL,0); + if (t != NULL) { + chan = atoi(t); + if (chan < 0 || chan >= MAX_CHANS) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Invalid channel %d for KISSPORT command. Must be in range 0 thru %d.\n", line, chan, MAX_CHANS-1); + continue; + } + } + + // "KISSPORT 0" is used to remove the default entry. + + if (tcp_port == 0) { + p_misc_config->kiss_port[0] = 0; // Should all be wiped out? + } + else { + + // Try to find an empty slot. + // A duplicate TCP port number will overwrite the previous value. + + int slot = -1; + for (int i = 0; i < MAX_KISS_TCP_PORTS && slot == -1; i++) { + if (p_misc_config->kiss_port[i] == tcp_port) { + slot = i; + if ( ! (slot == 0 && tcp_port == DEFAULT_KISS_PORT)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Warning: Duplicate TCP port %d will overwrite previous value.\n", line, tcp_port); + } + } + else if (p_misc_config->kiss_port[i] == 0) { + slot = i; + } + } + if (slot >= 0) { + p_misc_config->kiss_port[slot] = tcp_port; + p_misc_config->kiss_chan[slot] = chan; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Too many KISSPORT commands.\n", line); + } + } + } + +/* + * NULLMODEM name [ speed ] - Device name for serial port or our end of the virtual "null modem" + * SERIALKISS name [ speed ] + * + * Version 1.5: Added SERIALKISS which is equivalent to NULLMODEM. + * The original name sort of made sense when it was used only for one end of a virtual + * null modem cable on Windows only. Now it is also available for Linux. + * TODO1.5: In retrospect, this doesn't seem like such a good name. + */ + + else if (strcasecmp(t, "NULLMODEM") == 0 || strcasecmp(t, "SERIALKISS") == 0) { + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Missing serial port name on line %d.\n", line); + continue; + } + else { + if (strlen(p_misc_config->kiss_serial_port) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Warning serial port name on line %d replaces earlier value.\n", line); + } + strlcpy (p_misc_config->kiss_serial_port, t, sizeof(p_misc_config->kiss_serial_port)); + p_misc_config->kiss_serial_speed = 0; + p_misc_config->kiss_serial_poll = 0; + } + + t = split(NULL,0); + if (t != NULL) { + p_misc_config->kiss_serial_speed = atoi(t); + } + } + +/* + * SERIALKISSPOLL name - Poll for serial port name that might come and go. + * e.g. /dev/rfcomm0 for bluetooth. + */ + + else if (strcasecmp(t, "SERIALKISSPOLL") == 0) { + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Missing serial port name on line %d.\n", line); + continue; + } + else { + if (strlen(p_misc_config->kiss_serial_port) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Warning serial port name on line %d replaces earlier value.\n", line); + } + strlcpy (p_misc_config->kiss_serial_port, t, sizeof(p_misc_config->kiss_serial_port)); + p_misc_config->kiss_serial_speed = 0; + p_misc_config->kiss_serial_poll = 1; // set polling. + } + } + + +/* + * KISSCOPY - Data from network KISS client is copied to all others. + * This does not apply to pseudo terminal KISS. + */ + + else if (strcasecmp(t, "KISSCOPY") == 0) { + p_misc_config->kiss_copy = 1; } + /* - * NULLMODEM - Device name for our end of the virtual "null modem" + * DNSSD - Enable or disable (1/0) dns-sd, DNS Service Discovery announcements + * DNSSDNAME - Set DNS-SD service name, defaults to "Dire Wolf on " */ - else if (strcasecmp(t, "nullmodem") == 0) { + + else if (strcasecmp(t, "DNSSD") == 0) { + int n; t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file: Missing device name for my end of the 'null modem' on line %d.\n", line); + dw_printf ("Line %d: Missing integer value for DNSSD command.\n", line); + continue; + } + n = atoi(t); + if (n == 0 || n == 1) { + p_misc_config->dns_sd_enabled = n; + } else { + p_misc_config->dns_sd_enabled = 0; + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Invalid integer value for DNSSD. Disabling dns-sd.\n", line); + } + } + + else if (strcasecmp(t, "DNSSDNAME") == 0) { + t = split(NULL, 1); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing service name for DNSSDNAME.\n", line); continue; } else { - strlcpy (p_misc_config->nullmodem, t, sizeof(p_misc_config->nullmodem)); + strlcpy(p_misc_config->dns_sd_name, t, sizeof(p_misc_config->dns_sd_name)); } } + + /* - * GPSNMEA - Device name for reading from GPS receiver. + * GPSNMEA serial-device [ speed ] - Direct connection to GPS receiver. */ else if (strcasecmp(t, "gpsnmea") == 0) { t = split(NULL,0); @@ -4126,8 +4828,15 @@ void config_init (char *fname, struct audio_s *p_audio_config, dw_printf ("Config file, line %d: Missing serial port name for GPS receiver.\n", line); continue; } + strlcpy (p_misc_config->gpsnmea_port, t, sizeof(p_misc_config->gpsnmea_port)); + + t = split(NULL,0); + if (t != NULL) { + int n = atoi(t); + p_misc_config->gpsnmea_speed = n; + } else { - strlcpy (p_misc_config->gpsnmea_port, t, sizeof(p_misc_config->gpsnmea_port)); + p_misc_config->gpsnmea_speed = 4800; // The standard at one time. } } @@ -4178,36 +4887,64 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * WAYPOINT - Generate WPL NMEA sentences for display on map. + * WAYPOINT - Generate WPL and AIS NMEA sentences for display on map. * * WAYPOINT serial-device [ formats ] + * WAYPOINT host:udpport [ formats ] * */ else if (strcasecmp(t, "waypoint") == 0) { + t = split(NULL,0); if (t == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file: Missing device name for WAYPOINT on line %d.\n", line); + dw_printf ("Config file: Missing output device for WAYPOINT on line %d.\n", line); continue; } + + /* If there is a ':' in the name, split it into hostname:udpportnum. */ + /* Otherwise assume it is serial port name. */ + + char *p = strchr (t, ':'); + if (p != NULL) { + *p = '\0'; + int n = atoi(p+1); + if (n >= MIN_IP_PORT_NUMBER && n <= MAX_IP_PORT_NUMBER) { + strlcpy (p_misc_config->waypoint_udp_hostname, t, sizeof(p_misc_config->waypoint_udp_hostname)); + if (strlen(p_misc_config->waypoint_udp_hostname) == 0) { + strlcpy (p_misc_config->waypoint_udp_hostname, "localhost", sizeof(p_misc_config->waypoint_udp_hostname)); + } + p_misc_config->waypoint_udp_portnum = n; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Invalid UDP port number %d for sending waypoints.\n", line, n); + } + } else { - strlcpy (p_misc_config->waypoint_port, t, sizeof(p_misc_config->waypoint_port)); + strlcpy (p_misc_config->waypoint_serial_port, t, sizeof(p_misc_config->waypoint_serial_port)); } + + /* Anything remaining is the formats to enable. */ + t = split(NULL,1); if (t != NULL) { for ( ; *t != '\0' ; t++ ) { switch (toupper(*t)) { case 'N': - p_misc_config->waypoint_formats |= WPT_FORMAT_NMEA_GENERIC; + p_misc_config->waypoint_formats |= WPL_FORMAT_NMEA_GENERIC; break; case 'G': - p_misc_config->waypoint_formats |= WPT_FORMAT_GARMIN; + p_misc_config->waypoint_formats |= WPL_FORMAT_GARMIN; break; case 'M': - p_misc_config->waypoint_formats |= WPT_FORMAT_MAGELLAN; + p_misc_config->waypoint_formats |= WPL_FORMAT_MAGELLAN; break; case 'K': - p_misc_config->waypoint_formats |= WPT_FORMAT_KENWOOD; + p_misc_config->waypoint_formats |= WPL_FORMAT_KENWOOD; + break; + case 'A': + p_misc_config->waypoint_formats |= WPL_FORMAT_AIS; break; case ' ': case ',': @@ -4222,7 +4959,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, } /* - * LOGDIR - Directory name for storing log files. Use "." for current working directory. + * LOGDIR - Directory name for automatically named daily log files. Use "." for current working directory. */ else if (strcasecmp(t, "logdir") == 0) { t = split(NULL,0); @@ -4232,7 +4969,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, continue; } else { - strlcpy (p_misc_config->logdir, t, sizeof(p_misc_config->logdir)); + if (strlen(p_misc_config->log_path) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: LOGDIR on line %d is replacing an earlier LOGDIR or LOGFILE.\n", line); + } + p_misc_config->log_daily_names = 1; + strlcpy (p_misc_config->log_path, t, sizeof(p_misc_config->log_path)); } t = split(NULL,0); if (t != NULL) { @@ -4241,6 +4983,31 @@ void config_init (char *fname, struct audio_s *p_audio_config, } } +/* + * LOGFILE - Log file name, including any directory part. + */ + else if (strcasecmp(t, "logfile") == 0) { + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Missing file name for LOGFILE on line %d.\n", line); + continue; + } + else { + if (strlen(p_misc_config->log_path) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: LOGFILE on line %d is replacing an earlier LOGDIR or LOGFILE.\n", line); + } + p_misc_config->log_daily_names = 0; + strlcpy (p_misc_config->log_path, t, sizeof(p_misc_config->log_path)); + } + t = split(NULL,0); + if (t != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: LOGFILE on line %d should have file name and nothing more.\n", line); + } + } + /* * BEACON channel delay every message * @@ -4251,7 +5018,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: Old style 'BEACON' has been replaced with new commands.\n", line); - dw_printf ("Use PBEACON, OBEACON, or CBEACON instead.\n"); + dw_printf ("Use PBEACON, OBEACON, TBEACON, or CBEACON instead.\n"); } @@ -4266,6 +5033,9 @@ void config_init (char *fname, struct audio_s *p_audio_config, * New style with keywords for options. */ +// TODO: maybe add proportional pathing so multiple beacon timing does not need to be manually constructed? +// http://www.aprs.org/newN/ProportionalPathing.txt + else if (strcasecmp(t, "PBEACON") == 0 || strcasecmp(t, "OBEACON") == 0 || strcasecmp(t, "TBEACON") == 0 || @@ -4466,6 +5236,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_misc_config->maxframe_basic = n; } else { + p_misc_config->maxframe_basic = AX25_K_MAXFRAME_BASIC_DEFAULT; text_color_set(DW_COLOR_ERROR); dw_printf ("Line %d: Invalid MAXFRAME value outside range of %d to %d. Using default %d.\n", line, AX25_K_MAXFRAME_BASIC_MIN, AX25_K_MAXFRAME_BASIC_MAX, p_misc_config->maxframe_basic); @@ -4491,6 +5262,7 @@ void config_init (char *fname, struct audio_s *p_audio_config, p_misc_config->maxframe_extended = n; } else { + p_misc_config->maxframe_extended = AX25_K_MAXFRAME_EXTENDED_DEFAULT; text_color_set(DW_COLOR_ERROR); dw_printf ("Line %d: Invalid EMAXFRAME value outside of range %d to %d. Using default %d.\n", line, AX25_K_MAXFRAME_EXTENDED_MIN, AX25_K_MAXFRAME_EXTENDED_MAX, p_misc_config->maxframe_extended); @@ -4520,6 +5292,77 @@ void config_init (char *fname, struct audio_s *p_audio_config, } +/* + * V20 address [ address ... ] - Stations known to support only AX.25 v2.0. + * When connecting to these, skip SABME and go right to SABM. + * Possible to have multiple and they are cumulative. + */ + + else if (strcasecmp(t, "V20") == 0) { + + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing address(es) for V20.\n", line); + continue; + } + + while (t != NULL) { + int const strict = 2; + char call_no_ssid[AX25_MAX_ADDR_LEN]; + int ssid, heard; + + if (ax25_parse_addr (AX25_DESTINATION, t, strict, call_no_ssid, &ssid, &heard)) { + p_misc_config->v20_addrs = (char**)realloc (p_misc_config->v20_addrs, sizeof(char*) * (p_misc_config->v20_count + 1)); + p_misc_config->v20_addrs[p_misc_config->v20_count++] = strdup(t); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Invalid station address for V20 command.\n", line); + + // continue processing any others following. + } + t = split(NULL,0); + } + } + + +/* + * NOXID address [ address ... ] - Stations known not to understand XID. + * After connecting to these (with v2.2 obviously), don't try using XID command. + * AX.25 for Linux is the one known case so far. + * Possible to have multiple and they are cumulative. + */ + + else if (strcasecmp(t, "NOXID") == 0) { + + t = split(NULL,0); + if (t == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Missing address(es) for NOXID.\n", line); + continue; + } + + while (t != NULL) { + int const strict = 2; + char call_no_ssid[AX25_MAX_ADDR_LEN]; + int ssid, heard; + + if (ax25_parse_addr (AX25_DESTINATION, t, strict, call_no_ssid, &ssid, &heard)) { + p_misc_config->noxid_addrs = (char**)realloc (p_misc_config->noxid_addrs, sizeof(char*) * (p_misc_config->noxid_count + 1)); + p_misc_config->noxid_addrs[p_misc_config->noxid_count++] = strdup(t); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Line %d: Invalid station address for NOXID command.\n", line); + + // continue processing any others following. + } + t = split(NULL,0); + } + } + + /* * Invalid command. */ @@ -4615,7 +5458,10 @@ void config_init (char *fname, struct audio_s *p_audio_config, } } - if (p_audio_config->achan[i].valid && strlen(p_igate_config->t2_login) > 0) { +/* When IGate is enabled, all radio channels must have a callsign associated. */ + + if (strlen(p_igate_config->t2_login) > 0 && + (p_audio_config->chan_medium[i] == MEDIUM_RADIO || p_audio_config->chan_medium[i] == MEDIUM_NETTNC)) { if (strcmp(p_audio_config->achan[i].mycall, "NOCALL") == 0 || strcmp(p_audio_config->achan[i].mycall, "N0CALL") == 0) { text_color_set(DW_COLOR_ERROR); @@ -4639,10 +5485,12 @@ void config_init (char *fname, struct audio_s *p_audio_config, // Apply default IS>RF IGate filter if none specified. New in 1.4. // This will handle eventual case of multiple transmit channels. - for (j=0; jachan[j].valid && strlen(p_igate_config->t2_login) > 0) { - if (p_digi_config->filter_str[MAX_CHANS][j] == NULL) { - p_digi_config->filter_str[MAX_CHANS][j] = strdup("i/30"); + if (strlen(p_igate_config->t2_login) > 0) { + for (j=0; jchan_medium[j] == MEDIUM_RADIO || p_audio_config->chan_medium[j] == MEDIUM_NETTNC) { + if (p_digi_config->filter_str[MAX_CHANS][j] == NULL) { + p_digi_config->filter_str[MAX_CHANS][j] = strdup("i/180"); + } } } } @@ -4661,6 +5509,10 @@ void config_init (char *fname, struct audio_s *p_audio_config, * Returns 1 for success, 0 for serious error. */ +// FIXME: provide error messages when non applicable option is used for particular beacon type. +// e.g. IBEACON DELAY=1 EVERY=1 SENDTO=IG OVERLAY=R SYMBOL="igate" LAT=37^44.46N LONG=122^27.19W COMMENT="N1KOL-1 IGATE" +// Just ignores overlay, symbol, lat, long, and comment. + static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_s *p_audio_config) { char *t; @@ -4676,22 +5528,26 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ b->sendto_type = SENDTO_XMIT; b->sendto_chan = 0; b->delay = 60; + b->slot = G_UNKNOWN; b->every = 600; //b->delay = 6; // temp test. //b->every = 3600; b->lat = G_UNKNOWN; b->lon = G_UNKNOWN; + b->ambiguity = 0; b->alt_m = G_UNKNOWN; b->symtab = '/'; b->symbol = '-'; /* house */ b->freq = G_UNKNOWN; b->tone = G_UNKNOWN; b->offset = G_UNKNOWN; + b->source = NULL; + b->dest = NULL; while ((t = split(NULL,0)) != NULL) { char keyword[20]; - char value[200]; + char value[1000]; char *e; char *p; @@ -4706,9 +5562,56 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ strlcpy (keyword, t, sizeof(keyword)); strlcpy (value, e+1, sizeof(value)); +// QUICK TEMP EXPERIMENT, maybe permanent new feature. +// Recognize \xnn as hexadecimal value. Handy for UTF-8 in comment. +// Maybe recognize the <0xnn> form that we print. +// +// # Convert between languages here: https://translate.google.com/ then +// # Convert to UTF-8 bytes here: https://codebeautify.org/utf8-converter +// +// pbeacon delay=0:05 every=0:30 sendto=R0 lat=12.5N long=69.97W comment="\xe3\x82\xa2\xe3\x83\x9e\xe3\x83\x81\xe3\x83\xa5\xe3\x82\xa2\xe7\x84\xa1\xe7\xb7\x9a \xce\xa1\xce\xb1\xce\xb4\xce\xb9\xce\xbf\xce\xb5\xcf\x81\xce\xb1\xcf\x83\xce\xb9\xcf\x84\xce\xb5\xcf\x87\xce\xbd\xce\xb9\xcf\x83\xce\xbc\xcf\x8c\xcf\x82" + + char temp[256]; + int tlen = 0; + + for (char *p = value; *p != '\0'; ) { + if (p[0] == '\\' && p[1] == 'x' && strlen(p) >= 4 && isxdigit(p[2]) && isxdigit(p[3])) { + int n = 0; + for (int i = 2; i < 4; i++) { + n = n * 16; + if (islower(p[i])) { + n += p[i] - 'a' + 10; + } + else if (isupper(p[i])) { + n += p[i] - 'A' + 10; + } + else { // must be digit due to isxdigit test above. + n += p[i] - '0'; + } + } + temp[tlen++] = n; + p += 4; + } + else { + temp[tlen++] = *p++; + } + } + temp[tlen] = '\0'; + strlcpy (value, temp, sizeof(value)); + +// end if (strcasecmp(keyword, "DELAY") == 0) { b->delay = parse_interval(value,line); } + else if (strcasecmp(keyword, "SLOT") == 0) { + int n = parse_interval(value,line); + if ( n < 1 || n > 3600) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Beacon time slot, %d, must be in range of 1 to 3600 seconds.\n", line, n); + continue; + } + b->slot = n; + } else if (strcasecmp(keyword, "EVERY") == 0) { b->every = parse_interval(value,line); } @@ -4719,9 +5622,10 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ } else if (value[0] == 'r' || value[0] == 'R') { int n = atoi(value+1); - if ( n < 0 || n >= MAX_CHANS || ! p_audio_config->achan[n].valid) { + if (( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE) + && p_audio_config->chan_medium[n] != MEDIUM_IGATE) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, n); + dw_printf ("Config file, line %d: Simulated receive on channel %d is not valid.\n", line, n); continue; } b->sendto_type = SENDTO_RECV; @@ -4729,7 +5633,8 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ } else if (value[0] == 't' || value[0] == 'T' || value[0] == 'x' || value[0] == 'X') { int n = atoi(value+1); - if ( n < 0 || n >= MAX_CHANS || ! p_audio_config->achan[n].valid) { + if (( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE) + && p_audio_config->chan_medium[n] != MEDIUM_IGATE) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, n); continue; @@ -4740,7 +5645,8 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ } else { int n = atoi(value); - if ( n < 0 || n >= MAX_CHANS || ! p_audio_config->achan[n].valid) { + if (( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE) + && p_audio_config->chan_medium[n] != MEDIUM_IGATE) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, n); continue; @@ -4749,6 +5655,17 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ b->sendto_chan = n; } } + else if (strcasecmp(keyword, "SOURCE") == 0) { + b->source = strdup(value); + for (p = b->source; *p != '\0'; p++) { + if (islower(*p)) { + *p = toupper(*p); /* silently force upper case. */ + } + } + if (strlen(b->source) > 9) { + b->source[9] = '\0'; + } + } else if (strcasecmp(keyword, "DEST") == 0) { b->dest = strdup(value); for (p = b->dest; *p != '\0'; p++) { @@ -4798,8 +5715,40 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ else if (strcasecmp(keyword, "LONG") == 0 || strcasecmp(keyword, "LON") == 0) { b->lon = parse_ll (value, LON, line); } + else if (strcasecmp(keyword, "AMBIGUITY") == 0 || strcasecmp(keyword, "AMBIG") == 0) { + int n = atoi(value); + if (n >= 0 && n <= 4) { + b->ambiguity = n; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: Location ambiguity, on line %d, must be in range of 0 to 4.\n", line); + } + } else if (strcasecmp(keyword, "ALT") == 0 || strcasecmp(keyword, "ALTITUDE") == 0) { - b->alt_m = atof(value); + + char *unit = strpbrk(value, "abcedfghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"); + if (unit != NULL) { + float meters = 0; + for (int j=0; jalt_m = atof(value); + } + else { + // valid unit + b->alt_m = atof(value) * meters; + } + } else { + // no unit specified + b->alt_m = atof(value); + } } else if (strcasecmp(keyword, "ZONE") == 0) { strlcpy(zone, value, sizeof(zone)); @@ -4826,8 +5775,9 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ else if (strcasecmp(keyword, "POWER") == 0) { b->power = atoi(value); } - else if (strcasecmp(keyword, "HEIGHT") == 0) { + else if (strcasecmp(keyword, "HEIGHT") == 0) { // This is in feet. b->height = atoi(value); + // TODO: ability to add units suffix, e.g. 10m } else if (strcasecmp(keyword, "GAIN") == 0) { b->gain = atoi(value); @@ -4865,11 +5815,17 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ if (b->custom_info != NULL && b->custom_infocmd != NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file, line %d: Can't use both INFO and INFOCMD at the same time..\n", line); + dw_printf ("Config file, line %d: Can't use both INFO and INFOCMD at the same time.\n", line); + } + + if (b->compress && b->ambiguity != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file, line %d: Position ambiguity can't be used with compressed location format.\n", line); + b->ambiguity = 0; } /* - * Convert UTM coordintes to lat / long. + * Convert UTM coordinates to lat / long. */ if (strlen(zone) > 0 || easting != G_UNKNOWN || northing != G_UNKNOWN) { @@ -4898,12 +5854,16 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ } else { text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file, line %d: When any of ZONE, EASTING, NORTHING specifed, they must all be specified.\n", line); + dw_printf ("Config file, line %d: When any of ZONE, EASTING, NORTHING specified, they must all be specified.\n", line); } } /* * Process symbol now that we have any later overlay. + * + * FIXME: Someone who used this was surprised to end up with Solar Powser (S-). + * overlay=S symbol="/-" + * We should complain if overlay used with symtab other than \. */ if (strlen(temp_symbol) > 0) { @@ -4936,19 +5896,32 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_ if (b->sendto_type == SENDTO_XMIT) { - if ( b->sendto_chan < 0 || b->sendto_chan >= MAX_CHANS || ! p_audio_config->achan[b->sendto_chan].valid) { + if (( b->sendto_chan < 0 || b->sendto_chan >= MAX_CHANS || p_audio_config->chan_medium[b->sendto_chan] == MEDIUM_NONE) + && p_audio_config->chan_medium[b->sendto_chan] != MEDIUM_IGATE) { text_color_set(DW_COLOR_ERROR); dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, b->sendto_chan); return (0); } - if ( strcmp(p_audio_config->achan[b->sendto_chan].mycall, "") == 0 || - strcmp(p_audio_config->achan[b->sendto_chan].mycall, "NOCALL") == 0 || - strcmp(p_audio_config->achan[b->sendto_chan].mycall, "N0CALL") == 0 ) { + if (p_audio_config->chan_medium[b->sendto_chan] == MEDIUM_IGATE) { // Prevent subscript out of bounds. + // Will be using call from chan 0 later. + if ( strcmp(p_audio_config->achan[0].mycall, "") == 0 || + strcmp(p_audio_config->achan[0].mycall, "NOCALL") == 0 || + strcmp(p_audio_config->achan[0].mycall, "N0CALL") == 0 ) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Config file: MYCALL must be set for channel %d before beaconing is allowed.\n", b->sendto_chan); - return (0); + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: MYCALL must be set for channel %d before beaconing is allowed.\n", 0); + return (0); + } + } else { + if ( strcmp(p_audio_config->achan[b->sendto_chan].mycall, "") == 0 || + strcmp(p_audio_config->achan[b->sendto_chan].mycall, "NOCALL") == 0 || + strcmp(p_audio_config->achan[b->sendto_chan].mycall, "N0CALL") == 0 ) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Config file: MYCALL must be set for channel %d before beaconing is allowed.\n", b->sendto_chan); + return (0); + } } } diff --git a/config.h b/src/config.h similarity index 60% rename from config.h rename to src/config.h index c895e290..360ac492 100644 --- a/config.h +++ b/src/config.h @@ -30,21 +30,54 @@ enum sendto_type_e { SENDTO_XMIT, SENDTO_IGATE, SENDTO_RECV }; #define MAX_BEACONS 30 +#define MAX_KISS_TCP_PORTS (MAX_CHANS+1) struct misc_config_s { - int agwpe_port; /* Port number for the "AGW TCPIP Socket Interface" */ - int kiss_port; /* Port number for the "KISS" protocol. */ + int agwpe_port; /* TCP Port number for the "AGW TCPIP Socket Interface" */ + + // Previously we allowed only a single TCP port for KISS. + // An increasing number of people want to run multiple radios. + // Unfortunately, most applications don't know how to deal with multi-radio TNCs. + // They ignore the channel on receive and always transmit to channel 0. + // Running multiple instances of direwolf is a work-around but this leads to + // more complex configuration and we lose the cross-channel digipeating capability. + // In release 1.7 we add a new feature to assign a single radio channel to a TCP port. + // e.g. + // KISSPORT 8001 # default, all channels. Radio channel = KISS channel. + // + // KISSPORT 7000 0 # Only radio channel 0 for receive. + // # Transmit to radio channel 0, ignoring KISS channel. + // + // KISSPORT 7001 1 # Only radio channel 1 for receive. KISS channel set to 0. + // # Transmit to radio channel 1, ignoring KISS channel. + + int kiss_port[MAX_KISS_TCP_PORTS]; /* TCP Port number for the "TCP KISS" protocol. */ + int kiss_chan[MAX_KISS_TCP_PORTS]; /* Radio Channel number for this port or -1 for all. */ + + int kiss_copy; /* Data from network KISS client is copied to all others. */ int enable_kiss_pt; /* Enable pseudo terminal for KISS. */ /* Want this to be off by default because it hangs */ /* after a while if nothing is reading from other end. */ - char nullmodem[20]; /* Serial port name for our end of the */ + char kiss_serial_port[20]; + /* Serial port name for our end of the */ /* virtual null modem for native Windows apps. */ + /* Version 1.5 add same capability for Linux. */ + + int kiss_serial_speed; /* Speed, in bps, for the KISS serial port. */ + /* If 0, just leave what was already there. */ + + int kiss_serial_poll; /* When using Bluetooth KISS, the /dev/rfcomm0 device */ + /* will appear and disappear as the remote application */ + /* opens and closes the virtual COM port. */ + /* When this is non-zero, we will check periodically to */ + /* see if the device has appeared and we will open it. */ char gpsnmea_port[20]; /* Serial port name for reading NMEA sentences from GPS. */ /* e.g. COM22, /dev/ttyACM0 */ - /* Currently no option for setting non-standard speed. */ + + int gpsnmea_speed; /* Speed for above, baud, default 4800. */ char gpsd_host[20]; /* Host for gpsd server. */ /* e.g. localhost, 192.168.1.2 */ @@ -53,20 +86,31 @@ struct misc_config_s { /* Default is 2947. */ - char waypoint_port[20]; /* Serial port name for sending NMEA waypoint sentences */ + char waypoint_serial_port[20]; /* Serial port name for sending NMEA waypoint sentences */ /* to a GPS map display or other mapping application. */ /* e.g. COM22, /dev/ttyACM0 */ /* Currently no option for setting non-standard speed. */ + /* This was done in 2014 and no one has complained yet. */ + + char waypoint_udp_hostname[80]; /* Destination host when using UDP. */ + + int waypoint_udp_portnum; /* UDP port. */ int waypoint_formats; /* Which sentence formats should be generated? */ -#define WPT_FORMAT_NMEA_GENERIC 0x01 /* N $GPWPT */ -#define WPT_FORMAT_GARMIN 0x02 /* G $PGRMW */ -#define WPT_FORMAT_MAGELLAN 0x04 /* M $PMGNWPL */ -#define WPT_FORMAT_KENWOOD 0x08 /* K $PKWDWPL */ +#define WPL_FORMAT_NMEA_GENERIC 0x01 /* N $GPWPL */ +#define WPL_FORMAT_GARMIN 0x02 /* G $PGRMW */ +#define WPL_FORMAT_MAGELLAN 0x04 /* M $PMGNWPL */ +#define WPL_FORMAT_KENWOOD 0x08 /* K $PKWDWPL */ +#define WPL_FORMAT_AIS 0x10 /* A !AIVDM */ + + int log_daily_names; /* True to generate new log file each day. */ - char logdir[80]; /* Directory for saving activity logs. */ + char log_path[80]; /* Either directory or full file name depending on above. */ + + int dns_sd_enabled; /* DNS Service Discovery announcement enabled. */ + char dns_sd_name[64]; /* Name announced on dns-sd; defaults to "Dire Wolf on " */ int sb_configured; /* TRUE if SmartBeaconing is configured. */ int sb_fast_speed; /* MPH */ @@ -93,6 +137,17 @@ struct misc_config_s { /* switching to SABM. This is to handle the case of an old */ /* TNC which simply ignores SABME rather than replying with FRMR. */ + char **v20_addrs; /* Stations known to understand only AX.25 v2.0 so we don't */ + /* waste time trying v2.2 first. */ + + int v20_count; /* Number of station addresses in array above. */ + + char **noxid_addrs; /* Stations known not to understand XID command so don't */ + /* waste time sending it and eventually giving up. */ + /* AX.25 for Linux is the one known case, so far, where */ + /* SABME is implemented but XID is not. */ + + int noxid_count; /* Number of station addresses in array above. */ // Beacons. @@ -120,12 +175,18 @@ struct misc_config_s { int delay; /* Seconds to delay before first transmission. */ + int slot; /* Seconds after hour for slotted time beacons. */ + /* If specified, it overrides any 'delay' value. */ + int every; /* Time between transmissions, seconds. */ /* Remains fixed for PBEACON and OBEACON. */ /* Dynamically adjusted for TBEACON. */ time_t next; /* Unix time to transmit next one. */ + char *source; /* NULL or explicit AX.25 source address to use */ + /* instead of the mycall value for the channel. */ + char *dest; /* NULL or explicit AX.25 destination to use */ /* instead of the software version such as APDW11. */ @@ -146,13 +207,14 @@ struct misc_config_s { double lat; /* Latitude and longitude. */ double lon; + int ambiguity; /* Number of lower digits to trim from location. 0 (default), 1, 2, 3, 4. */ float alt_m; /* Altitude in meters. */ char symtab; /* Symbol table: / or \ or overlay character. */ char symbol; /* Symbol code. */ float power; /* For PHG. */ - float height; + float height; /* HAAT in feet */ float gain; /* Original protocol spec was unclear. */ /* Addendum 1.1 clarifies it is dBi not dBd. */ diff --git a/decode_aprs.c b/src/decode_aprs.c similarity index 75% rename from decode_aprs.c rename to src/decode_aprs.c index 00f8e80d..ab933278 100644 --- a/decode_aprs.c +++ b/src/decode_aprs.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2014, 2015 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2017, 2022 John Langner, WB2OSZ // // 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 @@ -17,6 +17,7 @@ // along with this program. If not, see . // +// TODO: Better error messages for examples here: http://lists.tapr.org/pipermail/aprssig_lists.tapr.org/2023-July/date.html /*------------------------------------------------------------------ * @@ -26,7 +27,7 @@ * * Description: Present the packet contents in human readable format. * This is a fairly complete implementation with error messages - * pointing out various specication violations. + * pointing out various specification violations. * * Assumptions: ax25_from_frame() has been called to * separate the header and information. @@ -54,6 +55,7 @@ #include "dwgpsnmea.h" #include "decode_aprs.h" #include "telemetry.h" +#include "ais.h" #define TRUE 1 @@ -109,12 +111,12 @@ static void aprs_status_report (decode_aprs_t *A, char *, int); static void aprs_general_query (decode_aprs_t *A, char *, int, int quiet); static void aprs_directed_station_query (decode_aprs_t *A, char *addressee, char *query, int quiet); static void aprs_telemetry (decode_aprs_t *A, char *info, int info_len, int quiet); +static void aprs_user_defined (decode_aprs_t *A, char *, int); static void aprs_raw_touch_tone (decode_aprs_t *A, char *, int); static void aprs_morse_code (decode_aprs_t *A, char *, int); static void aprs_positionless_weather_report (decode_aprs_t *A, unsigned char *, int); static void weather_data (decode_aprs_t *A, char *wdata, int wind_prefix); static void aprs_ultimeter (decode_aprs_t *A, char *, int); -static void third_party_header (decode_aprs_t *A, char *, int); static void decode_position (decode_aprs_t *A, position_t *ppos); static void decode_compressed_position (decode_aprs_t *A, compressed_position_t *ppos); static double get_latitude_8 (char *p, int quiet); @@ -139,6 +141,12 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen); * * quiet - Suppress error messages. * + * third_party_src - Specify when parsing a third party header. + * (decode_aprs is called recursively.) + * This is mostly found when an IGate transmits a message + * that came via APRS-IS. + * NULL when not third party payload. + * * Outputs: A-> g_symbol_table, g_symbol_code, * g_lat, g_lon, * g_speed_mph, g_course, g_altitude_ft, @@ -150,21 +158,30 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen); * *------------------------------------------------------------------*/ -void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) +void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet, char *third_party_src) { + //dw_printf ("DEBUG decode_aprs quiet=%d, third_party=%p\n", quiet, third_party_src); - char dest[AX25_MAX_ADDR_LEN]; unsigned char *pinfo; int info_len; info_len = ax25_get_info (pp, &pinfo); - memset (A, 0, sizeof (*A)); + //dw_printf ("DEBUG decode_aprs info=\"%s\"\n", pinfo); + + if (third_party_src == NULL) { + memset (A, 0, sizeof (*A)); + } A->g_quiet = quiet; - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "Unknown APRS Data Type Indicator \"%c\"", *pinfo); + if (isprint(*pinfo)) { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "ERROR!!! Unknown APRS Data Type Indicator \"%c\"", *pinfo); + } + else { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "ERROR!!! Unknown APRS Data Type Indicator: unprintable 0x%02x", *pinfo); + } A->g_symbol_table = '/'; /* Default to primary table. */ A->g_symbol_code = ' '; /* What should we have for default symbol? */ @@ -190,14 +207,61 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) A->g_footprint_lon = G_UNKNOWN; A->g_footprint_radius = G_UNKNOWN; +// TODO: Complain if obsolete WIDE or RELAY is found in via path. + +// TODO: complain if unused WIDEn is see in path. +// There is a report of UIDIGI decrementing ssid 1 to 0 and not marking it used. +// http://lists.tapr.org/pipermail/aprssig_lists.tapr.org/2022-May/049397.html + +// TODO: Complain if used digi is found after unused. Should never happen. + + +// If third-party header, try to decode just the payload. + + if (*pinfo == '}') { + + //dw_printf ("DEBUG decode_aprs recursively process third party header\n"); + // This must not be strict because the addresses in third party payload doesn't + // need to adhere to the AX.25 address format (i.e. 6 upper case alphanumeric.) + // SSID can be 2 alphanumeric characters. + // Addresses can include lower case, e.g. q construct. + + // e.g. WR2X-2>APRS,WA1PLE-13*:} + // K1BOS-B>APOSB,TCPIP,WR2X-2*:@122015z4221.42ND07111.93W&/A=000000SharkRF openSPOT3 MMDVM446.025 MA/SW + + packet_t pp_payload = ax25_from_text ((char*)pinfo+1, 0); + if (pp_payload != NULL) { + char payload_src[AX25_MAX_ADDR_LEN]; + memset(payload_src, 0, sizeof(payload_src)); + memcpy(payload_src, (char*)pinfo+1, sizeof(payload_src)-1); + char *q = strchr(payload_src, '>'); + if (q != NULL) *q = '\0'; + A->g_has_thirdparty_header = 1; + decode_aprs (A, pp_payload, quiet, payload_src); // 1 means used recursively + ax25_delete (pp_payload); + return; + } + else { + strlcpy (A->g_data_type_desc, "Third Party Header: Unable to parse payload.", sizeof(A->g_data_type_desc)); + ax25_get_addr_with_ssid (pp, AX25_SOURCE, A->g_src); + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, A->g_dest); + } + } /* * Extract source and destination including the SSID. */ - - ax25_get_addr_with_ssid (pp, AX25_SOURCE, A->g_src); - ax25_get_addr_with_ssid (pp, AX25_DESTINATION, dest); + if (third_party_src != NULL) { + strlcpy (A->g_src, third_party_src, sizeof(A->g_src)); + } + else { + ax25_get_addr_with_ssid (pp, AX25_SOURCE, A->g_src); + } + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, A->g_dest); + + //dw_printf ("DEBUG decode_aprs source=%s, dest=%s\n", A->g_src, A->g_dest); + /* * Report error if the information part contains a nul character. @@ -219,14 +283,30 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) if ( ( ! A->g_quiet ) && ( (int)strlen((char*)pinfo) != info_len) ) { text_color_set(DW_COLOR_ERROR); - dw_printf("'nul' character found in Information part. This should never happen.\n"); - dw_printf("It seems that %s is transmitting with defective software.\n", A->g_src); + dw_printf("'nul' character found in Information part. This should never happen with APRS.\n"); + dw_printf("If this is meant to be APRS, %s is transmitting with defective software.\n", A->g_src); if (strcmp((char*)pinfo, "4P") == 0) { dw_printf("The TM-D710 will do this intermittently. A firmware upgrade is needed to fix it.\n"); } } +/* + * Application might be in the destination field for most message types. + * MIC-E format has part of location in the destination field. + */ + + switch (*pinfo) { /* "DTI" data type identifier. */ + + case '\'': /* Old Mic-E Data */ + case '`': /* Current Mic-E Data */ + break; + + default: + decode_tocall (A, A->g_dest); + break; + } + switch (*pinfo) { /* "DTI" data type identifier. */ case '!': /* Position without timestamp (no APRS messaging). */ @@ -236,16 +316,17 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) if (strncmp((char*)pinfo, "!!", 2) == 0) { - aprs_ultimeter (A, (char*)pinfo, info_len); + aprs_ultimeter (A, (char*)pinfo, info_len); // TODO: produce obsolete error. } else { aprs_ll_pos (A, pinfo, info_len); } + A->g_packet_type = packet_type_position; break; - //case '#': /* Peet Bros U-II Weather station */ + //case '#': /* Peet Bros U-II Weather station */ // TODO: produce obsolete error. //case '*': /* Peet Bros U-II Weather station */ //break; @@ -253,11 +334,13 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) if (strncmp((char*)pinfo, "$ULTW", 5) == 0) { - aprs_ultimeter (A, (char*)pinfo, info_len); + aprs_ultimeter (A, (char*)pinfo, info_len); // TODO: produce obsolete error. + A->g_packet_type = packet_type_weather; } else { aprs_raw_nmea (A, pinfo, info_len); + A->g_packet_type = packet_type_position; } break; @@ -265,70 +348,98 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) case '`': /* Current Mic-E Data (not used in TM-D700) */ aprs_mic_e (A, pp, pinfo, info_len); + A->g_packet_type = packet_type_position; break; case ')': /* Item. */ aprs_item (A, pinfo, info_len); + A->g_packet_type = packet_type_item; break; case '/': /* Position with timestamp (no APRS messaging) */ case '@': /* Position with timestamp (with APRS messaging) */ aprs_ll_pos_time (A, pinfo, info_len); + A->g_packet_type = packet_type_position; break; - case ':': /* Message: for one person, a group, or a bulletin. */ + case ':': /* "Message" (special APRS meaning): for one person, a group, or a bulletin. */ /* Directed Station Query */ /* Telemetry metadata. */ aprs_message (A, pinfo, info_len, quiet); + + switch (A->g_message_subtype) { + case message_subtype_message: + case message_subtype_ack: + case message_subtype_rej: + A->g_packet_type = packet_type_message; + break; + + case message_subtype_nws: + A->g_packet_type = packet_type_nws; + break; + + case message_subtype_bulletin: + default: + break; + + case message_subtype_telem_parm: + case message_subtype_telem_unit: + case message_subtype_telem_eqns: + case message_subtype_telem_bits: + A->g_packet_type = packet_type_telemetry; + break; + + case message_subtype_directed_query: + A->g_packet_type = packet_type_query; + break; + } break; case ';': /* Object */ aprs_object (A, pinfo, info_len); + A->g_packet_type = packet_type_object; break; case '<': /* Station Capabilities */ aprs_station_capabilities (A, (char*)pinfo, info_len); + A->g_packet_type = packet_type_capabilities; break; case '>': /* Status Report */ aprs_status_report (A, (char*)pinfo, info_len); + A->g_packet_type = packet_type_status; break; case '?': /* General Query */ aprs_general_query (A, (char*)pinfo, info_len, quiet); + A->g_packet_type = packet_type_query; break; case 'T': /* Telemetry */ aprs_telemetry (A, (char*)pinfo, info_len, quiet); + A->g_packet_type = packet_type_telemetry; break; case '_': /* Positionless Weather Report */ aprs_positionless_weather_report (A, pinfo, info_len); + A->g_packet_type = packet_type_weather; break; case '{': /* user defined data */ - /* http://www.aprs.org/aprs11/expfmts.txt */ - if (strncmp((char*)pinfo, "{tt", 3) == 0) { - aprs_raw_touch_tone (A, (char*)pinfo, info_len); - } - else if (strncmp((char*)pinfo, "{mc", 3) == 0) { - aprs_morse_code (A, (char*)pinfo, info_len); - } - else { - //aprs_user_defined (A, pinfo, info_len); - } + aprs_user_defined (A, (char*)pinfo, info_len); + A->g_packet_type = packet_type_userdefined; break; case 't': /* Raw touch tone data - NOT PART OF STANDARD */ @@ -337,6 +448,7 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) /* Might move into user defined data, above. */ aprs_raw_touch_tone (A, (char*)pinfo, info_len); + // no packet type for t/ filter break; case 'm': /* Morse Code data - NOT PART OF STANDARD */ @@ -346,13 +458,12 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) /* Might move into user defined data, above. */ aprs_morse_code (A, (char*)pinfo, info_len); + // no packet type for t/ filter break; - case '}': /* third party header */ - - third_party_header (A, (char*)pinfo, info_len); - break; + //case '}': /* third party header */ + // was already caught earlier. //case '\r': /* CR or LF? */ //case '\n': @@ -366,28 +477,27 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) /* - * Look in other locations if not found in information field. + * Priority order for determining the symbol is: + * - Information part, where appropriate. Already done above. + * - Destination field starting with GPS, SPC, or SYM. + * - Source SSID - Confusing to most people. Even I forgot about it when + * someone questioned where the symbol came from. It's in the APRS + * protocol spec, end of Chapter 20. */ if (A->g_symbol_table == ' ' || A->g_symbol_code == ' ') { - symbols_from_dest_or_src (*pinfo, A->g_src, dest, &A->g_symbol_table, &A->g_symbol_code); - } - -/* - * Application might be in the destination field for most message types. - * MIC-E format has part of location in the destination field. - */ + // A symbol on a "message" makes no sense and confuses people. + // Third party too. Set from the payload. + // Maybe eliminate for a couple others. - switch (*pinfo) { /* "DTI" data type identifier. */ + //dw_printf ("DEBUG decode_aprs@end1 third_party=%d, symbol_table=%c, symbol_code=%c, *pinfo=%c\n", third_party, A->g_symbol_table, A->g_symbol_code, *pinfo); - case '\'': /* Old Mic-E Data */ - case '`': /* Current Mic-E Data */ - break; + if (*pinfo != ':' && *pinfo != '}') { + symbols_from_dest_or_src (*pinfo, A->g_src, A->g_dest, &A->g_symbol_table, &A->g_symbol_code); + } - default: - decode_tocall (A, dest); - break; + //dw_printf ("DEBUG decode_aprs@end2 third_party=%d, symbol_table=%c, symbol_code=%c, *pinfo=%c\n", third_party, A->g_symbol_table, A->g_symbol_code, *pinfo); } } /* end decode_aprs */ @@ -395,8 +505,7 @@ void decode_aprs (decode_aprs_t *A, packet_t pp, int quiet) void decode_aprs_print (decode_aprs_t *A) { - char stemp[200]; - //char tmp2[2]; + char stemp[500]; double absll; char news; int deg; @@ -404,18 +513,19 @@ void decode_aprs_print (decode_aprs_t *A) { char s_lat[30]; char s_lon[30]; int n; - char symbol_description[100]; /* * First line has: - * - message type + * - packet type * - object name * - symbol * - manufacturer/application * - mic-e status * - power/height/gain, range */ - strlcpy (stemp, A->g_msg_type, sizeof(stemp)); + strlcpy (stemp, A->g_data_type_desc, sizeof(stemp)); + + //dw_printf ("DEBUG decode_aprs_print stemp1=%s\n", stemp); if (strlen(A->g_name) > 0) { strlcat (stemp, ", \"", sizeof(stemp)); @@ -423,32 +533,59 @@ void decode_aprs_print (decode_aprs_t *A) { strlcat (stemp, "\"", sizeof(stemp)); } + //dw_printf ("DEBUG decode_aprs_print stemp2=%s\n", stemp); + + //dw_printf ("DEBUG decode_aprs_print symbol_code=%c=0x%02x\n", A->g_symbol_code, A->g_symbol_code); + if (A->g_symbol_code != ' ') { - symbols_get_description (A->g_symbol_table, A->g_symbol_code, symbol_description, sizeof(symbol_description)); + char symbol_description[100]; + symbols_get_description (A->g_symbol_table, A->g_symbol_code, symbol_description, sizeof(symbol_description)); + + //dw_printf ("DEBUG decode_aprs_print symbol_description_description=%s\n", symbol_description); + strlcat (stemp, ", ", sizeof(stemp)); strlcat (stemp, symbol_description, sizeof(stemp)); } + //dw_printf ("DEBUG decode_aprs_print stemp3=%s mfr=%s\n", stemp, A->g_mfr); + if (strlen(A->g_mfr) > 0) { - strlcat (stemp, ", ", sizeof(stemp)); - strlcat (stemp, A->g_mfr, sizeof(stemp)); + if (strcmp(A->g_dest, "APRS") == 0 || + strcmp(A->g_dest, "BEACON") == 0 || + strcmp(A->g_dest, "ID") == 0) { + strlcat (stemp, "\nUse of \"", sizeof(stemp)); + strlcat (stemp, A->g_dest, sizeof(stemp)); + strlcat (stemp, "\" in the destination field is obsolete.", sizeof(stemp)); + strlcat (stemp, " You can help to improve the quality of APRS signals.", sizeof(stemp)); + strlcat (stemp, "\nTell the sender (", sizeof(stemp)); + strlcat (stemp, A->g_src, sizeof(stemp)); + strlcat (stemp, ") to use the proper product identifier from", sizeof(stemp)); + strlcat (stemp, " https://github.com/aprsorg/aprs-deviceid ", sizeof(stemp)); + } + else { + strlcat (stemp, ", ", sizeof(stemp)); + strlcat (stemp, A->g_mfr, sizeof(stemp)); + } } + //dw_printf ("DEBUG decode_aprs_print stemp4=%s\n", stemp); + if (strlen(A->g_mic_e_status) > 0) { strlcat (stemp, ", ", sizeof(stemp)); strlcat (stemp, A->g_mic_e_status, sizeof(stemp)); } + //dw_printf ("DEBUG decode_aprs_print stemp5=%s\n", stemp); if (A->g_power > 0) { char phg[100]; - /* Protcol spec doesn't mention whether this is dBd or dBi. */ + /* Protocol spec doesn't mention whether this is dBd or dBi. */ /* Clarified later. */ /* http://eng.usna.navy.mil/~bruninga/aprs/aprs11.html */ /* "The Antenna Gain in the PHG format on page 28 is in dBi." */ - snprintf (phg, sizeof(phg), ", %d W height=%d %ddBi %s", A->g_power, A->g_height, A->g_gain, A->g_directivity); + snprintf (phg, sizeof(phg), ", %d W height(HAAT)=%dft=%.0fm %ddBi %s", A->g_power, A->g_height, DW_FEET_TO_METERS(A->g_height), A->g_gain, A->g_directivity); strlcat (stemp, phg, sizeof(stemp)); } @@ -458,7 +595,13 @@ void decode_aprs_print (decode_aprs_t *A) { snprintf (rng, sizeof(rng), ", range=%.1f", A->g_range); strlcat (stemp, rng, sizeof(stemp)); } - text_color_set(DW_COLOR_DECODED); + + if (strncmp(stemp, "ERROR", 5) == 0) { + text_color_set(DW_COLOR_ERROR); + } + else { + text_color_set(DW_COLOR_DECODED); + } dw_printf("%s\n", stemp); /* @@ -493,7 +636,7 @@ void decode_aprs_print (decode_aprs_t *A) { if (A->g_lat != G_UNKNOWN || A->g_lon != G_UNKNOWN) { -// Have location but it is posible one part is invalid. +// Have location but it is possible one part is invalid. if (A->g_lat != G_UNKNOWN) { @@ -540,10 +683,10 @@ void decode_aprs_print (decode_aprs_t *A) { }; if (A->g_speed_mph != G_UNKNOWN) { - char spd[20]; + char spd[32]; if (strlen(stemp) > 0) strlcat (stemp, ", ", sizeof(stemp)); - snprintf (spd, sizeof(spd), "%.0f MPH", A->g_speed_mph); + snprintf (spd, sizeof(spd), "%.0f km/h (%.0f MPH)", DW_MILES_TO_KM(A->g_speed_mph), A->g_speed_mph); strlcat (stemp, spd, sizeof(stemp)); }; @@ -556,10 +699,10 @@ void decode_aprs_print (decode_aprs_t *A) { }; if (A->g_altitude_ft != G_UNKNOWN) { - char alt[20]; + char alt[32]; if (strlen(stemp) > 0) strlcat (stemp, ", ", sizeof(stemp)); - snprintf (alt, sizeof(alt), "alt %.0f ft", A->g_altitude_ft); + snprintf (alt, sizeof(alt), "alt %.0f m (%.0f ft)", DW_FEET_TO_METERS(A->g_altitude_ft), A->g_altitude_ft); strlcat (stemp, alt, sizeof(stemp)); }; @@ -661,6 +804,26 @@ void decode_aprs_print (decode_aprs_t *A) { * To be part of a valid UTF-8 sequence, it would need to be followed by 10xxxxxx. */ +// For values 00-7F, ASCII, Unicode, and ISO Latin-1 are all the same. +// ISO Latin-1 adds 80-FF range with a few common symbols, such as degree, and +// letters, with diacritical marks, for many European languages. +// Unicode range 80-FF is called "Latin-1 Supplement." Exactly the same as ISO Latin-1. +// For UTF-8, an additional byte is inserted. +// Unicode UTF-8 +// ------- ----- +// 8x C2 8x Insert C2, keep original +// 9x C2 9x " +// Ax C2 Ax " +// Bx C2 Bx " +// Cx C3 8x Insert C3, subtract 40 from original +// Dx C3 9x " +// Ex C3 Ax " +// Fx C3 Bx " +// +// Can we use this knowledge to provide guidance on other ISO Latin-1 characters besides degree? +// Should we? +// Reference: https://www.fileformat.info/info/unicode/utf8test.htm + if ( ! A->g_quiet) { for (j=0; jg_msg_type, "Position", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Position", sizeof(A->g_data_type_desc)); p = (struct aprs_ll_pos_s *)info; q = (struct aprs_compressed_pos_s *)info; @@ -743,8 +906,34 @@ static void aprs_ll_pos (decode_aprs_t *A, unsigned char *info, int ilen) /* In this case, we expect 7 byte "data extension" */ /* for the wind direction and speed. */ - strlcpy (A->g_msg_type, "Weather Report", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Weather Report", sizeof(A->g_data_type_desc)); weather_data (A, p->comment, TRUE); +/* +Here is an interesting case. +The protocol spec states that a position report with symbol _ is a special case +and the information part must contain wxnow.txt format weather data. +But, here we see it being generated like a normal position report. + +N8VIM>BEACON,AB1OC-10*,WIDE2-1:!4240.85N/07133.99W_PHG72604/ Pepperell, MA. WX. 442.9+ PL100<0x0d> +Didn't find wind direction in form c999. +Didn't find wind speed in form s999. +Didn't find wind gust in form g999. +Didn't find temperature in form t999. +Weather Report, WEATHER Station (blue) +N 42 40.8500, W 071 33.9900 +, "PHG72604/ Pepperell, MA. WX. 442.9+ PL100" + +It seems, to me, that this is a violation of the protocol spec. +Then, immediately following, we have a positionless weather report in Ultimeter format. + +N8VIM>APN391,AB1OC-10*,WIDE2-1:$ULTW006F00CA01421C52275800008A00000102FA000F04A6000B002A<0x0d><0x0a> +Ultimeter, Kantronics KPC-3 rom versions +wind 6.9 mph, direction 284, temperature 32.2, barometer 29.75, humidity 76 + +aprs.fi merges these two together. Is that anywhere in the protocol spec or +just a heuristic added after noticing a pair of packets like this? +*/ + } else { /* Regular position report. */ @@ -762,7 +951,7 @@ static void aprs_ll_pos (decode_aprs_t *A, unsigned char *info, int ilen) /* compressed data so we don't expect a 7 byte "data */ /* extension" for them. */ - strlcpy (A->g_msg_type, "Weather Report", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Weather Report", sizeof(A->g_data_type_desc)); weather_data (A, q->comment, FALSE); } else { @@ -832,7 +1021,7 @@ static void aprs_ll_pos_time (decode_aprs_t *A, unsigned char *info, int ilen) } *q; - strlcpy (A->g_msg_type, "Position with time", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Position with time", sizeof(A->g_data_type_desc)); time_t ts = 0; @@ -851,7 +1040,7 @@ static void aprs_ll_pos_time (decode_aprs_t *A, unsigned char *info, int ilen) /* In this case, we expect 7 byte "data extension" */ /* for the wind direction and speed. */ - strlcpy (A->g_msg_type, "Weather Report", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Weather Report", sizeof(A->g_data_type_desc)); weather_data (A, p->comment, TRUE); } else { @@ -872,7 +1061,7 @@ static void aprs_ll_pos_time (decode_aprs_t *A, unsigned char *info, int ilen) /* compressed data so we don't expect a 7 byte "data */ /* extension" for them. */ - strlcpy (A->g_msg_type, "Weather Report", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Weather Report", sizeof(A->g_data_type_desc)); weather_data (A, q->comment, FALSE); } else { @@ -920,20 +1109,24 @@ static void aprs_ll_pos_time (decode_aprs_t *A, unsigned char *info, int ilen) static void aprs_raw_nmea (decode_aprs_t *A, unsigned char *info, int ilen) { - if (strncmp((char*)info, "$GPRMC,", 7) == 0) + if (strncmp((char*)info, "$GPRMC,", 7) == 0 || + strncmp((char*)info, "$GNRMC,", 7) == 0) { float speed_knots = G_UNKNOWN; (void) dwgpsnmea_gprmc ((char*)info, A->g_quiet, &(A->g_lat), &(A->g_lon), &speed_knots, &(A->g_course)); A->g_speed_mph = DW_KNOTS_TO_MPH(speed_knots); + strlcpy (A->g_data_type_desc, "Raw GPS data", sizeof(A->g_data_type_desc)); } - else if (strncmp((char*)info, "$GPGGA,", 7) == 0) + else if (strncmp((char*)info, "$GPGGA,", 7) == 0 || + strncmp((char*)info, "$GNGGA,", 7) == 0) { float alt_meters = G_UNKNOWN; int num_sat = 0; (void) dwgpsnmea_gpgga ((char*)info, A->g_quiet, &(A->g_lat), &(A->g_lon), &alt_meters, &num_sat); A->g_altitude_ft = DW_METERS_TO_FEET(alt_meters); + strlcpy (A->g_data_type_desc, "Raw GPS data", sizeof(A->g_data_type_desc)); } // TODO (low): add a few other sentence types. @@ -946,7 +1139,9 @@ static void aprs_raw_nmea (decode_aprs_t *A, unsigned char *info, int ilen) * * Function: aprs_mic_e * - * Purpose: Decode MIC-E (also Kenwood D7 & D700) packet. + * Purpose: Decode MIC-E (e.g. Kenwood D7 & D700) packet. + * This format is an overzelous quest to make the packet as short as possible. + * It uses non-printable characters and hacks wrapped in kludges. * * Inputs: info - Pointer to Information field. * ilen - Information field length. @@ -955,29 +1150,120 @@ static void aprs_raw_nmea (decode_aprs_t *A, unsigned char *info, int ilen) * * Description: * - * Destination Address Field - + * AX.25 Destination Address Field - * - * The 7-byte Destination Address field contains + * The 6-byte Destination Address field contains * the following encoded information: * - * * The 6 latitude digits. - * * A 3-bit Mic-E message identifier, specifying one of 7 Standard Mic-E - * Message Codes or one of 7 Custom Message Codes or an Emergency - * Message Code. - * * The North/South and West/East Indicators. - * * The Longitude Offset Indicator. - * * The generic APRS digipeater path code. - * + * Byte 1: Lat digit 1, message bit A + * Byte 2: Lat digit 2, message bit B + * Byte 3: Lat digit 3, message bit C + * Byte 4: Lat digit 4, N/S lat indicator + * Byte 5: Lat digit 5, Longitude offset + * Byte 6: Lat digit 6, W/E Long indicator + * * * "Although the destination address appears to be quite unconventional, it is * still a valid AX.25 address, consisting only of printable 7-bit ASCII values." * - * References: Mic-E TYPE CODES -- http://www.aprs.org/aprs12/mic-e-types.txt + * AX.25 Information Field - Starts with ' or ` + * + * Bytes 1,2,3: Longitude + * Bytes 4,5,6: Speed and Course + * Byte 6: Symbol code + * Byte 7: Symbol Table ID + * + * The rest of it is a complicated comment field which can hold various information + * and must be intrepreted in a particular order. At this point we look for any + * prefix and/or suffix to identify the equipment type. + * + * References: Mic-E TYPE CODES -- http://www.aprs.org/aprs12/mic-e-types.txt + * Mic-E TEST EXAMPLES -- http://www.aprs.org/aprs12/mic-e-examples.txt + * + * Next, we have what Addedum 1.2 calls the "type byte." This prefix can be + * space Original MIC-E. + * > Kenwood HT. + * ] Kenwood Mobile. + * none. + * + * We also need to look at the last byte or two + * for a possible suffix to distinguish equipment types. Examples: + * >...... is D7 + * >......= is D72 + * >......^ is D74 + * + * For other brands, it gets worse. There might a 2 character suffix. + * The prefix indicates whether messaging-capable. Examples: + * `....._.% Yaesu FTM-400DR + * `......_) Yaesu FTM-100D + * `......_3 Yaesu FT5D + * + * '......|3 Byonics TinyTrack3 + * '......|4 Byonics TinyTrack4 + * + * Any prefix and suffix must be removed before futher processsing. + * + * Pick one: MIC-E Telemetry Data or "Status Text" (called a comment everywhere else). + * + * If the character after the symbol table id is "," (comma) or 0x1d, we have telemetry. + * (Is this obsoleted by the base-91 telemetry?) + * + * ` Two 2-character hexadecimal numbers. (Channels 1 & 3) + * ' Five 2-character hexadecimal numbers. + * + * Anything left over is a comment which can contain various types of information. + * + * If present, the MIC-E compressed altitude must be first. + * It is three base-91 characters followed by "}". + * Examples: "4T} "4T} ]"4T} + * + * We can also have frequency specification -- http://www.aprs.org/info/freqspec.txt + * + * Warning: Some Kenwood radios add CR at the end, in apparent violation of the spec. + * Watch out so it doesn't get included when looking for equipment type suffix. * * Mic-E TEST EXAMPLES -- http://www.aprs.org/aprs12/mic-e-examples.txt * - * Examples: `b9Z!4y>/>"4N}Paul's_TH-D7 + * Examples: Observed on the air. + * + * KB1HNZ-9>TSSP5P,W1IMD,WIDE1,KQ1L-8,N3LLO-3,WIDE2*:`b6,l}#>/]"48}449.225MHz<0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff><0xff>=<0x0d> + * + * ` b6, l}# >/ ] "48} 449.225MHz ...... = <0x0d> + * mic-e long. cs sym prefix alt. freq comment suffix must-ignore + * Kenwood D710 + *--------------- + * + * N1JDU-9>ECCU8Y,W1MHL*,WIDE2-1:'cZ<0x7f>l#H>/]Go fly a kite!<0x0d> * - * TODO: Destination SSID can contain generic digipeater path. + * ' cZ<0x7f> l#H >/ ] ..... <0x0d> + * mic-e long. cs sym prefix comment no-suffix must-ignore + * Kenwood D700 + *--------------- + * + * KC1HHO-7>T2PX5R,WA1PLE-4,WIDE1*,WIDE2-1:`c_snp(k/`"4B}official relay station NTS_(<0x0d> + * + * ` c_s np( k/ ` "4B} ....... _( <0x0d> + * mic-e long. cs sym prefix alt comment suffix must-ignore + * FT2D + *--------------- + * + * N1CMD-12>T3PQ1Y,KA1GJU-3,WIDE1,WA1PLE-4*:`cP#l!Fk/'"7H}|!%&-']|!w`&!|3 + * + * ` cP# l!F k/ ' "7H} |!%&-']| !w`&! |3 + * mic-e long. cs sym prefix alt base91telemetry DAO suffix + * TinyTrack3 + *--------------- + * + * W1STJ-3>T2UR4X,WA1PLE-4,WIDE1*,WIDE2-1:`c@&l#.-/`"5,}146.685MHz T100 -060 146.520 Simplex or Voice Alert_%<0x0d> + * + * ` c@& l#. -/ ` "5,} 146.685MHz T100 -060 .............. _% <0x0d> + * mic-e long. cs sym prefix alt frequency-specification comment suffix must-ignore + * FTM-400DR + *--------------- + * + * + * + * + * TODO: Destination SSID can contain generic digipeater path. (?) * * Bugs: Doesn't handle ambiguous position. "space" treated as zero. * Invalid data results in a message but latitude is not set to unknown. @@ -1017,7 +1303,7 @@ N1ZZN-9>T2SP0W:'c_Vm6hk/`"49}Byonics TinyTrack4|4 # The next group starts with metacharacter "T" which can be any of space > ] ` ' # But space is for original Mic-E, # > and ] are for Kenwood, -# so ` ' would probably be less ambigous choices but any appear to be valid. +# so ` ' would probably be less ambiguous choices but any appear to be valid. N1ZZN-9>T2SP0W:'c_Vm6hk/`"49}Hamhud\9 N1ZZN-9>T2SP0W:'c_Vm6hk/`"49}Argent/9 @@ -1098,7 +1384,7 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int char sym_table_id; } *p; - char dest[10]; + char dest[12]; int ch; int n; int offset; @@ -1108,7 +1394,7 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int const char *cust_text[8] = {"Emergency", "Custom-6", "Custom-5", "Custom-4", "Custom-3", "Custom-2", "Custom-1", "Custom-0" }; unsigned char *pfirst, *plast; - strlcpy (A->g_msg_type, "MIC-E", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "MIC-E", sizeof(A->g_data_type_desc)); p = (struct aprs_mic_e_s *)info; @@ -1125,7 +1411,7 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int mic_e_digit(A, dest[5], 0, &std_msg, &cust_msg)) / 6000.0; -/* 4th character of desination indicates north / south. */ +/* 4th character of destination indicates north / south. */ if ((dest[3] >= '0' && dest[3] <= '9') || dest[3] == 'L') { /* South */ @@ -1167,7 +1453,7 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int /* First character of information field is longitude in degrees. */ /* It is possible for the unprintable DEL character to occur here. */ -/* 5th character of desination indicates longitude offset of +100. */ +/* 5th character of destination indicates longitude offset of +100. */ /* Not quite that simple :-( */ ch = p->lon[0]; @@ -1256,7 +1542,7 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int } } -/* 6th character of destintation indicates east / west. */ +/* 6th character of destination indicates east / west. */ /* * Example of apparently invalid encoding. 6th character missing. @@ -1355,19 +1641,29 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int #define isT(c) ((c) == ' ' || (c) == '>' || (c) == ']' || (c) == '`' || (c) == '\'') -// Last updated Sept. 2016 for TH-D74A +// Last Updated Dec. 2021 + +// This does not change very often but I'm wondering if we could parse +// http://www.aprs.org/aprs12/mic-e-types.txt similar to how we use tocalls.txt. + +// TODO: Use https://github.com/aprsorg/aprs-deviceid rather than hardcoding. if (isT(*pfirst)) { - + +// "legacy" formats. + if (*pfirst == ' ' ) { strlcpy (A->g_mfr, "Original MIC-E", sizeof(A->g_mfr)); pfirst++; } else if (*pfirst == '>' && *plast == '=') { strlcpy (A->g_mfr, "Kenwood TH-D72", sizeof(A->g_mfr)); pfirst++; plast--; } else if (*pfirst == '>' && *plast == '^') { strlcpy (A->g_mfr, "Kenwood TH-D74", sizeof(A->g_mfr)); pfirst++; plast--; } + else if (*pfirst == '>' && *plast == '&') { strlcpy (A->g_mfr, "Kenwood TH-D75", sizeof(A->g_mfr)); pfirst++; plast--; } else if (*pfirst == '>' ) { strlcpy (A->g_mfr, "Kenwood TH-D7A", sizeof(A->g_mfr)); pfirst++; } else if (*pfirst == ']' && *plast == '=') { strlcpy (A->g_mfr, "Kenwood TM-D710", sizeof(A->g_mfr)); pfirst++; plast--; } else if (*pfirst == ']' ) { strlcpy (A->g_mfr, "Kenwood TM-D700", sizeof(A->g_mfr)); pfirst++; } +// ` should be used for message capable devices. + else if (*pfirst == '`' && *(plast-1) == '_' && *plast == ' ') { strlcpy (A->g_mfr, "Yaesu VX-8", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '"') { strlcpy (A->g_mfr, "Yaesu FTM-350", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '#') { strlcpy (A->g_mfr, "Yaesu VX-8G", sizeof(A->g_mfr)); pfirst++; plast-=2; } @@ -1375,10 +1671,20 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '%') { strlcpy (A->g_mfr, "Yaesu FTM-400DR", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '`' && *(plast-1) == '_' && *plast == ')') { strlcpy (A->g_mfr, "Yaesu FTM-100D", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '(') { strlcpy (A->g_mfr, "Yaesu FT2D", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '0') { strlcpy (A->g_mfr, "Yaesu FT3D", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '3') { strlcpy (A->g_mfr, "Yaesu FT5D", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '1') { strlcpy (A->g_mfr, "Yaesu FTM-300D", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if (*pfirst == '`' && *(plast-1) == '_' && *plast == '5') { strlcpy (A->g_mfr, "Yaesu FTM-500D", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '`' && *(plast-1) == ' ' && *plast == 'X') { strlcpy (A->g_mfr, "AP510", sizeof(A->g_mfr)); pfirst++; plast-=2; } - else if (*pfirst == '`' ) { strlcpy (A->g_mfr, "Mic-Emsg", sizeof(A->g_mfr)); pfirst++; } + else if (*pfirst == '`' && *(plast-1) == '(' && *plast == '5') { strlcpy (A->g_mfr, "Anytone D578UV", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if (*pfirst == '`' ) { strlcpy (A->g_mfr, "Generic Mic-Emsg", sizeof(A->g_mfr)); pfirst++; } + +// ' should be used for trackers (not message capable). + + else if (*pfirst == '\'' && *(plast-1) == '(' && *plast == '5') { strlcpy (A->g_mfr, "Anytone D578UV", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if (*pfirst == '\'' && *(plast-1) == '(' && *plast == '8') { strlcpy (A->g_mfr, "Anytone D878UV", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '\'' && *(plast-1) == '|' && *plast == '3') { strlcpy (A->g_mfr, "Byonics TinyTrack3", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '\'' && *(plast-1) == '|' && *plast == '4') { strlcpy (A->g_mfr, "Byonics TinyTrack4", sizeof(A->g_mfr)); pfirst++; plast-=2; } @@ -1386,13 +1692,13 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int else if (*pfirst == '\'' && *(plast-1) == ':' && *plast == '4') { strlcpy (A->g_mfr, "SCS GmbH & Co. P4dragon DR-7400 modems", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if (*pfirst == '\'' && *(plast-1) == ':' && *plast == '8') { strlcpy (A->g_mfr, "SCS GmbH & Co. P4dragon DR-7800 modems", sizeof(A->g_mfr)); pfirst++; plast-=2; } - else if (*pfirst == '\'' ) { strlcpy (A->g_mfr, "McTrackr", sizeof(A->g_mfr)); pfirst++; } + else if (*pfirst == '\'' ) { strlcpy (A->g_mfr, "Generic McTrackr", sizeof(A->g_mfr)); pfirst++; } else if ( *(plast-1) == '\\' ) { strlcpy (A->g_mfr, "Hamhud ?", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if ( *(plast-1) == '/' ) { strlcpy (A->g_mfr, "Argent ?", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if ( *(plast-1) == '^' ) { strlcpy (A->g_mfr, "HinzTec anyfrog", sizeof(A->g_mfr)); pfirst++; plast-=2; } else if ( *(plast-1) == '*' ) { strlcpy (A->g_mfr, "APOZxx www.KissOZ.dk Tracker. OZ1EKD and OZ7HVO", sizeof(A->g_mfr)); pfirst++; plast-=2; } - else if ( *(plast-1) == '~' ) { strlcpy (A->g_mfr, "OTHER", sizeof(A->g_mfr)); pfirst++; plast-=2; } + else if ( *(plast-1) == '~' ) { strlcpy (A->g_mfr, "Unknown OTHER", sizeof(A->g_mfr)); pfirst++; plast-=2; } } /* @@ -1445,11 +1751,11 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int * Purpose: Decode "Message Format." * The word message is used loosely all over the place, but it has a very specific meaning here. * - * Inputs: info - Pointer to Information field. + * Inputs: info - Pointer to Information field. Be careful not to modify it here! * ilen - Information field length. - * quiet - supress error messages. + * quiet - suppress error messages. * - * Outputs: A->g_msg_type Text description for screen display. + * Outputs: A->g_data_type_desc Text description for screen display. * * A->g_addressee To whom is it addressed. * Could be a specific station, alias, bulletin, etc. @@ -1461,40 +1767,57 @@ static void aprs_mic_e (decode_aprs_t *A, packet_t pp, unsigned char *info, int * * A->g_message_number Message number if any. Required for ack/rej. * - * Description: An APRS message is a text string with a specifed addressee. + * Description: An APRS message is a text string with a specified addressee. * * It's a lot more complicated with different types of addressees * and replies with acknowledgement or rejection. * - * There is even a special case for telemetry metadata. + * Is it an elegant generalization to lump all of these special cases + * together or was it a big mistake that will cause confusion and incorrect + * implementations? The decision to put telemetry metadata here is baffling. * * - * Cases: :xxxxxxxxx:PARM. Telemetry metadata, parameter name + * Cases: :BLNxxxxxx: ... Bulletin. + * :NWSxxxxxx: ... National Weather Service Bulletin. + * http://www.aprs.org/APRS-docs/WX.TXT + * :SKYxxxxxx: ... Need reference. + * :CWAxxxxxx: ... Need reference. + * :BOMxxxxxx: ... Australian version. + * + * :xxxxxxxxx:PARM. Telemetry metadata, parameter name * :xxxxxxxxx:UNIT. Telemetry metadata, unit/label - * :xxxxxxxxx:EQNS. Telemetry metadata, Equation Coefficents + * :xxxxxxxxx:EQNS. Telemetry metadata, Equation Coefficients * :xxxxxxxxx:BITS. Telemetry metadata, Bit Sense/Project Name * :xxxxxxxxx:? Directed Station Query - * :xxxxxxxxx:ack Message acknowledged (received) - * :xxxxxxxxx:rej Message rejected (unable to accept) + * :xxxxxxxxx:ackNNNN Message acknowledged (received) + * :xxxxxxxxx:rejNNNNN Message rejected (unable to accept) * * :xxxxxxxxx: ... Message with no message number. * (Text may not contain the { character because * it indicates beginning of optional message number.) - * :xxxxxxxxx: ... {num Message with message number. + * :xxxxxxxxx: ... {NNNNN Message with message number, 1 to 5 alphanumeric. + * :xxxxxxxxx: ... {mm} Message with new style message number. + * :xxxxxxxxx: ... {mm}aa Message with new style message number and ack. + * + * + * Reference: http://www.aprs.org/txt/messages101.txt + * http://www.aprs.org/aprs11/replyacks.txt <-- New (1999) adding ack to outgoing message. * *------------------------------------------------------------------*/ -static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int quiet) +static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int quiet) { struct aprs_message_s { char dti; /* : */ char addressee[9]; char colon; /* : */ - char message[73]; /* 0-67 characters for message */ - /* Optional { followed by 1-5 characters for message number */ + char message[256-1-9-1]; /* Officially up to 67 characters for message text. */ + /* Relaxing seemingly arbitrary restriction here; it doesn't need to fit on a punched card. */ + /* Wouldn't surprise me if others did not pay attention to the limit. */ + /* Optional '{' followed by 1-5 alphanumeric characters for message number */ - /* If the first chracter is '?' it is a Directed Station Query. */ + /* If the first character is '?' it is a Directed Station Query. */ } *p; char addressee[AX25_MAX_ADDR_LEN]; @@ -1502,7 +1825,7 @@ static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int q p = (struct aprs_message_s *)info; - strlcpy (A->g_msg_type, "APRS Message", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "APRS Message", sizeof(A->g_data_type_desc)); A->g_message_subtype = message_subtype_message; /* until found otherwise */ if (ilen < 11) { @@ -1517,7 +1840,8 @@ static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int q if (p->colon != ':') { if (! quiet) { text_color_set(DW_COLOR_ERROR); - dw_printf("APRS Message must begin with : 9 character addressee :\n"); + dw_printf("APRS Message must begin with ':' 9 character addressee ':'\n"); + dw_printf("Spaces must be added to shorter addressee to make 9 characters.\n"); } A->g_message_subtype = message_subtype_invalid; return; @@ -1532,8 +1856,96 @@ static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int q addressee[i--] = '\0'; } + // Anytone AT-D878UV 2 plus would pad out station name to 6 characters + // before appending the SSID. e.g. "AE7MK -5 " + + // Test cases. First is valid. Others should produce errors: + // + // cbeacon sendto=r0 delay=0:10 info=":AE7MK-5 :test0" + // cbeacon sendto=r0 delay=0:15 info=":AE7MK-5:test1" + // cbeacon sendto=r0 delay=0:20 info=":AE7MK -5 :test2" + // cbeacon sendto=r0 delay=0:25 info=":AE7 -5 :test3" + + static regex_t bad_addressee_re; /* Probably bad addressee. */ + static int first_time = 1; + + if (first_time) { + char emsg[100]; + int e = regcomp (&bad_addressee_re, "[A-Z0-9]+ +-[0-9]", REG_EXTENDED); + if (e) { + regerror (e, &bad_addressee_re, emsg, sizeof(emsg)); + dw_printf("%s:%d: %s\n", __FILE__, __LINE__, emsg); + } + first_time = 0; + } + +#define MAXMATCH_AT 2 + regmatch_t match[MAXMATCH_AT]; + + if (regexec (&bad_addressee_re, addressee, MAXMATCH_AT, match, 0) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Malformed addressee with space between station name and SSID.\n"); + dw_printf("Please tell message sender this is invalid.\n"); + } + strlcpy (A->g_addressee, addressee, sizeof(A->g_addressee)); +/* + * Addressee starting with BLN or NWS is a bulletin. + */ + if (strlen(addressee) >= 3 && strncmp(addressee,"BLN",3) == 0) { + + // Interpret 3 cases of identifiers. + // BLN9 "general bulletin" has a single digit. + // BLNX "announcement" has a single uppercase letter. + // BLN9xxxxx "group bulletin" has single digit group id and group name up to 5 characters. + + if (strlen(addressee) == 4 && isdigit(addressee[3])) { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "General Bulletin with identifier \"%s\"", addressee+3); + } + else if (strlen(addressee) == 4 && isupper(addressee[3])) { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Announcement with identifier \"%s\"", addressee+3); + } + if (strlen(addressee) >=5 && isdigit(addressee[3])) { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Group Bulletin with identifier \"%c\", group name \"%s\"", addressee[3], addressee+4); + } + else { + // Not one of the official formats. + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Bulletin with identifier \"%s\"", addressee+3); + } + A->g_message_subtype = message_subtype_bulletin; + strlcpy (A->g_comment, p->message, sizeof(A->g_comment)); + } + + + // Weather bulletins have addressee starting with NWS, SKY, CWA, or BOM. + // The protocol spec and http://www.aprs.org/APRS-docs/WX.TXT state that + // the 3 letter prefix must be followed by a dash. + // However, https://www.aprs-is.net/WX/ also lists the underscore + // alternative for the compressed format. Xastir implements this. + + else if (strlen(addressee) >= 3 && strncmp(addressee,"NWS",3) == 0) { + + if (strlen(addressee) >=4 && addressee[3] == '-') { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Weather bulletin with identifier \"%s\"", addressee+4); + } + else if (strlen(addressee) >=4 && addressee[3] == '_') { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Compressed Weather bulletin with identifier \"%s\"", addressee+4); + } + else { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Weather bulletin is missing - or _ after %.3s", addressee); + } + A->g_message_subtype = message_subtype_nws; + strlcpy (A->g_comment, p->message, sizeof(A->g_comment)); + } + + else if (strlen(addressee) >= 3 && (strncmp(addressee,"SKY",3) == 0 || strncmp(addressee,"CWA",3) == 0 || strncmp(addressee,"BOM",3) == 0)) { + // SKY... or CWA... https://www.aprs-is.net/WX/ + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Weather bulletin with identifier \"%s\"", addressee+4); + A->g_message_subtype = message_subtype_nws; + strlcpy (A->g_comment, p->message, sizeof(A->g_comment)); + } + /* * Special message formats contain telemetry metadata. @@ -1546,23 +1958,23 @@ static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int q * Why not use other characters after the "T" for metadata? */ - if (strncmp(p->message,"PARM.",5) == 0) { - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "Telemetry Parameter Name Message for \"%s\"", addressee); + else if (strncmp(p->message,"PARM.",5) == 0) { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Telemetry Parameter Name for \"%s\"", addressee); A->g_message_subtype = message_subtype_telem_parm; telemetry_name_message (addressee, p->message+5); } else if (strncmp(p->message,"UNIT.",5) == 0) { - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "Telemetry Unit/Label Message for \"%s\"", addressee); + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Telemetry Unit/Label for \"%s\"", addressee); A->g_message_subtype = message_subtype_telem_unit; telemetry_unit_label_message (addressee, p->message+5); } else if (strncmp(p->message,"EQNS.",5) == 0) { - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "Telemetry Equation Coefficents Message for \"%s\"", addressee); + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Telemetry Equation Coefficients for \"%s\"", addressee); A->g_message_subtype = message_subtype_telem_eqns; telemetry_coefficents_message (addressee, p->message+5, quiet); } else if (strncmp(p->message,"BITS.",5) == 0) { - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "Telemetry Bit Sense/Project Name Message for \"%s\"", addressee); + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "Telemetry Bit Sense/Project Name for \"%s\"", addressee); A->g_message_subtype = message_subtype_telem_bits; telemetry_bit_sense_message (addressee, p->message+5, quiet); } @@ -1573,7 +1985,7 @@ static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int q else if (p->message[0] == '?') { - strlcpy (A->g_msg_type, "Directed Station Query", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Directed Station Query", sizeof(A->g_data_type_desc)); A->g_message_subtype = message_subtype_directed_query; aprs_directed_station_query (A, addressee, p->message+1, quiet); @@ -1581,30 +1993,128 @@ static void aprs_message (decode_aprs_t *A, unsigned char *info, int ilen, int q /* ack or rej? Message number is required for these. */ - else if (strncmp(p->message,"ack",3) == 0) { - strlcpy (A->g_message_number, p->message + 3, sizeof(A->g_message_number)); - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "ACK message %s for \"%s\"", A->g_message_number, addressee); + else if (strncasecmp(p->message,"ack",3) == 0) { + if (strncmp(p->message,"ack",3) != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("ERROR: \"%s\" must be lower case \"ack\"\n", p->message); + } + else { + strlcpy (A->g_message_number, p->message + 3, sizeof(A->g_message_number)); + if (strlen(A->g_message_number) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("ERROR: Message number is missing after \"ack\".\n"); + } + } + + // Xastir puts a carriage return on the end. + char *p = strchr(A->g_message_number, '\r'); + if (p != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf("The APRS protocol specification says nothing about a possible carriage return after the\n"); + dw_printf("message id. Adding CR might prevent proper interoperability with with other applications.\n"); + *p = '\0'; + } + + if (strlen(A->g_message_number) >= 3 && A->g_message_number[2] == '}') A->g_message_number[2] = '\0'; + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "\"%s\" ACKnowledged message number \"%s\" from \"%s\"", A->g_src, A->g_message_number, addressee); A->g_message_subtype = message_subtype_ack; } - else if (strncmp(p->message,"rej",3) == 0) { - strlcpy (A->g_message_number, p->message + 3, sizeof(A->g_message_number)); - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "REJ message %s for \"%s\"", A->g_message_number, addressee); + else if (strncasecmp(p->message,"rej",3) == 0) { + if (strncmp(p->message,"rej",3) != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("ERROR: \"%s\" must be lower case \"rej\"\n", p->message); + } + else { + strlcpy (A->g_message_number, p->message + 3, sizeof(A->g_message_number)); + if (strlen(A->g_message_number) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("ERROR: Message number is missing after \"rej\".\n"); + } + } + + // Xastir puts a carriage return on the end. + char *p = strchr(A->g_message_number, '\r'); + if (p != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf("The APRS protocol specification says nothing about a possible carriage return after the\n"); + dw_printf("message id. Adding CR might prevent proper interoperability with with other applications.\n"); + *p = '\0'; + } + + if (strlen(A->g_message_number) >= 3 && A->g_message_number[2] == '}') A->g_message_number[2] = '\0'; + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "\"%s\" REJected message number \"%s\" from \"%s\"", A->g_src, A->g_message_number, addressee); A->g_message_subtype = message_subtype_ack; } -/* message number is optional here. */ +// Message to a particular station or a bulletin. +// message number is optional here. +// Test cases. Wrap in third party too. +// A>B::WA1XYX-15:Howdy y'all +// A>B::WA1XYX-15:Howdy y'all{12345 +// A>B::WA1XYX-15:Howdy y'all{12} +// A>B::WA1XYX-15:Howdy y'all{12}34 +// A>B::WA1XYX-15:Howdy y'all{toolong +// X>Y:}A>B::WA1XYX-15:Howdy y'all +// X>Y:}A>B::WA1XYX-15:Howdy y'all{12345 +// X>Y:}A>B::WA1XYX-15:Howdy y'all{12} +// X>Y:}A>B::WA1XYX-15:Howdy y'all{12}34 +// X>Y:}A>B::WA1XYX-15:Howdy y'all{toolong else { + // Normal messaage case. Look for message number. char *pno = strchr(p->message, '{'); if (pno != NULL) { strlcpy (A->g_message_number, pno+1, sizeof(A->g_message_number)); + + // Xastir puts a carriage return on the end. + char *p = strchr(A->g_message_number, '\r'); + if (p != NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf("The APRS protocol specification says nothing about a possible carriage return after the\n"); + dw_printf("message id. Adding CR might prevent proper interoperability with with other applications.\n"); + *p = '\0'; + } + + int mlen = strlen(A->g_message_number); + if (mlen < 1 || mlen > 5) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Message number \"%s\" has length outside range of 1 to 5.\n", A->g_message_number); + } + + // TODO: Complain if not alphanumeric. + + char ack[8] = ""; + + if (mlen >= 3 && A->g_message_number[2] == '}') { + // New (1999) style. + A->g_message_number[2] = '\0'; + strlcpy (ack, A->g_message_number + 3, sizeof(ack)); + } + + if (strlen(ack) > 0) { + // With ACK. Message number should be 2 characters. + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "APRS Message, number \"%s\", from \"%s\" to \"%s\", with ACK for \"%s\"", A->g_message_number, A->g_src, addressee, ack); + } + else { + // Message number can be 1-5 characters. + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "APRS Message, number \"%s\", from \"%s\" to \"%s\"", A->g_message_number, A->g_src, addressee); + } + } + else { + // No message number. + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "APRS Message, with no number, from \"%s\" to \"%s\"", A->g_src, addressee); } - snprintf (A->g_msg_type, sizeof(A->g_msg_type), "APRS Message %s for \"%s\"", A->g_message_number, addressee); + A->g_message_subtype = message_subtype_message; /* No location so don't use process_comment () */ strlcpy (A->g_comment, p->message, sizeof(A->g_comment)); + // Remove message number when displaying message text. + pno = strchr(A->g_comment, '{'); + if (pno != NULL) { + *pno = '\0'; + } } } @@ -1676,11 +2186,11 @@ static void aprs_object (decode_aprs_t *A, unsigned char *info, int ilen) } if (p->live_killed == '*') - strlcpy (A->g_msg_type, "Object", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Object", sizeof(A->g_data_type_desc)); else if (p->live_killed == '_') - strlcpy (A->g_msg_type, "Killed Object", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Killed Object", sizeof(A->g_data_type_desc)); else - strlcpy (A->g_msg_type, "Object - invalid live/killed", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Object - invalid live/killed", sizeof(A->g_data_type_desc)); ts = get_timestamp (A, p->time_stamp); @@ -1693,7 +2203,7 @@ static void aprs_object (decode_aprs_t *A, unsigned char *info, int ilen) /* In this case, we expect 7 byte "data extension" */ /* for the wind direction and speed. */ - strlcpy (A->g_msg_type, "Weather Report with Object", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Weather Report with Object", sizeof(A->g_data_type_desc)); weather_data (A, p->comment, TRUE); } else { @@ -1712,7 +2222,7 @@ static void aprs_object (decode_aprs_t *A, unsigned char *info, int ilen) /* of weather report and object with compressed */ /* position. */ - strlcpy (A->g_msg_type, "Weather Report with Object", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Weather Report with Object", sizeof(A->g_data_type_desc)); weather_data (A, q->comment, FALSE); } else { @@ -1793,15 +2303,15 @@ static void aprs_item (decode_aprs_t *A, unsigned char *info, int ilen) } if (p->name[i] == '!') - strlcpy (A->g_msg_type, "Item", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Item", sizeof(A->g_data_type_desc)); else if (p->name[i] == '_') - strlcpy (A->g_msg_type, "Killed Item", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Killed Item", sizeof(A->g_data_type_desc)); else { if ( ! A->g_quiet) { text_color_set(DW_COLOR_ERROR); dw_printf("Item name too long or not followed by ! or _.\n"); } - strlcpy (A->g_msg_type, "Object - invalid live/killed", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Object - invalid live/killed", sizeof(A->g_data_type_desc)); } ppos = p->name + i + 1; @@ -1845,7 +2355,7 @@ static void aprs_item (decode_aprs_t *A, unsigned char *info, int ilen) static void aprs_station_capabilities (decode_aprs_t *A, char *info, int ilen) { - strlcpy (A->g_msg_type, "Station Capabilities", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Station Capabilities", sizeof(A->g_data_type_desc)); // process_comment() not applicable here because it // extracts information found in certain formats. @@ -1932,7 +2442,7 @@ static void aprs_status_report (decode_aprs_t *A, char *info, int ilen) } *ps; - strlcpy (A->g_msg_type, "Status Report", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Status Report", sizeof(A->g_data_type_desc)); pt = (struct aprs_status_time_s *)info; pm4 = (struct aprs_status_m4_s *)info; @@ -1959,6 +2469,7 @@ static void aprs_status_report (decode_aprs_t *A, char *info, int ilen) /* * Do we have format with 6 character Maidenhead locator? */ + else if (get_maidenhead (A, pm6->mhead6) == 6) { memset (A->g_maidenhead, 0, sizeof(A->g_maidenhead)); @@ -2103,6 +2614,20 @@ static void aprs_status_report (decode_aprs_t *A, char *info, int ilen) * *------------------------------------------------------------------*/ +/* +https://groups.io/g/direwolf/topic/95961245#7357 + +What APRS queries should DireWolf respond to? Well, it should be configurable whether it responds to queries at all, in case some other application is using DireWolf as a dumb TNC (KISS or AGWPE style) and wants to handle the queries itself. + +Assuming query responding is enabled, the following broadcast queries should be supported (if the corresponding data is configured in DireWolf): + +?APRS (I am an APRS station) +?IGATE (I am operating as a I-gate) +?WX (I am providing local weather data in my beacon) + +*/ + + static void aprs_general_query (decode_aprs_t *A, char *info, int ilen, int quiet) { char *q2; @@ -2112,7 +2637,7 @@ static void aprs_general_query (decode_aprs_t *A, char *info, int ilen, int quie double lat, lon; float radius; - strlcpy (A->g_msg_type, "General Query", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "General Query", sizeof(A->g_data_type_desc)); /* * First make a copy because we will modify it while parsing it. @@ -2255,6 +2780,28 @@ static void aprs_general_query (decode_aprs_t *A, char *info, int ilen, int quie * *------------------------------------------------------------------*/ +/* +https://groups.io/g/direwolf/topic/95961245#7357 + +The following directed queries (sent as bodies of APRS text messages) would also be useful (if corresponding data configured): + +?APRSP (force my current beacon) +?APRST and ?PING (trace my path to requestor) +?APRSD (all stations directly heard [no digipeat hops] by local station) +?APRSO (any Objects/Items originated by this station) +?APRSH (how often or how many times the specified 3rd station was heard by the queried station) +?APRSS (immediately send the Status message if configured) (can DireWolf do Status messages?) + +Lynn KJ4ERJ and I have implemented a non-standard query which might be useful: + +?VER (send the human-readable software version of the queried station) + +Hope this is useful. It's just my $.02. + +Andrew, KA2DDO +author of YAAC +*/ + static void aprs_directed_station_query (decode_aprs_t *A, char *addressee, char *query, int quiet) { //char query_type[20]; /* Does the query type always need to be exactly 5 characters? */ @@ -2294,7 +2841,7 @@ static void aprs_directed_station_query (decode_aprs_t *A, char *addressee, char static void aprs_telemetry (decode_aprs_t *A, char *info, int ilen, int quiet) { - strlcpy (A->g_msg_type, "Telemetry", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Telemetry", sizeof(A->g_data_type_desc)); telemetry_data_original (A->g_src, info, quiet, A->g_telemetry, sizeof(A->g_telemetry), A->g_comment, sizeof(A->g_comment)); @@ -2302,6 +2849,56 @@ static void aprs_telemetry (decode_aprs_t *A, char *info, int ilen, int quiet) } /* end aprs_telemetry */ +/*------------------------------------------------------------------ + * + * Function: aprs_user_defined + * + * Purpose: Decode user defined data. + * + * Inputs: info - Pointer to Information field. + * ilen - Information field length. + * + * Description: APRS Protocol Specification, Chapter 18 + * User IDs allocated here: http://www.aprs.org/aprs11/expfmts.txt + * + *------------------------------------------------------------------*/ + +static void aprs_user_defined (decode_aprs_t *A, char *info, int ilen) +{ + if (strncmp(info, "{tt", 3) == 0 || // Historical. + strncmp(info, "{DT", 3) == 0) { // Official after registering {D* + aprs_raw_touch_tone (A, info, ilen); + } + else if (strncmp(info, "{mc", 3) == 0 || // Historical. + strncmp(info, "{DM", 3) == 0) { // Official after registering {D* + aprs_morse_code (A, info, ilen); + } + else if (info[0] == '{' && info[1] == USER_DEF_USER_ID && info[2] == USER_DEF_TYPE_AIS) { + double lat, lon; + float knots, course; + float alt_meters; + + ais_parse (info+3, 0, A->g_data_type_desc, sizeof(A->g_data_type_desc), A->g_name, sizeof(A->g_name), + &lat, &lon, &knots, &course, &alt_meters, &(A->g_symbol_table), &(A->g_symbol_code), + A->g_comment, sizeof(A->g_comment)); + + A->g_lat = lat; + A->g_lon = lon; + A->g_speed_mph = DW_KNOTS_TO_MPH(knots); + A->g_course = course; + A->g_altitude_ft = DW_METERS_TO_FEET(alt_meters); + strcpy (A->g_mfr, ""); + } + else if (strncmp(info, "{{", 2) == 0) { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "User-Defined Experimental"); + } + else { + snprintf (A->g_data_type_desc, sizeof(A->g_data_type_desc), "User-Defined Data"); + } + +} /* end aprs_user_defined */ + + /*------------------------------------------------------------------ * * Function: aprs_raw_touch_tone @@ -2321,7 +2918,7 @@ static void aprs_telemetry (decode_aprs_t *A, char *info, int ilen, int quiet) static void aprs_raw_touch_tone (decode_aprs_t *A, char *info, int ilen) { - strlcpy (A->g_msg_type, "Raw Touch Tone Data", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Raw Touch Tone Data", sizeof(A->g_data_type_desc)); /* Just copy the info field without the message type. */ @@ -2352,7 +2949,7 @@ static void aprs_raw_touch_tone (decode_aprs_t *A, char *info, int ilen) static void aprs_morse_code (decode_aprs_t *A, char *info, int ilen) { - strlcpy (A->g_msg_type, "Morse Code Data", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Morse Code Data", sizeof(A->g_data_type_desc)); /* Just copy the info field without the message type. */ @@ -2392,7 +2989,7 @@ static void aprs_positionless_weather_report (decode_aprs_t *A, unsigned char *i } *p; - strlcpy (A->g_msg_type, "Positionless Weather Report", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Positionless Weather Report", sizeof(A->g_data_type_desc)); //time_t ts = 0; @@ -2797,7 +3394,7 @@ static void aprs_ultimeter (decode_aprs_t *A, char *info, int ilen) int n; - strlcpy (A->g_msg_type, "Ultimeter", sizeof(A->g_msg_type)); + strlcpy (A->g_data_type_desc, "Ultimeter", sizeof(A->g_data_type_desc)); if (*info == '$') { @@ -2876,31 +3473,6 @@ static void aprs_ultimeter (decode_aprs_t *A, char *info, int ilen) } /* end aprs_ultimeter */ -/*------------------------------------------------------------------ - * - * Function: third_party_header - * - * Purpose: Decode packet from a third party network. - * - * Inputs: info - Pointer to Information field. - * ilen - Information field length. - * - * Outputs: A->g_comment - * - * Description: - * - *------------------------------------------------------------------*/ - -static void third_party_header (decode_aprs_t *A, char *info, int ilen) -{ - - strlcpy (A->g_msg_type, "Third Party Header", sizeof(A->g_msg_type)); - - /* more later? */ - -} /* end third_party_header */ - - /*------------------------------------------------------------------ * @@ -3183,7 +3755,7 @@ double get_latitude_8 (char *p, int quiet) else { if ( ! quiet) { text_color_set(DW_COLOR_ERROR); - dw_printf("Error: '%c' found for latitude hemisphere. Specification requires upper case N or s.\n", plat->ns); + dw_printf("Error: '%c' found for latitude hemisphere. Specification requires upper case N or S.\n", plat->ns); } return (G_UNKNOWN); } @@ -3436,6 +4008,8 @@ time_t get_timestamp (decode_aprs_t *A, char *p) time_t ts; ts = time(NULL); + // FIXME: use gmtime_r instead. + // Besides not being thread safe, gmtime could possibly return null. ptm = gmtime(&ts); pdhm = (void *)p; @@ -3505,9 +4079,14 @@ time_t get_timestamp (decode_aprs_t *A, char *p) * It is composed of: * a pair of letters in range A to R. * a pair of digits in range of 0 to 9. - * a pair of letters in range of A to X. + * an optional pair of letters in range of A to X. + * + * The spec says: + * "All letters must be transmitted in upper case. + * Letters may be received in upper case or lower case." * - * The APRS spec says that all letters must be transmitted in upper case. + * Typically the second set of letters is written in lower case. + * An earlier version incorrectly produced an error if lower case found. * * * Examples from APRS spec: @@ -3528,25 +4107,10 @@ int get_maidenhead (decode_aprs_t *A, char *p) /* We have 4 characters matching the rule. */ - if (islower(p[0]) || islower(p[1])) { - if ( ! A->g_quiet) { - text_color_set(DW_COLOR_ERROR); - dw_printf("Warning: Lower case letter in Maidenhead locator. Specification requires upper case.\n"); - } - } - if (toupper(p[4]) >= 'A' && toupper(p[4]) <= 'X' && toupper(p[5]) >= 'A' && toupper(p[5]) <= 'X') { /* We have 6 characters matching the rule. */ - - if (islower(p[4]) || islower(p[5])) { - if ( ! A->g_quiet) { - text_color_set(DW_COLOR_ERROR); - dw_printf("Warning: Lower case letter in Maidenhead locator. Specification requires upper case.\n"); - } - } - return 6; } @@ -3648,6 +4212,9 @@ static int data_extension_comment (decode_aprs_t *A, char *pdext) strlcpy (A->g_directivity, dir[pdext[6]-'0'], sizeof(A->g_directivity)); } +// TODO: look for another 0-9 A-Z followed by a / +// http://www.aprs.org/aprs12/probes.txt + process_comment (A, pdext+7, -1); return 1; } @@ -3704,8 +4271,12 @@ static int data_extension_comment (decode_aprs_t *A, char *pdext) * * Windows version: File must be in current working directory. * - * Linux version: Search order is current working directory - * then /usr/share/direwolf directory. + * Linux version: Search order is current working directory then + * /usr/local/share/direwolf + * /usr/share/direwolf/tocalls.txt + * + * Mac: Like Linux and then + * /opt/local/share/direwolf * *------------------------------------------------------------------*/ @@ -3717,7 +4288,7 @@ static int data_extension_comment (decode_aprs_t *A, char *pdext) // Dec. 2016 tocalls.txt has 153 destination addresses. -#define MAX_TOCALLS 200 +#define MAX_TOCALLS 250 static struct tocalls_s { unsigned char len; @@ -3728,13 +4299,24 @@ static struct tocalls_s { static int num_tocalls = 0; // Make sure the array is null terminated. +// If search order is changed, do the same in symbols.c for consistency. + static const char *search_locations[] = { - (const char *) "tocalls.txt", + (const char *) "tocalls.txt", // CWD + (const char *) "data/tocalls.txt", // Windows with CMake + (const char *) "../data/tocalls.txt", // ? #ifndef __WIN32__ - (const char *) "/usr/share/direwolf/tocalls.txt", (const char *) "/usr/local/share/direwolf/tocalls.txt", + (const char *) "/usr/share/direwolf/tocalls.txt", #endif - (const char *) NULL +#if __APPLE__ + // https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/messages/2458 + // Adding the /opt/local tree since macports typically installs there. Users might want their + // INSTALLDIR (see Makefile.macosx) to mirror that. If so, then we need to search the /opt/local + // path as well. + (const char *) "/opt/local/share/direwolf/tocalls.txt", +#endif + (const char *) NULL // Important - Indicates end of list. }; static int tocall_cmp (const void *px, const void *py) @@ -3857,11 +4439,8 @@ static void decode_tocall (decode_aprs_t *A, char *dest) * models before getting to the more generic APY. */ -#if defined(__WIN32__) || defined(__OpenBSD__) || defined(__FreeBSD__) || defined(__APPLE__) qsort (tocalls, num_tocalls, sizeof(struct tocalls_s), tocall_cmp); -#else - qsort (tocalls, num_tocalls, sizeof(struct tocalls_s), (__compar_fn_t)tocall_cmp); -#endif + } else { if ( ! A->g_quiet) { @@ -3961,7 +4540,7 @@ static void substr_se (char *dest, const char *src, int start, int endp1) * * !DAO! APRS precision and Datum option. * - * Protocol reference, end of chaper 6. + * Protocol reference, end of chapter 6. * * /A=123456 Altitude * @@ -4001,7 +4580,7 @@ static void substr_se (char *dest, const char *src, int start, int endp1) * This same thing has been observed from others and is intermittent. * * * AGW Tracker can send UTF-16 if an option is selected. This can introduce nul bytes. - * This is wrong. It should be using UTF-8 and I'm not going to accomodate it here. + * This is wrong. It should be using UTF-8 and I'm not going to accommodate it here. * * * The digipeater and IGate functions should pass along anything exactly the @@ -4057,6 +4636,7 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) static regex_t base91_tel_re; /* Base 91 compressed telemetry data. */ + int e; char emsg[100]; #define MAXMATCH 4 @@ -4146,9 +4726,7 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) dw_printf("%s:%d: %s\n", __FILE__, __LINE__, emsg); } -// TODO: Would like to restrict to even length something like this: ([!-{][!-{]){2,7} - - e = regcomp (&base91_tel_re, "\\|([!-{]{4,14})\\|", REG_EXTENDED); + e = regcomp (&base91_tel_re, "\\|(([!-{][!-{]){2,7})\\|", REG_EXTENDED); if (e) { regerror (e, &base91_tel_re, emsg, sizeof(emsg)); dw_printf("%s:%d: %s\n", __FILE__, __LINE__, emsg); @@ -4168,7 +4746,7 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) /* * Watch out for buffer overflow. * KG6AZZ reports that there is a local digipeater that seems to - * malfunction ocassionally. It corrupts the packet, as it is + * malfunction occasionally. It corrupts the packet, as it is * digipeated, causing the comment to be hundreds of characters long. */ @@ -4346,20 +4924,20 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) /* * Telemetry data, in base 91 compressed format appears as 2 to 7 pairs - * of base 91 digits, surrounded by | at start and end. + * of base 91 digits, ! thru {, surrounded by | at start and end. */ if (regexec (&base91_tel_re, A->g_comment, MAXMATCH, match, 0) == 0) { - char tdata[30]; /* Should be 4 to 14 characters. */ + char tdata[30]; /* Should be even number of 4 to 14 characters. */ - //dw_printf("compressed telemetry start=%d, end=%d\n", (int)(match[0].rm_so), (int)(match[0].rm_eo)); + //dw_printf("compressed telemetry start=%d, end=%d\n", (int)(match[0].rm_so), (int)(match[0].rm_eo)); substr_se (tdata, A->g_comment, match[1].rm_so, match[1].rm_eo); - //dw_printf("compressed telemetry data = \"%s\"\n", tdata); + //dw_printf("compressed telemetry data = \"%s\"\n", tdata); telemetry_data_base91 (A->g_src, tdata, A->g_telemetry, sizeof(A->g_telemetry)); @@ -4375,8 +4953,10 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) * This would not make sense to use this with a compressed location which * already has much greater resolution. * - * It surprized me to see this in a MIC-E message. + * It surprised me to see this in a MIC-E message. * MIC-E has resolution of .01 minute so it would make sense to have it as an option. + * We also find an example in http://www.aprs.org/aprs12/mic-e-examples.txt + * 'abc123R/'123}FFF.FFFMHztext.../A=123456...!DAO! Mv */ if (regexec (&dao_re, A->g_comment, MAXMATCH, match, 0) == 0) @@ -4386,7 +4966,7 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) int a = A->g_comment[match[0].rm_so+2]; int o = A->g_comment[match[0].rm_so+3]; - //dw_printf("start=%d, end=%d\n", (int)(match[0].rm_so), (int)(match[0].rm_eo)); + //dw_printf("DAO start=%d, end=%d\n", (int)(match[0].rm_so), (int)(match[0].rm_eo)); /* @@ -4437,11 +5017,44 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) * to stretch the numeric range to be 0 to 99. */ +/* + * Here are a couple situations where it is seen. + * + * W8SAT-1>T2UV0P:`qC<0x1f>l!Xu\'"69}WMNI EDS Response Unit #1|+/%0'n|!w:X!|3 + * + * Let's break that down into pieces. + * + * W8SAT-1>T2UV0P:`qC<0x1f>l!Xu\'"69} MIC-E format + * N 42 56.0000, W 085 39.0300, + * 0 MPH, course 160, alt 709 ft + * WMNI EDS Response Unit #1 comment + * |+/%0'n| base 91 telemetry + * !w:X! DAO + * |3 Tiny Track 3 + * + * Comment earlier points out that MIC-E format has resolution of 0.01 minute, + * same as non-compressed format, so the DAO does work out, after thinking + * about it for a while. + * We also find a MIC-E example with !DAO! here: http://www.aprs.org/aprs12/mic-e-examples.txt + * + * Another one: + * + * KS4FUN-12>3X0PRU,W6CX-3,BKELEY,WIDE2*:`2^=l!<0x1c>+/'"48}MT-RTG|%B%p'a|!wqR!|3 + * + * MIC-E, Red Cross, Special + * N 38 00.2588, W 122 06.3354 + * 0 MPH, course 100, alt 108 ft + * MT-RTG comment + * |%B%p'a| Seq=397, A1=443, A2=610 + * !wqR! DAO + * |3 Byonics TinyTrack3 + * + */ + /* * The spec appears to be wrong. It says '}' is the maximum value when it should be '{'. */ - if (isdigit91(a)) { A->g_lat += (a - B91_MIN) * 1.1 / 600000.0 * sign(A->g_lat); } @@ -4583,10 +5196,21 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) * * WB2OSZ-1>APN383,qAR,N1EDU-2:!4237.14NS07120.83W#PHG7130Chelmsford, MA * + * New for 1.5: + * + * Also allow hexadecimal bytes for raw AX.25 or KISS. e.g. + * + * 00 82 a0 ae ae 62 60 e0 82 96 68 84 40 40 60 9c 68 b0 ae 86 40 e0 40 ae 92 88 8a 64 63 03 f0 3e 45 4d 36 34 6e 65 2f 23 20 45 63 68 6f 6c 69 6e 6b 20 31 34 35 2e 33 31 30 2f 31 30 30 68 7a 20 54 6f 6e 65 + * + * If it begins with 00 or C0 (which would be impossible for AX.25 address) process as KISS. + * Also print these formats. * * Outputs: stdout * * Description: Compile like this to make a standalone test program. + * Just run "make" and this will be built along with everything else. + * The point I'm trying to make is that DECAMAIN must be defined + * to enable the main program here for a standalone application. * * gcc -o decode_aprs -DDECAMAIN decode_aprs.c ax25_pad.c ... * @@ -4614,11 +5238,16 @@ static void process_comment (decode_aprs_t *A, char *pstart, int clen) * TODO: To make it more useful, * - Remove any leading timestamp. * - Remove any "qA*" and following from the path. + * - Handle non-APRS frames properly. * *------------------------------------------------------------------*/ #if DECAMAIN +#include "kiss_frame.h" + + + /* Stub for stand-alone decoder. */ void nmea_send_waypoint (char *wname_in, double dlat, double dlong, char symtab, char symbol, @@ -4627,11 +5256,48 @@ void nmea_send_waypoint (char *wname_in, double dlat, double dlong, char symtab, return; } +// TODO: hex_dump is currently in server.c and we don't want to drag that in. +// Someday put it in a more reasonable place, with other general utilities, and remove the private copy here. + + +static void hex_dump (unsigned char *p, int len) +{ + int n, i, offset; + + offset = 0; + while (len > 0) { + n = len < 16 ? len : 16; + dw_printf (" %03x: ", offset); + for (i=0; iachan[chan].valid) { + if (save_audio_config_p->chan_medium[chan] == MEDIUM_RADIO) { char *p; char just_letters[16]; @@ -122,7 +116,6 @@ int demod_init (struct audio_s *pa) * This can be increased by: * Multiple frequencies. * Multiple letters (not sure if I will continue this). - * New interleaved decoders. * * num_slicers is set to max by the "+" option. */ @@ -136,6 +129,20 @@ int demod_init (struct audio_s *pa) break; case MODEM_AFSK: + case MODEM_EAS: + + if (save_audio_config_p->achan[chan].modem_type == MODEM_EAS) { + if (save_audio_config_p->achan[chan].fix_bits != RETRY_NONE) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Channel %d: FIX_BITS option has been turned off for EAS.\n", chan); + save_audio_config_p->achan[chan].fix_bits = RETRY_NONE; + } + if (save_audio_config_p->achan[chan].passall != 0) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Channel %d: PASSALL option has been turned off for EAS.\n", chan); + save_audio_config_p->achan[chan].passall = 0; + } + } /* * Tear apart the profile and put it back together in a normalized form: @@ -187,46 +194,57 @@ int demod_init (struct audio_s *pa) assert (num_letters == (int)(strlen(just_letters))); /* - * Pick a good default demodulator if none specified. + * Pick a good default demodulator if none specified. + * Previously, we had "D" optimized for 300 bps. + * Gone in 1.7 so it is always "A+". */ if (num_letters == 0) { + strlcpy (just_letters, "A", sizeof(just_letters)); + num_letters = strlen(just_letters); - if (save_audio_config_p->achan[chan].baud < 600) { - - /* This has been optimized for 300 baud. */ - - strlcpy (just_letters, "D", sizeof(just_letters)); + if (have_plus != -1) have_plus = 1; // Add as default for version 1.2 + // If not explicitly turned off. + } - } - else { +/* + * Special case for ARM. + * The higher end ARM chips have loads of power but many people + * are using a single core Pi Zero or similar. + * (I'm still using a model 1 for my digipeater/IGate!) + * Decreasing CPU requirement has a negligible impact on decoding performance. + * + * atest -PA- 01_Track_1.wav --> 1002 packets decoded. + * atest -PA- -D3 01_Track_1.wav --> 997 packets decoded. + * + * Someone concerned about 1/2 of one percent difference can add "-D 1" + */ #if __arm__ - /* We probably don't have a lot of CPU power available. */ - /* Previously we would use F if possible otherwise fall back to A. */ - - /* In version 1.2, new default is E+ /3. */ - strlcpy (just_letters, "E", sizeof(just_letters)); // version 1.2 now E. - if (have_plus != -1) have_plus = 1; // Add as default for version 1.2 - // If not explicitly turned off. - if (save_audio_config_p->achan[chan].decimate == 0) { - if (save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec > 40000) { - save_audio_config_p->achan[chan].decimate = 3; - } - } -#else - strlcpy (just_letters, "E", sizeof(just_letters)); // version 1.2 changed C to E. - if (have_plus != -1) have_plus = 1; // Add as default for version 1.2 - // If not explicitly turned off. -#endif + if (save_audio_config_p->achan[chan].decimate == 0) { + if (save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec > 40000) { + save_audio_config_p->achan[chan].decimate = 3; } - num_letters = 1; } +#endif +/* + * Number of filter taps is proportional to number of audio samples in a "symbol" duration. + * These can get extremely large for low speeds, e.g. 300 baud. + * In this case, increase the decimation ration. Crude approximation. Could be improved. + */ + if (save_audio_config_p->achan[chan].decimate == 0 && + save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec > 40000 && + save_audio_config_p->achan[chan].baud < 600) { + + // Avoid enormous number of filter taps. + + save_audio_config_p->achan[chan].decimate = 3; + } - assert (num_letters == (int)(strlen(just_letters))); /* * Put it back together again. */ + assert (num_letters == (int)(strlen(just_letters))); /* At this point, have_plus can have 3 values: */ /* 1 = turned on, either explicitly or by applied default */ @@ -275,7 +293,7 @@ int demod_init (struct audio_s *pa) if (save_audio_config_p->achan[chan].decimate == 0) { save_audio_config_p->achan[chan].decimate = 1; - if (strchr (just_letters, 'D') != NULL && save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec > 40000) { + if (strchr (just_letters, 'B') != NULL && save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec > 40000) { save_audio_config_p->achan[chan].decimate = 3; } } @@ -315,48 +333,6 @@ int demod_init (struct audio_s *pa) save_audio_config_p->achan[chan].num_subchan = num_letters; -/* - * Quick hack with special case for another experiment. - * Do this in a more general way if it turns out to be useful. - */ - save_audio_config_p->achan[chan].interleave = 1; - if (strcasecmp(save_audio_config_p->achan[chan].profiles, "EE") == 0) { - save_audio_config_p->achan[chan].interleave = 2; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "EEE") == 0) { - save_audio_config_p->achan[chan].interleave = 3; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "EEEE") == 0) { - save_audio_config_p->achan[chan].interleave = 4; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "EEEEE") == 0) { - save_audio_config_p->achan[chan].interleave = 5; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "GG") == 0) { - save_audio_config_p->achan[chan].interleave = 2; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "GGG") == 0) { - save_audio_config_p->achan[chan].interleave = 3; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "GGG+") == 0) { - save_audio_config_p->achan[chan].interleave = 3; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "GGGG") == 0) { - save_audio_config_p->achan[chan].interleave = 4; - save_audio_config_p->achan[chan].decimate = 1; - } - else if (strcasecmp(save_audio_config_p->achan[chan].profiles, "GGGGG") == 0) { - save_audio_config_p->achan[chan].interleave = 5; - save_audio_config_p->achan[chan].decimate = 1; - } - if (save_audio_config_p->achan[chan].num_subchan != num_letters) { text_color_set(DW_COLOR_ERROR); dw_printf ("INTERNAL ERROR, %s:%d, chan=%d, num_subchan(%d) != strlen(\"%s\")\n", @@ -385,7 +361,7 @@ int demod_init (struct audio_s *pa) dw_printf (" %d.%d: %c %d & %d\n", chan, d, profile, mark, space); } - demod_afsk_init (save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec / (save_audio_config_p->achan[chan].decimate * save_audio_config_p->achan[chan].interleave), + demod_afsk_init (save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec / save_audio_config_p->achan[chan].decimate, save_audio_config_p->achan[chan].baud, mark, space, @@ -520,6 +496,31 @@ int demod_init (struct audio_s *pa) case MODEM_QPSK: // New for 1.4 + // In versions 1.4 and 1.5, V.26 "Alternative A" was used. + // years later, I discover that the MFJ-2400 used "Alternative B." + // It looks like the other two manufacturers use the same but we + // can't be sure until we find one for compatibility testing. + + // In version 1.6 we add a choice for the user. + // If neither one was explicitly specified, print a message and take + // a default. My current thinking is that we default to direwolf <= 1.5 + // compatible for version 1.6 and MFJ compatible after that. + + if (save_audio_config_p->achan[chan].v26_alternative == V26_UNSPECIFIED) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Two incompatible versions of 2400 bps QPSK are now available.\n"); + dw_printf ("For compatibility with direwolf <= 1.5, use 'V26A' modem option in config file.\n"); + dw_printf ("For compatibility MFJ-2400 use 'V26B' modem option in config file.\n"); + dw_printf ("Command line options -j and -J can be used for channel 0.\n"); + dw_printf ("For more information, read the Dire Wolf User Guide and\n"); + dw_printf ("2400-4800-PSK-for-APRS-Packet-Radio.pdf.\n"); + dw_printf ("The default is now MFJ-2400 compatibility mode.\n"); + + save_audio_config_p->achan[chan].v26_alternative = V26_DEFAULT; + } + + // TODO: See how much CPU this takes on ARM and decide if we should have different defaults. if (strlen(save_audio_config_p->achan[chan].profiles) == 0) { @@ -539,6 +540,12 @@ int demod_init (struct audio_s *pa) save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec); if (save_audio_config_p->achan[chan].decimate != 1) dw_printf (" / %d", save_audio_config_p->achan[chan].decimate); + + if (save_audio_config_p->achan[chan].v26_alternative == V26_B) + dw_printf (", compatible with MFJ-2400"); + else + dw_printf (", compatible with earlier direwolf"); + if (save_audio_config_p->achan[chan].dtmf_decode != DTMF_DECODE_OFF) dw_printf (", DTMF decoder enabled"); dw_printf (".\n"); @@ -556,6 +563,7 @@ int demod_init (struct audio_s *pa) // save_audio_config_p->achan[chan].modem_type, profile); demod_psk_init (save_audio_config_p->achan[chan].modem_type, + save_audio_config_p->achan[chan].v26_alternative, save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec / save_audio_config_p->achan[chan].decimate, save_audio_config_p->achan[chan].baud, profile, @@ -610,6 +618,7 @@ int demod_init (struct audio_s *pa) // save_audio_config_p->achan[chan].modem_type, profile); demod_psk_init (save_audio_config_p->achan[chan].modem_type, + save_audio_config_p->achan[chan].v26_alternative, save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec / save_audio_config_p->achan[chan].decimate, save_audio_config_p->achan[chan].baud, profile, @@ -630,34 +639,103 @@ int demod_init (struct audio_s *pa) case MODEM_BASEBAND: case MODEM_SCRAMBLE: + case MODEM_AIS: default: /* Not AFSK */ { + // For AIS we will accept only a good CRC without any fixup attempts. + // Even with that, there are still a lot of CRC false matches with random noise. + + if (save_audio_config_p->achan[chan].modem_type == MODEM_AIS) { + if (save_audio_config_p->achan[chan].fix_bits != RETRY_NONE) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Channel %d: FIX_BITS option has been turned off for AIS.\n", chan); + save_audio_config_p->achan[chan].fix_bits = RETRY_NONE; + } + if (save_audio_config_p->achan[chan].passall != 0) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Channel %d: PASSALL option has been turned off for AIS.\n", chan); + save_audio_config_p->achan[chan].passall = 0; + } + } + if (strcmp(save_audio_config_p->achan[chan].profiles, "") == 0) { /* Apply default if not set earlier. */ /* Not sure if it should be on for ARM too. */ /* Need to take a look at CPU usage and performance difference. */ -#ifndef __arm__ + /* Version 1.5: Remove special case for ARM. */ + /* We want higher performance to be the default. */ + /* "MODEM 9600 -" can be used on very slow CPU if necessary. */ + strlcpy (save_audio_config_p->achan[chan].profiles, "+", sizeof(save_audio_config_p->achan[chan].profiles)); -#endif } -#ifdef TUNE_UPSAMPLE - upsample = TUNE_UPSAMPLE; -#endif +/* + * We need a minimum number of audio samples per bit time for good performance. + * Easier to check here because demod_9600_init might have an adjusted sample rate. + */ + + float ratio = (float)(save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec) + / (float)(save_audio_config_p->achan[chan].baud); + +/* + * Set reasonable upsample ratio if user did not override. + */ + + if (save_audio_config_p->achan[chan].upsample == 0) { + + if (ratio < 4) { + // This is extreme. + // No one should be using a sample rate this low but + // amazingly a recording with 22050 rate can be decoded. + // 3 and 4 are the same. Need more tests. + + save_audio_config_p->achan[chan].upsample = 4; + } + else if (ratio < 5) { + + // example: 44100 / 9600 is 4.59 + // 3 is slightly better than 2 or 4. + + save_audio_config_p->achan[chan].upsample = 3; + } + else if (ratio < 10) { + + // example: 48000 / 9600 = 5 + // 3 is slightly better than 2 or 4. + + save_audio_config_p->achan[chan].upsample = 3; + } + else if (ratio < 15) { + + // ... guessing + + save_audio_config_p->achan[chan].upsample = 2; + } + else { // >= 15 + // + // An example of this might be ..... + // Probably no benefit. + + save_audio_config_p->achan[chan].upsample = 1; + } + } -#ifdef TUNE_ZEROSTUFF - zerostuff = TUNE_ZEROSTUFF; +#ifdef TUNE_UPSAMPLE + save_audio_config_p->achan[chan].upsample = TUNE_UPSAMPLE; #endif text_color_set(DW_COLOR_DEBUG); - dw_printf ("Channel %d: %d baud, K9NG/G3RUH, %s, %d sample rate x %d", - chan, save_audio_config_p->achan[chan].baud, + dw_printf ("Channel %d: %d baud, %s, %s, %d sample rate x %d", + chan, + save_audio_config_p->achan[chan].baud, + save_audio_config_p->achan[chan].modem_type == MODEM_AIS ? "AIS" : "K9NG/G3RUH", save_audio_config_p->achan[chan].profiles, - save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec, upsample); + save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec, + save_audio_config_p->achan[chan].upsample); if (save_audio_config_p->achan[chan].dtmf_decode != DTMF_DECODE_OFF) dw_printf (", DTMF decoder enabled"); dw_printf (".\n"); @@ -665,6 +743,7 @@ int demod_init (struct audio_s *pa) struct demodulator_state_s *D; D = &demodulator_state[chan][0]; // first subchannel + save_audio_config_p->achan[chan].num_subchan = 1; save_audio_config_p->achan[chan].num_slicers = 1; @@ -677,12 +756,6 @@ int demod_init (struct audio_s *pa) } - /* We need a minimum number of audio samples per bit time for good performance. */ - /* Easier to check here because demod_9600_init might have an adjusted sample rate. */ - - float ratio = (float)(save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec) - / (float)(save_audio_config_p->achan[chan].baud); - text_color_set(DW_COLOR_INFO); dw_printf ("The ratio of audio samples per sec (%d) to data rate in baud (%d) is %.1f\n", save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec, @@ -694,6 +767,9 @@ int demod_init (struct audio_s *pa) } else if (ratio < 5) { dw_printf ("This is on the low side for best performance. Can you use a higher sample rate?\n"); + if (save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec == 44100) { + dw_printf ("For example, can you use 48000 rather than 44100?\n"); + } } else if (ratio < 6) { dw_printf ("Increasing the sample rate should improve decoder performance.\n"); @@ -705,7 +781,10 @@ int demod_init (struct audio_s *pa) dw_printf ("This is a suitable ratio for good performance.\n"); } - demod_9600_init (upsample * save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec, save_audio_config_p->achan[chan].baud, D); + demod_9600_init (save_audio_config_p->achan[chan].modem_type, + save_audio_config_p->adev[ACHAN2ADEV(chan)].samples_per_sec, + save_audio_config_p->achan[chan].upsample, + save_audio_config_p->achan[chan].baud, D); if (strchr(save_audio_config_p->achan[chan].profiles, '+') != NULL) { @@ -725,10 +804,23 @@ int demod_init (struct audio_s *pa) } /* switch on modulation type. */ - } /* if channel number is valid */ + } /* if channel medium is radio */ + +// FIXME dw_printf ("-------- end of loop for chn %d \n", chan); } /* for chan ... */ + // Now the virtual channels. FIXME: could be single loop. + + for (chan = MAX_CHANS; chan < MAX_TOTAL_CHANS; chan++) { + +// FIXME dw_printf ("-------- virtual channel loop %d \n", chan); + + if (chan == save_audio_config_p->igate_vchannel) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Channel %d: IGate virtual channel.\n", chan); + } + } return (0); @@ -740,7 +832,7 @@ int demod_init (struct audio_s *pa) * * Name: demod_get_sample * - * Purpose: Get one audio sample fromt the specified sound input source. + * Purpose: Get one audio sample from the specified sound input source. * * Inputs: a - Index for audio device. 0 = first. * @@ -750,7 +842,7 @@ int demod_init (struct audio_s *pa) * Global In: save_audio_config_p->adev[ACHAN2ADEV(chan)].bits_per_sample - So we know whether to * read 1 or 2 bytes from audio stream. * - * Description: Grab 1 or two btyes depending on data source. + * Description: Grab 1 or two bytes depending on data source. * * When processing stereo, the caller will call this * at twice the normal rate to obtain alternating left @@ -760,17 +852,15 @@ int demod_init (struct audio_s *pa) #define FSK_READ_ERR (256*256) - __attribute__((hot)) int demod_get_sample (int a) { int x1, x2; - signed short sam; /* short to force sign extention. */ + signed short sam; /* short to force sign extension. */ assert (save_audio_config_p->adev[a].bits_per_sample == 8 || save_audio_config_p->adev[a].bits_per_sample == 16); - if (save_audio_config_p->adev[a].bits_per_sample == 8) { x1 = audio_get(a); @@ -837,12 +927,27 @@ int demod_get_sample (int a) * *--------------------------------------------------------------------*/ +static volatile int mute_input[MAX_CHANS]; + +// New in 1.7. +// A few people have a really bad audio cross talk situation where they receive their own transmissions. +// It usually doesn't cause a problem but it is confusing to look at. +// "half duplex" setting applied only to the transmit logic. i.e. wait for clear channel before sending. +// Receiving was still active. +// I think the simplest solution is to mute/unmute the audio input at this point if not full duplex. +// This is called from ptt_set for half duplex. + +void demod_mute_input (int chan, int mute_during_xmit) +{ + assert (chan >= 0 && chan < MAX_CHANS); + mute_input[chan] = mute_during_xmit; +} __attribute__((hot)) void demod_process_sample (int chan, int subchan, int sam) { float fsam; - int k; + //int k; struct demodulator_state_s *D; @@ -850,6 +955,10 @@ void demod_process_sample (int chan, int subchan, int sam) assert (chan >= 0 && chan < MAX_CHANS); assert (subchan >= 0 && subchan < MAX_SUBCHANS); + if (mute_input[chan]) { + sam = 0; + }; + D = &demodulator_state[chan][subchan]; @@ -897,6 +1006,7 @@ void demod_process_sample (int chan, int subchan, int sam) break; case MODEM_AFSK: + case MODEM_EAS: if (save_audio_config_p->achan[chan].decimate > 1) { @@ -930,49 +1040,10 @@ void demod_process_sample (int chan, int subchan, int sam) case MODEM_BASEBAND: case MODEM_SCRAMBLE: + case MODEM_AIS: default: - if (zerostuff) { - /* Literature says this is better if followed */ - /* by appropriate low pass filter. */ - /* So far, both are same in tests with different */ - /* optimal low pass filter parameters. */ - - for (k=1; kachan[chan].upsample, D); break; } /* switch modem_type */ @@ -1012,7 +1083,8 @@ alevel_t demod_get_audio_level (int chan, int subchan) alevel.rec = (int) (( D->alevel_rec_peak - D->alevel_rec_valley ) * 50.0f + 0.5f); - if (save_audio_config_p->achan[chan].modem_type == MODEM_AFSK) { + if (save_audio_config_p->achan[chan].modem_type == MODEM_AFSK || + save_audio_config_p->achan[chan].modem_type == MODEM_EAS) { /* For AFSK, we have mark and space amplitudes. */ diff --git a/demod.h b/src/demod.h similarity index 73% rename from demod.h rename to src/demod.h index 3233b9ba..f1120cd0 100644 --- a/demod.h +++ b/src/demod.h @@ -8,10 +8,13 @@ int demod_init (struct audio_s *pa); +void demod_mute_input (int chan, int mute); + int demod_get_sample (int a); void demod_process_sample (int chan, int subchan, int sam); void demod_print_agc (int chan, int subchan); -alevel_t demod_get_audio_level (int chan, int subchan); \ No newline at end of file +alevel_t demod_get_audio_level (int chan, int subchan); + diff --git a/demod_9600.c b/src/demod_9600.c similarity index 64% rename from demod_9600.c rename to src/demod_9600.c index 58c4c047..705d1fa7 100644 --- a/demod_9600.c +++ b/src/demod_9600.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2015 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2015, 2019, 2021 John Langner, WB2OSZ // // 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 @@ -25,7 +25,8 @@ * * Module: demod_9600.c * - * Purpose: Demodulator for scrambled baseband encoding. + * Purpose: Demodulator for baseband signal. + * This is used for AX.25 (with scrambling) and IL2P without. * * Input: Audio samples from either a file or the "sound card." * @@ -44,14 +45,26 @@ #include #include +// Fine tuning for different demodulator types. +// Don't remove this section. It is here for a reason. + +#define DCD_THRESH_ON 32 // Hysteresis: Can miss 0 out of 32 for detecting lock. + // This is best for actual on-the-air signals. + // Still too many brief false matches. +#define DCD_THRESH_OFF 8 // Might want a little more fine tuning. +#define DCD_GOOD_WIDTH 1024 // No more than 1024!!! + +#include "fsk_demod_state.h" // Values above override defaults. + #include "tune.h" -#include "fsk_demod_state.h" #include "hdlc_rec.h" #include "demod_9600.h" #include "textcolor.h" #include "dsp.h" + + static float slice_point[MAX_SUBCHANS]; @@ -113,9 +126,14 @@ static inline float agc (float in, float fast_attack, float slow_decay, float *p * * Purpose: Initialize the 9600 (or higher) baud demodulator. * - * Inputs: samples_per_sec - Number of samples per second. - * Might be upsampled in hopes of - * reducing the PLL jitter. + * Inputs: modem_type - Determines whether scrambling is used. + * + * samples_per_sec - Number of samples per second for audio. + * + * upsample - Factor to upsample the incoming stream. + * After a lot of experimentation, I discovered that + * it works better if the data is upsampled. + * This reduces the jitter for PLL synchronization. * * baud - Data rate in bits per second. * @@ -125,12 +143,16 @@ static inline float agc (float in, float fast_attack, float slow_decay, float *p * *----------------------------------------------------------------*/ -void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s *D) +void demod_9600_init (enum modem_t modem_type, int original_sample_rate, int upsample, int baud, struct demodulator_state_s *D) { float fc; int j; + if (upsample < 1) upsample = 1; + if (upsample > 4) upsample = 4; + memset (D, 0, sizeof(struct demodulator_state_s)); + D->modem_type = modem_type; D->num_slicers = 1; // Multiple profiles in future? @@ -141,25 +163,43 @@ void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s // case 'K': // upsample x3 with filtering. // case 'L': // upsample x4 with filtering. - D->lp_filter_len_bits = 76 * 9600.0 / (44100.0 * 2.0); + + D->lp_filter_len_bits = 1.0; // -U4 = 61 4.59 samples/symbol // Works best with odd number in some tests. Even is better in others. - //D->lp_filter_size = ((int) (0.5f * ( D->lp_filter_len_bits * (float)samples_per_sec / (float)baud ))) * 2 + 1; - D->lp_filter_size = (int) (( D->lp_filter_len_bits * (float)samples_per_sec / baud) + 0.5f); + //D->lp_filter_size = ((int) (0.5f * ( D->lp_filter_len_bits * (float)original_sample_rate / (float)baud ))) * 2 + 1; - D->lp_window = BP_WINDOW_HAMMING; - D->lpf_baud = 0.62; + // Just round to nearest integer. + D->lp_filter_size = (int) (( D->lp_filter_len_bits * (float)original_sample_rate / baud) + 0.5f); + + D->lp_window = BP_WINDOW_COSINE; + + D->lpf_baud = 1.00; D->agc_fast_attack = 0.080; D->agc_slow_decay = 0.00012; D->pll_locked_inertia = 0.89; D->pll_searching_inertia = 0.67; + // break; // } +#if 0 + text_color_set(DW_COLOR_DEBUG); + dw_printf ("---------- %s (%d, %d) -----------\n", __func__, samples_per_sec, baud); + dw_printf ("filter_len_bits = %.2f\n", D->lp_filter_len_bits); + dw_printf ("lp_filter_size = %d\n", D->lp_filter_size); + dw_printf ("lp_window = %d\n", D->lp_window); + dw_printf ("lpf_baud = %.2f\n", D->lpf_baud); + dw_printf ("samples per bit = %.1f\n", (double)samples_per_sec / baud); +#endif + + + // PLL needs to use the upsampled rate. + D->pll_step_per_sample = - (int) round(TICKS_PER_PLL_CYCLE * (double) baud / (double)samples_per_sec); + (int) round(TICKS_PER_PLL_CYCLE * (double) baud / (double)(original_sample_rate * upsample)); #ifdef TUNE_LP_WINDOW @@ -190,13 +230,87 @@ void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s D->pll_searching_inertia = TUNE_PLL_SEARCHING; #endif - fc = (float)baud * D->lpf_baud / (float)samples_per_sec; + // Initial filter (before scattering) is based on upsampled rate. + + fc = (float)baud * D->lpf_baud / (float)(original_sample_rate * upsample); //dw_printf ("demod_9600_init: call gen_lowpass(fc=%.2f, , size=%d, )\n", fc, D->lp_filter_size); - gen_lowpass (fc, D->lp_filter, D->lp_filter_size, D->lp_window); + gen_lowpass (fc, D->u.bb.lp_filter, D->lp_filter_size * upsample, D->lp_window); + +// New in 1.7 - +// Use a polyphase filter to reduce the CPU load. +// Originally I used zero stuffing to upsample. +// Here is the general idea. +// +// Suppose the input samples are 1 2 3 4 5 6 7 8 9 ... +// Filter coefficients are a b c d e f g h i ... +// +// With original sampling rate, the filtering would involve multiplying and adding: +// +// 1a 2b 3c 4d 5e 6f ... +// +// When upsampling by 3, each of these would need to be evaluated +// for each audio sample: +// +// 1a 0b 0c 2d 0e 0f 3g 0h 0i ... +// 0a 1b 0c 0d 2e 0f 0g 3h 0i ... +// 0a 0b 1c 0d 0e 2f 0g 0h 3i ... +// +// 2/3 of the multiplies are always by a stuffed zero. +// We can do this more efficiently by removing them. +// +// 1a 2d 3g ... +// 1b 2e 3h ... +// 1c 2f 3i ... +// +// We scatter the original filter across multiple shorter filters. +// Each input sample cycles around them to produce the upsampled rate. +// +// a d g ... +// b e h ... +// c f i ... +// +// There are countless sources of information DSP but this one is unique +// in that it is a college course that mentions APRS. +// https://www2.eecs.berkeley.edu/Courses/EE123 +// +// Was the effort worthwhile? Times on an RPi 3. +// +// command: atest -B9600 ~/walkabout9600[abc]-compressed*.wav +// +// These are 3 recordings of a portable system being carried out of +// range and back in again. It is a real world test for weak signals. +// +// options num decoded seconds x realtime +// 1.6 1.7 1.6 1.7 1.6 1.7 +// --- --- --- --- --- --- +// -P- 171 172 23.928 17.967 14.9 19.9 +// -P+ 180 180 54.688 48.772 6.5 7.3 +// -P- -F1 177 178 32.686 26.517 10.9 13.5 +// +// So, it turns out that -P+ doesn't have a dramatic improvement, only +// around 4%, for drastically increased CPU requirements. +// Maybe we should turn that off by default, especially for ARM. +// + + int k = 0; + for (int i = 0; i < D->lp_filter_size; i++) { + D->u.bb.lp_polyphase_1[i] = D->u.bb.lp_filter[k++]; + if (upsample >= 2) { + D->u.bb.lp_polyphase_2[i] = D->u.bb.lp_filter[k++]; + if (upsample >= 3) { + D->u.bb.lp_polyphase_3[i] = D->u.bb.lp_filter[k++]; + if (upsample >= 4) { + D->u.bb.lp_polyphase_4[i] = D->u.bb.lp_filter[k++]; + } + } + } + } + /* Version 1.2: Experiment with different slicing levels. */ + // Really didn't help that much because we should have a symmetrical signal. for (j = 0; j < MAX_SUBCHANS; j++) { slice_point[j] = 0.02f * (j - 0.5f * (MAX_SUBCHANS-1)); @@ -232,7 +346,7 @@ void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s * been distorted by going thru voice transceivers not * intended to pass this sort of "audio" signal. * - * Data is "scrambled" to reduce the amount of DC bias. + * For G3RUH mode, data is "scrambled" to reduce the amount of DC bias. * The data stream must be unscrambled at the receiving end. * * We also have a digital phase locked loop (PLL) @@ -249,6 +363,9 @@ void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s * of the function to be called for each bit recovered * from the demodulator. For now, it's simply hard-coded. * + * After experimentation, I found that this works better if + * the original signal is upsampled by 2x or even 4x. + * * References: 9600 Baud Packet Radio Modem Design * http://www.amsat.org/amsat/articles/g3ruh/109.html * @@ -263,63 +380,57 @@ void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s inline static void nudge_pll (int chan, int subchan, int slice, float demod_out, struct demodulator_state_s *D); +static void process_filtered_sample (int chan, float fsam, struct demodulator_state_s *D); + + __attribute__((hot)) -void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D) +void demod_9600_process_sample (int chan, int sam, int upsample, struct demodulator_state_s *D) { - float fsam; - float amp; - float demod_out; #if DEBUG4 static FILE *demod_log_fp = NULL; static int log_file_seq = 0; /* Part of log file name */ #endif - int subchan = 0; - int demod_data; /* Still scrambled. */ - assert (chan >= 0 && chan < MAX_CHANS); assert (subchan >= 0 && subchan < MAX_SUBCHANS); - -/* - * Filters use last 'filter_size' samples. - * - * First push the older samples down. - * - * Finally, put the most recent at the beginning. - * - * Future project? Rather than shifting the samples, - * it might be faster to add another variable to keep - * track of the most recent sample and change the - * indexing in the later loops that multipy and add. - */ - /* Scale to nice number for convenience. */ /* Consistent with the AFSK demodulator, we'd like to use */ /* only half of the dynamic range to have some headroom. */ /* i.e. input range +-16k becomes +-1 here and is */ /* displayed in the heard line as audio level 100. */ - fsam = sam / 16384.0; - -#if defined(TUNE_ZEROSTUFF) && TUNE_ZEROSTUFF == 0 -// experiment - no filtering. - - amp = fsam; + fsam = (float)sam / 16384.0f; + + // Low pass filter + push_sample (fsam, D->u.bb.audio_in, D->lp_filter_size); + + fsam = convolve (D->u.bb.audio_in, D->u.bb.lp_polyphase_1, D->lp_filter_size); + process_filtered_sample (chan, fsam, D); + if (upsample >= 2) { + fsam = convolve (D->u.bb.audio_in, D->u.bb.lp_polyphase_2, D->lp_filter_size); + process_filtered_sample (chan, fsam, D); + if (upsample >= 3) { + fsam = convolve (D->u.bb.audio_in, D->u.bb.lp_polyphase_3, D->lp_filter_size); + process_filtered_sample (chan, fsam, D); + if (upsample >= 4) { + fsam = convolve (D->u.bb.audio_in, D->u.bb.lp_polyphase_4, D->lp_filter_size); + process_filtered_sample (chan, fsam, D); + } + } + } +} -#else - push_sample (fsam, D->raw_cb, D->lp_filter_size); -/* - * Low pass filter to reduce noise yet pass the data. - */ +__attribute__((hot)) +static void process_filtered_sample (int chan, float fsam, struct demodulator_state_s *D) +{ - amp = convolve (D->raw_cb, D->lp_filter, D->lp_filter_size); -#endif + int subchan = 0; /* * Version 1.2: Capture the post-filtering amplitude for display. @@ -332,18 +443,18 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D // TODO: probably no need for this. Just use D->m_peak, D->m_valley - if (amp >= D->alevel_mark_peak) { - D->alevel_mark_peak = amp * D->quick_attack + D->alevel_mark_peak * (1.0f - D->quick_attack); + if (fsam >= D->alevel_mark_peak) { + D->alevel_mark_peak = fsam * D->quick_attack + D->alevel_mark_peak * (1.0f - D->quick_attack); } else { - D->alevel_mark_peak = amp * D->sluggish_decay + D->alevel_mark_peak * (1.0f - D->sluggish_decay); + D->alevel_mark_peak = fsam * D->sluggish_decay + D->alevel_mark_peak * (1.0f - D->sluggish_decay); } - if (amp <= D->alevel_space_peak) { - D->alevel_space_peak = amp * D->quick_attack + D->alevel_space_peak * (1.0f - D->quick_attack); + if (fsam <= D->alevel_space_peak) { + D->alevel_space_peak = fsam * D->quick_attack + D->alevel_space_peak * (1.0f - D->quick_attack); } else { - D->alevel_space_peak = amp * D->sluggish_decay + D->alevel_space_peak * (1.0f - D->sluggish_decay); + D->alevel_space_peak = fsam * D->sluggish_decay + D->alevel_space_peak * (1.0f - D->sluggish_decay); } /* @@ -354,12 +465,14 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D * This works by looking at the minimum and maximum signal peaks * and scaling the results to be roughly in the -1.0 to +1.0 range. */ + float demod_out; + int demod_data; /* Still scrambled. */ - demod_out = agc (amp, D->agc_fast_attack, D->agc_slow_decay, &(D->m_peak), &(D->m_valley)); + demod_out = agc (fsam, D->agc_fast_attack, D->agc_slow_decay, &(D->m_peak), &(D->m_valley)); // TODO: There is potential for multiple decoders with one filter. -//dw_printf ("peak=%.2f valley=%.2f amp=%.2f norm=%.2f\n", D->m_peak, D->m_valley, amp, norm); +//dw_printf ("peak=%.2f valley=%.2f fsam=%.2f norm=%.2f\n", D->m_peak, D->m_valley, fsam, norm); if (D->num_slicers <= 1) { @@ -390,7 +503,7 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D if (chan == 0) { if (1) { - //if (hdlc_rec_gathering (chan, subchan, slice)) { + //if (D->slicer[slice].data_detect) { char fname[30]; int slice = 0; @@ -408,7 +521,7 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D fprintf (demod_log_fp, "%.3f, %.3f, %.3f, %.3f, %.3f, %d, %.2f\n", fsam + 6, - amp + 4, + fsam + 4, D->m_peak + 4, D->m_valley + 4, demod_out + 2, @@ -451,7 +564,7 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D * * Returns: None * - * Descripton: A PLL is used to sample near the centers of the data bits. + * Description: A PLL is used to sample near the centers of the data bits. * * D->data_clock_pll is a SIGNED 32 bit variable. * When it overflows from a large positive value to a negative value, we @@ -481,23 +594,25 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D * * Results??? TBD * + * Version 1.6: New experiment where filter size to extract clock is not the same + * as filter to extract the data bit value. + * *--------------------------------------------------------------------*/ __attribute__((hot)) inline static void nudge_pll (int chan, int subchan, int slice, float demod_out_f, struct demodulator_state_s *D) { - -/* - */ - D->slicer[slice].prev_d_c_pll = D->slicer[slice].data_clock_pll; - D->slicer[slice].data_clock_pll += D->pll_step_per_sample; + + // Perform the add as unsigned to avoid signed overflow error. + D->slicer[slice].data_clock_pll = (signed)((unsigned)(D->slicer[slice].data_clock_pll) + (unsigned)(D->pll_step_per_sample)); if ( D->slicer[slice].prev_d_c_pll > 1000000000 && D->slicer[slice].data_clock_pll < -1000000000) { /* Overflow. Was large positive, wrapped around, now large negative. */ - hdlc_rec_bit (chan, subchan, slice, demod_out_f > 0, 1, D->slicer[slice].lfsr); + hdlc_rec_bit (chan, subchan, slice, demod_out_f > 0, D->modem_type == MODEM_SCRAMBLE, D->slicer[slice].lfsr); + pll_dcd_each_symbol2 (D, chan, subchan, slice); } /* @@ -508,11 +623,11 @@ inline static void nudge_pll (int chan, int subchan, int slice, float demod_out_ // Note: Test for this demodulator, not overall for channel. - float target = 0; + pll_dcd_signal_transition2 (D, slice, D->slicer[slice].data_clock_pll); - target = D->pll_step_per_sample * demod_out_f / (demod_out_f - D->slicer[slice].prev_demod_out_f); + float target = D->pll_step_per_sample * demod_out_f / (demod_out_f - D->slicer[slice].prev_demod_out_f); - if (hdlc_rec_gathering (chan, subchan, slice)) { + if (D->slicer[slice].data_detect) { D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_locked_inertia + target * (1.0f - D->pll_locked_inertia) ); } else { @@ -524,7 +639,7 @@ inline static void nudge_pll (int chan, int subchan, int slice, float demod_out_ #if DEBUG5 //if (chan == 0) { - if (hdlc_rec_gathering (chan,subchan,slice)) { + if (D->slicer[slice].data_detect) { char fname[30]; diff --git a/demod_9600.h b/src/demod_9600.h similarity index 54% rename from demod_9600.h rename to src/demod_9600.h index a7647112..51fc15e4 100644 --- a/demod_9600.h +++ b/src/demod_9600.h @@ -6,9 +6,9 @@ #include "fsk_demod_state.h" -void demod_9600_init (int samples_per_sec, int baud, struct demodulator_state_s *D); +void demod_9600_init (enum modem_t modem_type, int original_sample_rate, int upsample, int baud, struct demodulator_state_s *D); -void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D); +void demod_9600_process_sample (int chan, int sam, int upsample, struct demodulator_state_s *D); diff --git a/src/demod_afsk.c b/src/demod_afsk.c new file mode 100644 index 00000000..b4d6c295 --- /dev/null +++ b/src/demod_afsk.c @@ -0,0 +1,944 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2020 John Langner, WB2OSZ +// +// 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, see . +// + + +// #define DEBUG1 1 /* display debugging info */ + +// #define DEBUG3 1 /* print carrier detect changes. */ + +// #define DEBUG4 1 /* capture AFSK demodulator output to log files */ + /* Can be used to make nice plots. */ + +// #define DEBUG5 1 // Write just demodulated bit stream to file. */ + + +/*------------------------------------------------------------------ + * + * Module: demod_afsk.c + * + * Purpose: Demodulator for Audio Frequency Shift Keying (AFSK). + * + * Input: Audio samples from either a file or the "sound card." + * + * Outputs: Calls hdlc_rec_bit() for each bit demodulated. + * + *---------------------------------------------------------------*/ + +#include "direwolf.h" + +#include +#include +#include +#include +#include +#include +#include +#include + +#include "audio.h" +#include "tune.h" +#include "fsk_demod_state.h" +#include "fsk_gen_filter.h" +#include "hdlc_rec.h" +#include "textcolor.h" +#include "demod_afsk.h" +#include "dsp.h" + +#define MIN(a,b) ((a)<(b)?(a):(b)) +#define MAX(a,b) ((a)>(b)?(a):(b)) + +#define TUNE(envvar,param,name,fmt) { \ + char *e = getenv(envvar); \ + if (e != NULL) { \ + param = atof(e); \ + text_color_set (DW_COLOR_ERROR); \ + dw_printf ("TUNE: " name " = " fmt "\n", param); \ + } } + + +// Cosine table indexed by unsigned byte. +static float fcos256_table[256]; + +#define fcos256(x) (fcos256_table[((x)>>24)&0xff]) +#define fsin256(x) (fcos256_table[(((x)>>24)-64)&0xff]) + +static void nudge_pll (int chan, int subchan, int slice, float demod_out, struct demodulator_state_s *D, float amplitude); + + +/* Quick approximation to sqrt(x*x + y*y) */ +/* No benefit for regular PC. */ +/* Might help with microcomputer platform??? */ + +__attribute__((hot)) __attribute__((always_inline)) +static inline float fast_hypot(float x, float y) +{ +#if 0 + x = fabsf(x); + y = fabsf(y); + + if (x > y) { + return (x * .941246f + y * .41f); + } + else { + return (y * .941246f + x * .41f); + } +#else + return (hypotf(x,y)); +#endif +} + + +/* Add sample to buffer and shift the rest down. */ + +__attribute__((hot)) __attribute__((always_inline)) +static inline void push_sample (float val, float *buff, int size) +{ + memmove(buff+1,buff,(size-1)*sizeof(float)); + buff[0] = val; +} + + +/* FIR filter kernel. */ + +__attribute__((hot)) __attribute__((always_inline)) +static inline float convolve (const float *__restrict__ data, const float *__restrict__ filter, int filter_taps) +{ + float sum = 0.0f; + int j; + +//#pragma GCC ivdep // ignored until gcc 4.9 + for (j=0; j= *ppeak) { + *ppeak = in * fast_attack + *ppeak * (1.0f - fast_attack); + } + else { + *ppeak = in * slow_decay + *ppeak * (1.0f - slow_decay); + } + + if (in <= *pvalley) { + *pvalley = in * fast_attack + *pvalley * (1.0f - fast_attack); + } + else { + *pvalley = in * slow_decay + *pvalley * (1.0f - slow_decay); + } + +#if 1 + float x = in; + if (x > *ppeak) x = *ppeak; // experiment: clip to envelope? + if (x < *pvalley) x = *pvalley; +#endif + if (*ppeak > *pvalley) { + + return ((x - 0.5f * (*ppeak + *pvalley)) / (*ppeak - *pvalley)); // my original AGC + + //return (( x - 0.5f * (*ppeak + *pvalley )) * ( *ppeak - *pvalley )); // see note below. + //return (x - 0.5f * (*ppeak + *pvalley)); // not as good either. + } + return (0.0f); +} + + +// K6JQ pointed me to this wonderful article: +// Improved Automatic Threshold Correction Methods for FSK by Kok Chen, W7AY. +// http://www.w7ay.net/site/Technical/ATC/index.html +// +// The stated problem is a little different, selective fading for HF RTTY, but the +// general idea is the similar: Compensating for imbalance of the two tones. +// +// The stronger tone probably has a better S/N ratio so we apply a larger +// weight to it. Effectively it is comparing power rather than amplitude. +// This is the optimal method from the article referenced. +// +// Interesting idea but it did not work as well as the original AGC in this case. +// For VHF FM we are not dealing with rapid deep selective fading of one tone. +// Instead we have an imbalance which is the same for the whole frame. +// It might be interesting to try this with HF SSB packet which is much like RTTY. +// +// I use the term valley rather than noise floor. +// After a little algebra, it looks remarkably similar to the function above. +// +// return (( x - valley ) * ( peak - valley ) - 0.5f * ( peak - valley ) * ( peak - valley )); +// return (( x - valley ) - 0.5f * ( peak - valley )) * ( peak - valley )); +// return (( x - 0.5f * (peak + valley )) * ( peak - valley )); + + + +/* + * for multi-slicer experiment. + */ + +#define MIN_G 0.5f +#define MAX_G 4.0f + +/* TODO: static */ float space_gain[MAX_SUBCHANS]; + + + +/*------------------------------------------------------------------ + * + * Name: demod_afsk_init + * + * Purpose: Initialization for an AFSK demodulator. + * Select appropriate parameters and set up filters. + * + * Inputs: samples_per_sec + * baud + * mark_freq + * space_freq + * + * D - Pointer to demodulator state for given channel. + * + * Outputs: + * + * Returns: None. + * + * Bugs: This doesn't do much error checking so don't give it + * anything crazy. + * + *----------------------------------------------------------------*/ + +void demod_afsk_init (int samples_per_sec, int baud, int mark_freq, + int space_freq, char profile, struct demodulator_state_s *D) +{ + + int j; + + for (j = 0; j < 256; j++) { + fcos256_table[j] = cosf((float)j * 2.0f * (float)M_PI / 256.0f); + } + + memset (D, 0, sizeof(struct demodulator_state_s)); + D->num_slicers = 1; + +#if DEBUG1 + dw_printf ("demod_afsk_init (rate=%d, baud=%d, mark=%d, space=%d, profile=%c\n", + samples_per_sec, baud, mark_freq, space_freq, profile); +#endif + D->profile = profile; + + switch (D->profile) { + + case 'A': // Official name + case 'E': // For compatibility during transition + + D->profile = 'A'; + + /* New in version 1.7 */ + /* This is a simpler version of what has been used all along. */ + /* Rather than convolving each sample with a pre-computed mark and */ + /* space filter, we have two free running local oscillators. */ + /* Also see if we can do better with a Root Raised Cosine filter */ + /* which supposedly reduces intersymbol interference. */ + + D->use_prefilter = 1; /* first, a bandpass filter. */ + + if (baud > 600) { + D->prefilter_baud = 0.155; + // Low cutoff below mark, high cutoff above space + // as fraction of the symbol rate. + // Intuitively you might expect this to be about + // half the symbol rate, e.g. 600 Hz outside + // the two tones of interest for 1200 baud. + // It turns out that narrower is better. + + D->pre_filter_len_sym = 383 * 1200. / 44100.; // about 8 symbols + D->pre_window = BP_WINDOW_TRUNCATED; + } + else { + D->prefilter_baud = 0.87; // TOTO: fine tune + D->pre_filter_len_sym = 1.857; + D->pre_window = BP_WINDOW_COSINE; + } + + // Local oscillators for Mark and Space tones. + + D->u.afsk.m_osc_phase = 0; + D->u.afsk.m_osc_delta = round ( pow(2., 32.) * (double)mark_freq / (double)samples_per_sec ); + + D->u.afsk.s_osc_phase = 0; + D->u.afsk.s_osc_delta = round ( pow(2., 32.) * (double)space_freq / (double)samples_per_sec ); + + D->u.afsk.use_rrc = 1; + TUNE("TUNE_USE_RRC", D->u.afsk.use_rrc, "use_rrc", "%d") + + if (D->u.afsk.use_rrc) { + D->u.afsk.rrc_width_sym = 2.80; + D->u.afsk.rrc_rolloff = 0.20; + } + else { + D->lpf_baud = 0.14; + D->lp_filter_width_sym = 1.388; + D->lp_window = BP_WINDOW_TRUNCATED; + } + + D->agc_fast_attack = 0.70; + D->agc_slow_decay = 0.000090; + + D->pll_locked_inertia = 0.74; + D->pll_searching_inertia = 0.50; + break; + + case 'B': // official name + case 'D': // backward compatibility + + D->profile = 'B'; + + // Experiment for version 1.7. + // Up to this point, I've always used separate mark and space + // filters and compared the amplitudes. + // Another technique for an FM demodulator is to mix with + // the center frequency and look for the rate of change of the phase. + + D->use_prefilter = 1; /* first, a bandpass filter. */ + + if (baud > 600) { + D->prefilter_baud = 0.19; + // Low cutoff below mark, high cutoff above space + // as fraction of the symbol rate. + // Intuitively you might expect this to be about + // half the symbol rate, e.g. 600 Hz outside + // the two tones of interest for 1200 baud. + // It turns out that narrower is better. + + D->pre_filter_len_sym = 8.163; // Filter length in symbol times. + D->pre_window = BP_WINDOW_TRUNCATED; + } + else { + D->prefilter_baud = 0.87; // TOTO: fine tune + D->pre_filter_len_sym = 1.857; + D->pre_window = BP_WINDOW_COSINE; + } + + // Local oscillator for Center frequency. + + D->u.afsk.c_osc_phase = 0; + D->u.afsk.c_osc_delta = round ( pow(2., 32.) * 0.5 * (mark_freq + space_freq) / (double)samples_per_sec ); + + D->u.afsk.use_rrc = 1; + TUNE("TUNE_USE_RRC", D->u.afsk.use_rrc, "use_rrc", "%d") + + if (D->u.afsk.use_rrc) { + D->u.afsk.rrc_width_sym = 2.00; + D->u.afsk.rrc_rolloff = 0.40; + } + else { + D->lpf_baud = 0.5; + D->lp_filter_width_sym = 1.714286; // 63 * 1200. / 44100.; + D->lp_window = BP_WINDOW_TRUNCATED; + } + + // For scaling phase shift into normallized -1 to +1 range for mark and space. + D->u.afsk.normalize_rpsam = 1.0 / (0.5 * abs(mark_freq - space_freq) * 2 * M_PI / samples_per_sec); + + // New "B" demodulator does not use AGC but demod.c needs this to derive "quick" and + // "sluggish" values for overall signal amplitude. That probably should be independent + // of these values. + D->agc_fast_attack = 0.70; + D->agc_slow_decay = 0.000090; + + D->pll_locked_inertia = 0.74; + D->pll_searching_inertia = 0.50; + + D->alevel_mark_peak = -1; // Disable received signal (m/s) display. + D->alevel_space_peak = -1; + break; + + default: + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid AFSK demodulator profile = %c\n", profile); + exit (1); + } + + + TUNE("TUNE_PRE_BAUD", D->prefilter_baud, "prefilter_baud", "%.3f") + TUNE("TUNE_PRE_WINDOW", D->pre_window, "pre_window", "%d") + + TUNE("TUNE_LPF_BAUD", D->lpf_baud, "lpf_baud", "%.3f") + TUNE("TUNE_LP_WINDOW", D->lp_window, "lp_window", "%d") + + TUNE("TUNE_RRC_ROLLOFF", D->u.afsk.rrc_rolloff, "rrc_rolloff", "%.2f") + TUNE("TUNE_RRC_WIDTH_SYM", D->u.afsk.rrc_width_sym, "rrc_width_sym", "%.2f") + + TUNE("TUNE_AGC_FAST", D->agc_fast_attack, "agc_fast_attack", "%.3f") + TUNE("TUNE_AGC_SLOW", D->agc_slow_decay, "agc_slow_decay", "%.6f") + + TUNE("TUNE_PLL_LOCKED", D->pll_locked_inertia, "pll_locked_inertia", "%.2f") + TUNE("TUNE_PLL_SEARCHING", D->pll_searching_inertia, "pll_searching_inertia", "%.2f") + + +/* + * Calculate constants used for timing. + * The audio sample rate must be at least a few times the data rate. + * + * Baud is an integer so we hack in a fine adjustment for EAS. + * Probably makes no difference because the DPLL keeps it in sync. + * + * A fraction if a Hz would make no difference for the filters. + */ + if (baud == 521) { + D->pll_step_per_sample = (int) round((TICKS_PER_PLL_CYCLE * (double)520.83) / ((double)samples_per_sec)); + } + else { + D->pll_step_per_sample = (int) round((TICKS_PER_PLL_CYCLE * (double)baud) / ((double)samples_per_sec)); + } + +/* + * Optionally apply a bandpass ("pre") filter to attenuate + * frequencies outside the range of interest. + */ + + if (D->use_prefilter) { + + // odd number is a little better + D->pre_filter_taps = ((int)( D->pre_filter_len_sym * (float)samples_per_sec / (float)baud )) | 1; + + TUNE("TUNE_PRE_FILTER_TAPS", D->pre_filter_taps, "pre_filter_taps", "%d") + +// TODO: Size comes out to 417 for 1200 bps with 48000 sample rate. +// The message is upsetting. Can we handle this better? + + if (D->pre_filter_taps > MAX_FILTER_SIZE) { + text_color_set (DW_COLOR_ERROR); + dw_printf ("Warning: Calculated pre filter size of %d is too large.\n", D->pre_filter_taps); + dw_printf ("Decrease the audio sample rate or increase the decimation factor.\n"); + dw_printf ("You can use -D2 or -D3, on the command line, to down-sample the audio rate\n"); + dw_printf ("before demodulating. This greatly decreases the CPU requirements with little\n"); + dw_printf ("impact on the decoding performance. This is useful for a slow ARM processor,\n"); + dw_printf ("such as with a Raspberry Pi model 1.\n"); + D->pre_filter_taps = (MAX_FILTER_SIZE - 1) | 1; + } + + float f1 = MIN(mark_freq,space_freq) - D->prefilter_baud * baud; + float f2 = MAX(mark_freq,space_freq) + D->prefilter_baud * baud; +#if 0 + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Generating prefilter %.0f to %.0f Hz.\n", f1, f2); +#endif + f1 = f1 / (float)samples_per_sec; + f2 = f2 / (float)samples_per_sec; + + gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_taps, D->pre_window); + } + +/* + * Now the lowpass filter. + * In version 1.7 a Root Raised Cosine filter is added as an alternative + * to the generic low pass filter. + * In both cases, lp_filter and lp_filter_taps are used but the + * contents will be generated differently. Later code does not care. + */ + if (D->u.afsk.use_rrc) { + + assert (D->u.afsk.rrc_width_sym >= 1 && D->u.afsk.rrc_width_sym <= 16); + assert (D->u.afsk.rrc_rolloff >= 0. && D->u.afsk.rrc_rolloff <= 1.); + + D->lp_filter_taps = ((int) (D->u.afsk.rrc_width_sym * (float)samples_per_sec / baud)) | 1; // odd works better + + TUNE("TUNE_LP_FILTER_TAPS", D->lp_filter_taps, "lp_filter_taps (RRC)", "%d") + + if (D->lp_filter_taps > MAX_FILTER_SIZE) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Calculated RRC low pass filter size of %d is too large.\n", D->lp_filter_taps); + dw_printf ("Decrease the audio sample rate or increase the decimation factor or\n"); + dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", MAX_FILTER_SIZE); + D->lp_filter_taps = (MAX_FILTER_SIZE - 1) | 1; + } + + assert (D->lp_filter_taps > 8 && D->lp_filter_taps <= MAX_FILTER_SIZE); + (void)gen_rrc_lowpass (D->lp_filter, D->lp_filter_taps, D->u.afsk.rrc_rolloff, (float)samples_per_sec / baud); + } + else { + D->lp_filter_taps = (int) round( D->lp_filter_width_sym * (float)samples_per_sec / (float)baud ); + + TUNE("TUNE_LP_FILTER_TAPS", D->lp_filter_taps, "lp_filter_taps (FIR)", "%d") + + if (D->lp_filter_taps > MAX_FILTER_SIZE) { + text_color_set (DW_COLOR_ERROR); + dw_printf ("Calculated FIR low pass filter size of %d is too large.\n", D->lp_filter_taps); + dw_printf ("Decrease the audio sample rate or increase the decimation factor or\n"); + dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", MAX_FILTER_SIZE); + D->lp_filter_taps = (MAX_FILTER_SIZE - 1) | 1; + } + + assert (D->lp_filter_taps > 8 && D->lp_filter_taps <= MAX_FILTER_SIZE); + + float fc = baud * D->lpf_baud / (float)samples_per_sec; + gen_lowpass (fc, D->lp_filter, D->lp_filter_taps, D->lp_window); + } + + +/* + * Starting with version 1.2 + * try using multiple slicing points instead of the traditional AGC. + */ + space_gain[0] = MIN_G; + float step = powf(10.0, log10f(MAX_G/MIN_G) / (MAX_SUBCHANS-1)); + for (j=1; j= 0 && chan < MAX_CHANS); + assert (subchan >= 0 && subchan < MAX_SUBCHANS); + +/* + * Filters use last 'filter_taps' samples. + * + * First push the older samples down. + * + * Finally, put the most recent at the beginning. + * + * Future project? Can we do better than shifting each time? + */ + + /* Scale to nice number. */ + + float fsam = (float)sam / 16384.0f; + + switch (D->profile) { + + case 'E': + default: + case 'A': { + /* ========== New in Version 1.7 ========== */ + + // Cleaner & simpler than earlier 'A' thru 'E' + + if (D->use_prefilter) { + push_sample (fsam, D->raw_cb, D->pre_filter_taps); + fsam = convolve (D->raw_cb, D->pre_filter, D->pre_filter_taps); + } + + push_sample (fsam * fcos256(D->u.afsk.m_osc_phase), D->u.afsk.m_I_raw, D->lp_filter_taps); + push_sample (fsam * fsin256(D->u.afsk.m_osc_phase), D->u.afsk.m_Q_raw, D->lp_filter_taps); + D->u.afsk.m_osc_phase += D->u.afsk.m_osc_delta; + + push_sample (fsam * fcos256(D->u.afsk.s_osc_phase), D->u.afsk.s_I_raw, D->lp_filter_taps); + push_sample (fsam * fsin256(D->u.afsk.s_osc_phase), D->u.afsk.s_Q_raw, D->lp_filter_taps); + D->u.afsk.s_osc_phase += D->u.afsk.s_osc_delta; + + float m_I = convolve (D->u.afsk.m_I_raw, D->lp_filter, D->lp_filter_taps); + float m_Q = convolve (D->u.afsk.m_Q_raw, D->lp_filter, D->lp_filter_taps); + float m_amp = fast_hypot(m_I, m_Q); + + float s_I = convolve (D->u.afsk.s_I_raw, D->lp_filter, D->lp_filter_taps); + float s_Q = convolve (D->u.afsk.s_Q_raw, D->lp_filter, D->lp_filter_taps); + float s_amp = fast_hypot(s_I, s_Q); + +/* + * Capture the mark and space peak amplitudes for display. + * It uses fast attack and slow decay to get an idea of the + * overall amplitude. + */ + if (m_amp >= D->alevel_mark_peak) { + D->alevel_mark_peak = m_amp * D->quick_attack + D->alevel_mark_peak * (1.0f - D->quick_attack); + } + else { + D->alevel_mark_peak = m_amp * D->sluggish_decay + D->alevel_mark_peak * (1.0f - D->sluggish_decay); + } + + if (s_amp >= D->alevel_space_peak) { + D->alevel_space_peak = s_amp * D->quick_attack + D->alevel_space_peak * (1.0f - D->quick_attack); + } + else { + D->alevel_space_peak = s_amp * D->sluggish_decay + D->alevel_space_peak * (1.0f - D->sluggish_decay); + } + + if (D->num_slicers <= 1) { + + // Which tone is stonger? That's simple with an ideal signal. + // However, we don't see too many ideal signals. + // Due to mismatching pre-emphasis and de-emphasis, the two + // tones will often have greatly different amplitudes so we use + // automatic gain control (AGC) to scale each to the same range + // before comparing. + // This is probably over complicated and could be combined with + // the signal amplitude measurement, above. + // It works so let's move along to other topics. + + float m_norm = agc (m_amp, D->agc_fast_attack, D->agc_slow_decay, &(D->m_peak), &(D->m_valley)); + float s_norm = agc (s_amp, D->agc_fast_attack, D->agc_slow_decay, &(D->s_peak), &(D->s_valley)); + + // The normalized values should be around -0.5 to +0.5 so the difference + // should work out to be around -1 to +1. + // This is important because nudge_pll uses the demod_out amplitude to assign + // a quality or confidence score to the symbol. + + float demod_out = m_norm - s_norm; + + // Tested and it looks good. Range of about -1 to +1. + //printf ("JWL DEBUG demod A with agc = %6.2f\n", demod_out); + + nudge_pll (chan, subchan, 0, demod_out, D, 1.0); + + } + else { + // Multiple slice case. + // Rather than trying to find the best threshold location, use multiple + // slicer thresholds in parallel. + // The best slicing point will vary from packet to packet but should + // remain about the same for a given packet. + + // We are not performing the AGC step here but still want the envelope + // for caluculating the confidence level (or quality) of the sample. + + (void) agc (m_amp, D->agc_fast_attack, D->agc_slow_decay, &(D->m_peak), &(D->m_valley)); + (void) agc (s_amp, D->agc_fast_attack, D->agc_slow_decay, &(D->s_peak), &(D->s_valley)); + + for (int slice=0; slicenum_slicers; slice++) { + float demod_out = m_amp - s_amp * space_gain[slice]; + float amp = 0.5f * (D->m_peak - D->m_valley + (D->s_peak - D->s_valley) * space_gain[slice]); + if (amp < 0.0000001f) amp = 1; // avoid divide by zero with no signal. + + // Tested and it looks good. Range of about -1 to +1 relative to amp. + // Biased one way or the other depending on the space gain. + //printf ("JWL DEBUG demod A with slicer %d: %6.2f / %6.2f = %6.2f\n", slice, demod_out, amp, demod_out/amp); + + nudge_pll (chan, subchan, slice, demod_out, D, amp); + } + } + } + break; + + case 'D': + case 'B': { + /* ========== Version 1.7 Experiment ========== */ + + // New - Convert frequency to a value proportional to frequency. + + if (D->use_prefilter) { + push_sample (fsam, D->raw_cb, D->pre_filter_taps); + fsam = convolve (D->raw_cb, D->pre_filter, D->pre_filter_taps); + } + + push_sample (fsam * fcos256(D->u.afsk.c_osc_phase), D->u.afsk.c_I_raw, D->lp_filter_taps); + push_sample (fsam * fsin256(D->u.afsk.c_osc_phase), D->u.afsk.c_Q_raw, D->lp_filter_taps); + D->u.afsk.c_osc_phase += D->u.afsk.c_osc_delta; + + float c_I = convolve (D->u.afsk.c_I_raw, D->lp_filter, D->lp_filter_taps); + float c_Q = convolve (D->u.afsk.c_Q_raw, D->lp_filter, D->lp_filter_taps); + + float phase = atan2f (c_Q, c_I); + float rate = phase - D->u.afsk.prev_phase; + if (rate > M_PI) rate -= 2 * M_PI; + else if (rate < -M_PI) rate += 2 * M_PI; + D->u.afsk.prev_phase = phase; + + // Rate is radians per audio sample interval or something like that. + // Scale scale that into -1 to +1 for expected tones. + + float norm_rate = rate * D->u.afsk.normalize_rpsam; + + // We really don't have mark and space amplitudes available in this case. + + if (D->num_slicers <= 1) { + + float demod_out = norm_rate; + // Tested and it looks good. Range roughly -1 to +1. + //printf ("JWL DEBUG demod B single = %6.2f\n", demod_out); + + nudge_pll (chan, subchan, 0, demod_out, D, 1.0); + + } + else { + + // This would be useful for HF SSB where a tuning error + // would shift the frequency. Multiple slicing points would + // then compensate for differences in transmit/receive frequencies. + // + // Where should we set the thresholds? + // I'm thinking something like: + // -.5 -.375 -.25 -.125 0 .125 .25 .375 .5 + // + // Assuming a 300 Hz shift, this would put slicing thresholds up + // to +-75 Hz from the center. + + for (int slice=0; slicenum_slicers; slice++) { + + float offset = -0.5 + slice * (1. / (D->num_slicers - 1)); + float demod_out = norm_rate + offset; + + //printf ("JWL DEBUG demod B slice %d, offset = %6.3f, demod_out = %6.2f\n", slice, offset, demod_out); + + nudge_pll (chan, subchan, slice, demod_out, D, 1.0); + } + } + } + break; + } + +#if DEBUG4 + + if (chan == 0) { + if (D->slicer[slice].data_detect) { + char fname[30]; + + + if (demod_log_fp == NULL) { + seq++; + snprintf (fname, sizeof(fname), "demod/%04d.csv", seq); + if (seq == 1) mkdir ("demod", 0777); + + demod_log_fp = fopen (fname, "w"); + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Starting demodulator log file %s\n", fname); + fprintf (demod_log_fp, "Audio, Mark, Space, Demod, Data, Clock\n"); + } + fprintf (demod_log_fp, "%.3f, %.3f, %.3f, %.3f, %.2f, %.2f\n", fsam + 3.5, m_norm + 2, s_norm + 2, + (m_norm - s_norm) / 2 + 1.5, + demod_data ? .9 : .55, + (D->data_clock_pll & 0x80000000) ? .1 : .45); + } + else { + if (demod_log_fp != NULL) { + fclose (demod_log_fp); + demod_log_fp = NULL; + } + } + } + +#endif + + +} /* end demod_afsk_process_sample */ + + + +/* + * Finally, a PLL is used to sample near the centers of the data bits. + * + * D points to a demodulator for a channel/subchannel pair so we don't + * have to keep recalculating it. + * + * D->data_clock_pll is a SIGNED 32 bit variable. + * When it overflows from a large positive value to a negative value, we + * sample a data bit from the demodulated signal. + * + * Ideally, the the demodulated signal transitions should be near + * zero we we sample mid way between the transitions. + * + * Nudge the PLL by removing some small fraction from the value of + * data_clock_pll, pushing it closer to zero. + * + * This adjustment will never change the sign so it won't cause + * any erratic data bit sampling. + * + * If we adjust it too quickly, the clock will have too much jitter. + * If we adjust it too slowly, it will take too long to lock on to a new signal. + * + * Be a little more aggressive about adjusting the PLL + * phase when searching for a signal. Don't change it as much when + * locked on to a signal. + * + * I don't think the optimal value will depend on the audio sample rate + * because this happens for each transition from the demodulator. + */ + +__attribute__((hot)) +static void nudge_pll (int chan, int subchan, int slice, float demod_out, struct demodulator_state_s *D, float amplitude) +{ + D->slicer[slice].prev_d_c_pll = D->slicer[slice].data_clock_pll; + + + // Perform the add as unsigned to avoid signed overflow error. + D->slicer[slice].data_clock_pll = (signed)((unsigned)(D->slicer[slice].data_clock_pll) + (unsigned)(D->pll_step_per_sample)); + + //text_color_set(DW_COLOR_DEBUG); + // dw_printf ("prev = %lx, new data clock pll = %lx\n" D->prev_d_c_pll, D->data_clock_pll); + + if (D->slicer[slice].data_clock_pll < 0 && D->slicer[slice].prev_d_c_pll > 0) { + + /* Overflow - this is where we sample. */ + // Assign it a confidence level or quality, 0 to 100, based on the amplitude. + // Those very close to 0 are suspect. We'll get back to this later. + + int quality = fabsf(demod_out) * 100.0f / amplitude; + if (quality > 100) quality = 100; + +#if DEBUG5 + // Write bit stream to a file. + + static FILE *bsfp = NULL; + static int bcount = 0; + if (chan == 0 && subchan == 0 && slice == 0) { + if (bsfp == NULL) { + bsfp = fopen ("bitstream.txt", "w"); + } + fprintf (bsfp, "%d", demod_out > 0); + bcount++; + if (bcount % 64 == 0) { + fprintf (bsfp, "\n"); + } + } + +#endif + + +#if 1 + hdlc_rec_bit (chan, subchan, slice, demod_out > 0, 0, quality); +#else // TODO: new feature to measure data speed error. +// Maybe hdlc_rec_bit could provide indication when frame starts. + hdlc_rec_bit_new (chan, subchan, slice, demod_out > 0, 0, quality, + &(D->slicer[slice].pll_nudge_total), &(D->slicer[slice].pll_symbol_count)); + D->slicer[slice].pll_symbol_count++; +#endif + pll_dcd_each_symbol2 (D, chan, subchan, slice); + } + + // Transitions nudge the DPLL phase toward the incoming signal. + + int demod_data = demod_out > 0; + if (demod_data != D->slicer[slice].prev_demod_data) { + + pll_dcd_signal_transition2 (D, slice, D->slicer[slice].data_clock_pll); + +// TODO: signed int before = (signed int)(D->slicer[slice].data_clock_pll); // Treat as signed. + if (D->slicer[slice].data_detect) { + D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_locked_inertia); + } + else { + D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_searching_inertia); + } +// TODO: D->slicer[slice].pll_nudge_total += (int64_t)((signed int)(D->slicer[slice].data_clock_pll)) - (int64_t)before; + } + +/* + * Remember demodulator output so we can compare next time. + */ + D->slicer[slice].prev_demod_data = demod_data; + +} /* end nudge_pll */ + + +/* end demod_afsk.c */ diff --git a/demod_afsk.h b/src/demod_afsk.h similarity index 100% rename from demod_afsk.h rename to src/demod_afsk.h diff --git a/demod_psk.c b/src/demod_psk.c similarity index 51% rename from demod_psk.c rename to src/demod_psk.c index db9407aa..bc058185 100644 --- a/demod_psk.c +++ b/src/demod_psk.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2016 John Langner, WB2OSZ +// +// Copyright (C) 2016, 2019 John Langner, WB2OSZ // // 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 @@ -20,35 +20,17 @@ //#define DEBUG1 1 /* display debugging info */ -//#define DEBUG3 1 /* print carrier detect changes. */ - -//#define DEBUG4 1 /* capture PSK demodulator output to log files */ - - /*------------------------------------------------------------------ * * Module: demod_psk.c * - * Purpose: Demodulator for Phase Shift Keying (PSK). - * - * This is my initial attempt at implementing a 2400 bps mode. - * The MFJ-2400 & AEA PK232-2400 used V.26 / Bell 201 so I will follow that precedent. + * Purpose: Demodulator for 2400 and 4800 bits per second Phase Shift Keying (PSK). * - * * Input: Audio samples from either a file or the "sound card." * * Outputs: Calls hdlc_rec_bit() for each bit demodulated. * - * Current Status: New for Version 1.4. - * - * Don't know if this is correct and/or compatible with - * other implementations. - * There is a lot of stuff going on here with phase - * shifting, gray code, bit order for the dibit, NRZI and - * bit-stuffing for HDLC. Plenty of opportunity for - * misinterpreting a protocol spec or just stupid mistakes. - * * References: MFJ-2400 Product description and manual: * * http://www.mfjenterprises.com/Product.php?productid=MFJ-2400 @@ -63,8 +45,7 @@ * http://www.brazoriacountyares.org/winlink-collection/TNC%20manuals/Kantronics/2400_modem_operators_guide@rgf.pdf * * - * The MFJ and AEA both use the EXAR XR-2123 PSK modem chip. - * The Kantronics has a P423 ??? + * From what I'm able to gather, they all used the EXAR XR-2123 PSK modem chip. * * Can't find the chip specs on the EXAR website so Google it. * @@ -79,19 +60,19 @@ * "bis" and "ter" are from Latin for second and third. * I used the "ter" version which has phase shifts of 0, 90, 180, and 270 degrees. * - * There are other references to an alternative B which uses other multiples of 45. - * The XR-2123 data sheet mentions only multiples of 90. That's what I went with. - * - * The XR-2123 does not perform the scrambling as specified in V.26 so I wonder if - * the vendors implemented it in software or just left it out. - * I left out scrambling for now. Eventually, I'd like to get my hands on an old - * 2400 bps TNC for compatibility testing. + * There are earlier references to an alternative B which uses other phase shifts offset + * by another 45 degrees. * * After getting QPSK working, it was not much more effort to add V.27 with 8 phases. * * https://www.itu.int/rec/dologin_pub.asp?lang=e&id=T-REC-V.27bis-198811-I!!PDF-E&type=items * https://www.itu.int/rec/dologin_pub.asp?lang=e&id=T-REC-V.27ter-198811-I!!PDF-E&type=items * + * Compatibility: + * V.26 has two variations, A and B. Initially I implemented the A alternative. + * It later turned out that the MFJ-2400 used the B alternative. In version 1.6 you have a + * choice between compatibility with MFJ (and probably the others) or the original implementation. + * *---------------------------------------------------------------*/ #include "direwolf.h" @@ -105,10 +86,15 @@ #include #include +// Fine tuning for different demodulator types. + +#define DCD_THRESH_ON 30 // Hysteresis: Can miss 2 out of 32 for detecting lock. +#define DCD_THRESH_OFF 6 // Might want a little more fine tuning. +#define DCD_GOOD_WIDTH 512 +#include "fsk_demod_state.h" // Values above override defaults. #include "audio.h" #include "tune.h" -#include "fsk_demod_state.h" #include "fsk_gen_filter.h" #include "hdlc_rec.h" #include "textcolor.h" @@ -116,6 +102,15 @@ #include "dsp.h" + + + +static const int phase_to_gray_v26[4] = {0, 1, 3, 2}; +static const int phase_to_gray_v27[8] = {1, 0, 2, 3, 7, 6, 4, 5}; + + +static int phase_shift_to_symbol (float phase_shift, int bits_per_symbol, int *bit_quality); + /* Add sample to buffer and shift the rest down. */ __attribute__((hot)) __attribute__((always_inline)) @@ -134,6 +129,8 @@ static inline float convolve (const float *__restrict__ data, const float *__res float sum = 0.0; int j; +//Does pragma make any difference? Annoying warning on Mac. +//#pragma GCC ivdep for (j=0; jmodem_type = modem_type; - D->num_slicers = 1; // Haven't thought about this yet. Is it even applicable? - + D->u.psk.v26_alt = v26_alt; + D->num_slicers = 1; // Haven't thought about this yet. Is it even applicable? #ifdef TUNE_PROFILE @@ -210,10 +208,9 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char if (modem_type == MODEM_QPSK) { + assert (D->u.psk.v26_alt != V26_UNSPECIFIED); + correct_baud = bps / 2; - // Originally I thought of scaling it to the data rate, - // e.g. 2400 bps -> 1800 Hz, but decided to make it a - // constant since it is the same for V.26 and V.27. carrier_freq = 1800; #if DEBUG1 @@ -225,11 +222,11 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'P': /* Self correlation technique. */ - D->use_prefilter = 0; /* No bandpass filter. */ + D->u.psk.use_prefilter = 0; /* No bandpass filter. */ - D->lpf_baud = 0.60; - D->lp_filter_len_bits = 39. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 0.60; + D->u.psk.lp_filter_width_sym = 1.061; // 39. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.95; D->pll_searching_inertia = 0.50; @@ -238,14 +235,14 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'Q': /* Self correlation technique. */ - D->use_prefilter = 1; /* Add a bandpass filter. */ - D->prefilter_baud = 1.3; - D->pre_filter_len_bits = 55. * 1200. / 44100.; - D->pre_window = BP_WINDOW_COSINE; + D->u.psk.use_prefilter = 1; /* Add a bandpass filter. */ + D->u.psk.prefilter_baud = 1.3; + D->u.psk.pre_filter_width_sym = 1.497; // 55. * 1200. / 44100.; + D->u.psk.pre_window = BP_WINDOW_COSINE; - D->lpf_baud = 0.60; - D->lp_filter_len_bits = 39. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 0.60; + D->u.psk.lp_filter_width_sym = 1.061; // 39. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.87; D->pll_searching_inertia = 0.50; @@ -259,13 +256,13 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'R': /* Mix with local oscillator. */ - D->psk_use_lo = 1; + D->u.psk.psk_use_lo = 1; - D->use_prefilter = 0; /* No bandpass filter. */ + D->u.psk.use_prefilter = 0; /* No bandpass filter. */ - D->lpf_baud = 0.70; - D->lp_filter_len_bits = 37. * 1200. / 44100.; - D->lp_window = BP_WINDOW_TRUNCATED; + D->u.psk.lpf_baud = 0.70; + D->u.psk.lp_filter_width_sym = 1.007; // 37. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_TRUNCATED; D->pll_locked_inertia = 0.925; D->pll_searching_inertia = 0.50; @@ -274,16 +271,16 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'S': /* Mix with local oscillator. */ - D->psk_use_lo = 1; + D->u.psk.psk_use_lo = 1; - D->use_prefilter = 1; /* Add a bandpass filter. */ - D->prefilter_baud = 0.55; - D->pre_filter_len_bits = 74. * 1200. / 44100.; - D->pre_window = BP_WINDOW_FLATTOP; + D->u.psk.use_prefilter = 1; /* Add a bandpass filter. */ + D->u.psk.prefilter_baud = 0.55; + D->u.psk.pre_filter_width_sym = 2.014; // 74. * 1200. / 44100.; + D->u.psk.pre_window = BP_WINDOW_FLATTOP; - D->lpf_baud = 0.60; - D->lp_filter_len_bits = 39. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 0.60; + D->u.psk.lp_filter_width_sym = 1.061; // 39. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.925; D->pll_searching_inertia = 0.50; @@ -291,11 +288,11 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char break; } - D->ms_filter_len_bits = 1.25; // Delay line > 13/12 * symbol period + D->u.psk.delay_line_width_sym = 1.25; // Delay line > 13/12 * symbol period - D->coffs = (int) round( (11.f / 12.f) * (float)samples_per_sec / (float)correct_baud ); - D->boffs = (int) round( (float)samples_per_sec / (float)correct_baud ); - D->soffs = (int) round( (13.f / 12.f) * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.coffs = (int) round( (11.f / 12.f) * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.boffs = (int) round( (float)samples_per_sec / (float)correct_baud ); + D->u.psk.soffs = (int) round( (13.f / 12.f) * (float)samples_per_sec / (float)correct_baud ); } else { @@ -312,11 +309,11 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'T': /* Self correlation technique. */ - D->use_prefilter = 0; /* No bandpass filter. */ + D->u.psk.use_prefilter = 0; /* No bandpass filter. */ - D->lpf_baud = 1.15; - D->lp_filter_len_bits = 32. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 1.15; + D->u.psk.lp_filter_width_sym = 0.871; // 32. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.95; D->pll_searching_inertia = 0.50; @@ -325,14 +322,14 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'U': /* Self correlation technique. */ - D->use_prefilter = 1; /* Add a bandpass filter. */ - D->prefilter_baud = 0.9; - D->pre_filter_len_bits = 21. * 1200. / 44100.; - D->pre_window = BP_WINDOW_FLATTOP; + D->u.psk.use_prefilter = 1; /* Add a bandpass filter. */ + D->u.psk.prefilter_baud = 0.9; + D->u.psk.pre_filter_width_sym = 0.571; // 21. * 1200. / 44100.; + D->u.psk.pre_window = BP_WINDOW_FLATTOP; - D->lpf_baud = 1.15; - D->lp_filter_len_bits = 32. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 1.15; + D->u.psk.lp_filter_width_sym = 0.871; // 32. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.87; D->pll_searching_inertia = 0.50; @@ -346,13 +343,13 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'V': /* Mix with local oscillator. */ - D->psk_use_lo = 1; + D->u.psk.psk_use_lo = 1; - D->use_prefilter = 0; /* No bandpass filter. */ + D->u.psk.use_prefilter = 0; /* No bandpass filter. */ - D->lpf_baud = 0.85; - D->lp_filter_len_bits = 31. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 0.85; + D->u.psk.lp_filter_width_sym = 0.844; // 31. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.925; D->pll_searching_inertia = 0.50; @@ -361,16 +358,16 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char case 'W': /* Mix with local oscillator. */ - D->psk_use_lo = 1; + D->u.psk.psk_use_lo = 1; - D->use_prefilter = 1; /* Add a bandpass filter. */ - D->prefilter_baud = 0.85; - D->pre_filter_len_bits = 31. * 1200. / 44100.; - D->pre_window = BP_WINDOW_COSINE; + D->u.psk.use_prefilter = 1; /* Add a bandpass filter. */ + D->u.psk.prefilter_baud = 0.85; + D->u.psk.pre_filter_width_sym = 0.844; // 31. * 1200. / 44100.; + D->u.psk.pre_window = BP_WINDOW_COSINE; - D->lpf_baud = 0.85; - D->lp_filter_len_bits = 31. * 1200. / 44100.; - D->lp_window = BP_WINDOW_COSINE; + D->u.psk.lpf_baud = 0.85; + D->u.psk.lp_filter_width_sym = 0.844; // 31. * 1200. / 44100.; + D->u.psk.lp_window = BP_WINDOW_COSINE; D->pll_locked_inertia = 0.925; D->pll_searching_inertia = 0.50; @@ -378,42 +375,36 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char break; } - D->ms_filter_len_bits = 1.25; // Delay line > 10/9 * symbol period + D->u.psk.delay_line_width_sym = 1.25; // Delay line > 10/9 * symbol period - D->coffs = (int) round( (8.f / 9.f) * (float)samples_per_sec / (float)correct_baud ); - D->boffs = (int) round( (float)samples_per_sec / (float)correct_baud ); - D->soffs = (int) round( (10.f / 9.f) * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.coffs = (int) round( (8.f / 9.f) * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.boffs = (int) round( (float)samples_per_sec / (float)correct_baud ); + D->u.psk.soffs = (int) round( (10.f / 9.f) * (float)samples_per_sec / (float)correct_baud ); } - if (D->psk_use_lo) { - D->lo_step = (int) round( 256. * 256. * 256. * 256. * carrier_freq / (double)samples_per_sec); + if (D->u.psk.psk_use_lo) { + D->u.psk.lo_step = (int) round( 256. * 256. * 256. * 256. * carrier_freq / (double)samples_per_sec); + +// Our own sin table for speed later. - assert (MAX_FILTER_SIZE >= 256); for (j = 0; j < 256; j++) { - D->m_sin_table[j] = sinf(2.f * (float)M_PI * j / 256.f); + D->u.psk.sin_table256[j] = sinf(2.f * (float)M_PI * j / 256.f); } } #ifdef TUNE_PRE_BAUD - D->prefilter_baud = TUNE_PRE_BAUD; + D->u.psk.prefilter_baud = TUNE_PRE_BAUD; #endif #ifdef TUNE_PRE_WINDOW - D->pre_window = TUNE_PRE_WINDOW; + D->u.psk.pre_window = TUNE_PRE_WINDOW; #endif - - #ifdef TUNE_LPF_BAUD - D->lpf_baud = TUNE_LPF_BAUD; + D->u.psk.lpf_baud = TUNE_LPF_BAUD; #endif #ifdef TUNE_LP_WINDOW - D->lp_window = TUNE_LP_WINDOW; -#endif - - -#ifdef TUNE_HYST - D->hysteresis = TUNE_HYST; + D->u.psk.lp_window = TUNE_LP_WINDOW; #endif #if defined(TUNE_PLL_SEARCHING) @@ -435,44 +426,41 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char * Convert number of symbol times to number of taps. */ - D->pre_filter_size = (int) round( D->pre_filter_len_bits * (float)samples_per_sec / (float)correct_baud ); - D->ms_filter_size = (int) round( D->ms_filter_len_bits * (float)samples_per_sec / (float)correct_baud ); - D->lp_filter_size = (int) round( D->lp_filter_len_bits * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.pre_filter_taps = (int) round( D->u.psk.pre_filter_width_sym * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.delay_line_taps = (int) round( D->u.psk.delay_line_width_sym * (float)samples_per_sec / (float)correct_baud ); + D->u.psk.lp_filter_taps = (int) round( D->u.psk.lp_filter_width_sym * (float)samples_per_sec / (float)correct_baud ); -#ifdef TUNE_PRE_FILTER_SIZE - D->pre_filter_size = TUNE_PRE_FILTER_SIZE; +#ifdef TUNE_PRE_FILTER_TAPS + D->u.psk.pre_filter_taps = TUNE_PRE_FILTER_TAPS; #endif -#ifdef TUNE_LP_FILTER_SIZE - D->lp_filter_size = TUNE_LP_FILTER_SIZE; +#ifdef TUNE_lp_filter_taps + D->u.psk.lp_filter_taps = TUNE_lp_filter_taps; #endif - if (D->pre_filter_size > MAX_FILTER_SIZE) - { + if (D->u.psk.pre_filter_taps > MAX_FILTER_SIZE) { text_color_set (DW_COLOR_ERROR); - dw_printf ("Calculated filter size of %d is too large.\n", D->pre_filter_size); + dw_printf ("Calculated pre filter size of %d is too large.\n", D->u.psk.pre_filter_taps); dw_printf ("Decrease the audio sample rate or increase the baud rate or\n"); dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", MAX_FILTER_SIZE); exit (1); } - if (D->ms_filter_size > MAX_FILTER_SIZE) - { + if (D->u.psk.delay_line_taps > MAX_FILTER_SIZE) { text_color_set (DW_COLOR_ERROR); - dw_printf ("Calculated filter size of %d is too large.\n", D->ms_filter_size); + dw_printf ("Calculated delay line size of %d is too large.\n", D->u.psk.delay_line_taps); dw_printf ("Decrease the audio sample rate or increase the baud rate or\n"); dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", MAX_FILTER_SIZE); exit (1); } - if (D->lp_filter_size > MAX_FILTER_SIZE) - { + if (D->u.psk.lp_filter_taps > MAX_FILTER_SIZE) { text_color_set (DW_COLOR_ERROR); - dw_printf ("Calculated filter size of %d is too large.\n", D->pre_filter_size); + dw_printf ("Calculated low pass filter size of %d is too large.\n", D->u.psk.lp_filter_taps); dw_printf ("Decrease the audio sample rate or increase the baud rate or\n"); dw_printf ("recompile the application with MAX_FILTER_SIZE larger than %d.\n", MAX_FILTER_SIZE); @@ -482,14 +470,17 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char /* * Optionally apply a bandpass ("pre") filter to attenuate * frequencies outside the range of interest. + * It's a tradeoff. Attenuate frequencies outside the the range of interest + * but also distort the signal. This demodulator is not compuationally + * intensive so we can usually run both in parallel. */ - if (D->use_prefilter) { + if (D->u.psk.use_prefilter) { float f1, f2; - f1 = carrier_freq - D->prefilter_baud * correct_baud; - f2 = carrier_freq + D->prefilter_baud * correct_baud; -#if 0 + f1 = carrier_freq - D->u.psk.prefilter_baud * correct_baud; + f2 = carrier_freq + D->u.psk.prefilter_baud * correct_baud; +#if DEBUG1 text_color_set(DW_COLOR_DEBUG); dw_printf ("Generating prefilter %.0f to %.0f Hz.\n", (double)f1, (double)f2); #endif @@ -502,15 +493,15 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char f1 = f1 / (float)samples_per_sec; f2 = f2 / (float)samples_per_sec; - gen_bandpass (f1, f2, D->pre_filter, D->pre_filter_size, D->pre_window); + gen_bandpass (f1, f2, D->u.psk.pre_filter, D->u.psk.pre_filter_taps, D->u.psk.pre_window); } /* * Now the lowpass filter. */ - float fc = correct_baud * D->lpf_baud / (float)samples_per_sec; - gen_lowpass (fc, D->lp_filter, D->lp_filter_size, D->lp_window); + float fc = correct_baud * D->u.psk.lpf_baud / (float)samples_per_sec; + gen_lowpass (fc, D->u.psk.lp_filter, D->u.psk.lp_filter_taps, D->u.psk.lp_window); /* * No point in having multiple numbers for signal level. @@ -519,10 +510,98 @@ void demod_psk_init (enum modem_t modem_type, int samples_per_sec, int bps, char D->alevel_mark_peak = -1; D->alevel_space_peak = -1; +#if 0 + // QPSK - CSV format to make plot. + + printf ("Phase shift degrees, bit 0, quality 0, bit 1, quality 1\n"); + for (int degrees = 0; degrees <= 360; degrees++) { + float a = degrees * M_PI * 2./ 360.; + int bit_quality[3]; + + int new_gray = phase_shift_to_symbol (a, 2, bit_quality); + + float offset = 3 * 1.5; + printf ("%d, ", degrees); + printf ("%.3f, ", offset + (new_gray & 1)); offset -= 1.5; + printf ("%.3f, ", offset + (bit_quality[0] / 100.)); offset -= 1.5; + printf ("%.3f, ", offset + ((new_gray >> 1) & 1)); offset -= 1.5; + printf ("%.3f\n", offset + (bit_quality[1] / 100.)); + } +#endif + +#if 0 + // 8-PSK - CSV format to make plot. + + printf ("Phase shift degrees, bit 0, quality 0, bit 1, quality 1, bit 2, quality 2\n"); + for (int degrees = 0; degrees <= 360; degrees++) { + float a = degrees * M_PI * 2./ 360.; + int bit_quality[3]; + + int new_gray = phase_shift_to_symbol (a, 3, bit_quality); + + float offset = 5 * 1.5; + printf ("%d, ", degrees); + printf ("%.3f, ", offset + (new_gray & 1)); offset -= 1.5; + printf ("%.3f, ", offset + (bit_quality[0] / 100.)); offset -= 1.5; + printf ("%.3f, ", offset + ((new_gray >> 1) & 1)); offset -= 1.5; + printf ("%.3f, ", offset + (bit_quality[1] / 100.)); offset -= 1.5; + printf ("%.3f, ", offset + ((new_gray >> 2) & 1)); offset -= 1.5; + printf ("%.3f\n", offset + (bit_quality[2] / 100.)); + } +#endif } /* demod_psk_init */ +/*------------------------------------------------------------------- + * + * Name: phase_shift_to_symbol + * + * Purpose: Translate phase shift, between two symbols, into 2 or 3 bits. + * + * Inputs: phase_shift - in radians. + * + * bits_per_symbol - 2 for QPSK, 3 for 8PSK. + * + * Outputs: bit_quality[] - Value of 0 (at threshold) to 100 (perfect) for each bit. + * + * Returns: 2 or 3 bit symbol value in Gray code. + * + *--------------------------------------------------------------------*/ + +__attribute__((hot)) __attribute__((always_inline)) +static inline int phase_shift_to_symbol (float phase_shift, int bits_per_symbol, int * __restrict__ bit_quality) +{ +// Number of different symbol states. + assert (bits_per_symbol == 2 || bits_per_symbol == 3); + int N = 1 << bits_per_symbol; + assert (N == 4 || N == 8); + +// Scale angle to 1 per symbol then separate into integer and fractional parts. + float a = phase_shift * (float)N / (M_PI * 2.0f); + while (a >= (float)N) a -= (float)N; + while (a < 0.) a += (float)N; + int i = (int)a; + if (i == N) i = N-1; // Should be < N. Watch out for possible roundoff errors. + float f = a - (float)i; + assert (i >= 0 && i < N); + assert (f >= -0.001f && f <= 1.001f); + +// Interpolate between the ideal angles to get a level of certainty. + int result = 0; + for (int b = 0; b < bits_per_symbol; b++) { + float demod = bits_per_symbol == 2 ? + ((phase_to_gray_v26[i] >> b) & 1) * (1.0f - f) + ((phase_to_gray_v26[(i+1)&3] >> b) & 1) * f : + ((phase_to_gray_v27[i] >> b) & 1) * (1.0f - f) + ((phase_to_gray_v27[(i+1)&7] >> b) & 1) * f; +// Slice to get boolean value and quality measurement. + if (demod >= 0.5f) result |= 1<= 0 && chan < MAX_CHANS); assert (subchan >= 0 && subchan < MAX_SUBCHANS); - /* Scale to nice number for plotting during debug. */ - fsam = sam / 16384.0f; - + float fsam = sam / 16384.0f; /* * Optional bandpass filter before the phase detector. */ - if (D->use_prefilter) { - push_sample (fsam, D->raw_cb, D->pre_filter_size); - fsam = convolve (D->raw_cb, D->pre_filter, D->pre_filter_size); + if (D->u.psk.use_prefilter) { + push_sample (fsam, D->u.psk.audio_in, D->u.psk.pre_filter_taps); + fsam = convolve (D->u.psk.audio_in, D->u.psk.pre_filter, D->u.psk.pre_filter_taps); } - if (D->psk_use_lo) { - float a, delta; - int id; + if (D->u.psk.psk_use_lo) { /* * Mix with local oscillator to obtain phase. * The absolute phase doesn't matter. * We are just concerned with the change since the previous symbol. */ - sam_x_cos = fsam * D->m_sin_table[((D->lo_phase >> 24) + 64) & 0xff]; - - sam_x_sin = fsam * D->m_sin_table[(D->lo_phase >> 24) & 0xff]; + float sam_x_cos = fsam * D->u.psk.sin_table256[((D->u.psk.lo_phase >> 24) + 64) & 0xff]; + push_sample (sam_x_cos, D->u.psk.I_raw, D->u.psk.lp_filter_taps); + float I = convolve (D->u.psk.I_raw, D->u.psk.lp_filter, D->u.psk.lp_filter_taps); - push_sample (sam_x_cos, D->m_amp_cb, D->lp_filter_size); - I = convolve (D->m_amp_cb, D->lp_filter, D->lp_filter_size); + float sam_x_sin = fsam * D->u.psk.sin_table256[(D->u.psk.lo_phase >> 24) & 0xff]; + push_sample (sam_x_sin, D->u.psk.Q_raw, D->u.psk.lp_filter_taps); + float Q = convolve (D->u.psk.Q_raw, D->u.psk.lp_filter, D->u.psk.lp_filter_taps); - push_sample (sam_x_sin, D->s_amp_cb, D->lp_filter_size); - Q = convolve (D->s_amp_cb, D->lp_filter, D->lp_filter_size); + float a = my_atan2f(I,Q); - a = my_atan2f(I,Q); - push_sample (a, D->ms_in_cb, D->ms_filter_size); + // This is just a delay line of one symbol time. - delta = a - D->ms_in_cb[D->boffs]; + push_sample (a, D->u.psk.delay_line, D->u.psk.delay_line_taps); + float delta = a - D->u.psk.delay_line[D->u.psk.boffs]; - /* 256 units/cycle makes modulo processing easier. */ - /* Make sure it is positive before truncating to integer. */ - - id = ((int)((delta / (2.f * (float)M_PI) + 1.f) * 256.f)) & 0xff; - + int gray; + int bit_quality[3]; if (D->modem_type == MODEM_QPSK) { - demod_phase_shift = ((id + 32) >> 6) & 0x3; + if (D->u.psk.v26_alt == V26_B) { + gray = phase_shift_to_symbol (delta + (float)(-M_PI/4), 2, bit_quality);; // MFJ compatible + } + else { + gray = phase_shift_to_symbol (delta, 2, bit_quality); // Classic + } } else { - demod_phase_shift = ((id + 16) >> 5) & 0x7; + gray = phase_shift_to_symbol (delta, 3, bit_quality);; // 8-PSK } - nudge_pll (chan, subchan, slice, demod_phase_shift, D); + nudge_pll (chan, subchan, slice, gray, D, bit_quality); - D->lo_phase += D->lo_step; + D->u.psk.lo_phase += D->u.psk.lo_step; } else { /* * Correlate with previous symbol. We are looking for the phase shift. */ - push_sample (fsam, D->ms_in_cb, D->ms_filter_size); + push_sample (fsam, D->u.psk.delay_line, D->u.psk.delay_line_taps); - sam_x_cos = fsam * D->ms_in_cb[D->coffs]; - sam_x_sin = fsam * D->ms_in_cb[D->soffs]; + float sam_x_cos = fsam * D->u.psk.delay_line[D->u.psk.coffs]; + push_sample (sam_x_cos, D->u.psk.I_raw, D->u.psk.lp_filter_taps); + float I = convolve (D->u.psk.I_raw, D->u.psk.lp_filter, D->u.psk.lp_filter_taps); - push_sample (sam_x_cos, D->m_amp_cb, D->lp_filter_size); - I = convolve (D->m_amp_cb, D->lp_filter, D->lp_filter_size); + float sam_x_sin = fsam * D->u.psk.delay_line[D->u.psk.soffs]; + push_sample (sam_x_sin, D->u.psk.Q_raw, D->u.psk.lp_filter_taps); + float Q = convolve (D->u.psk.Q_raw, D->u.psk.lp_filter, D->u.psk.lp_filter_taps); - push_sample (sam_x_sin, D->s_amp_cb, D->lp_filter_size); - Q = convolve (D->s_amp_cb, D->lp_filter, D->lp_filter_size); + int gray; + int bit_quality[3]; + float delta = my_atan2f(I,Q); if (D->modem_type == MODEM_QPSK) { - -#if 1 // Speed up special case. - if (I > 0) { - if (Q > 0) - demod_phase_shift = 0; /* 0 to 90 degrees, etc. */ - else - demod_phase_shift = 1; + if (D->u.psk.v26_alt == V26_B) { + gray = phase_shift_to_symbol (delta + (float)(M_PI/2), 2, bit_quality); // MFJ compatible } else { - if (Q > 0) - demod_phase_shift = 3; - else - demod_phase_shift = 2; - } -#else - a = my_atan2f(I,Q); - int id = ((int)((a / (2.f * (float)M_PI) + 1.f) * 256.f)) & 0xff; - // 128 compensates for 180 degree phase shift due - // to 1 1/2 carrier cycles per symbol period. - demod_phase_shift = ((id + 128) >> 6) & 0x3; -#endif - } - else { - float a; - int idelta; - - a = my_atan2f(I,Q); - idelta = ((int)((a / (2.f * (float)M_PI) + 1.f) * 256.f)) & 0xff; - // 32 (90 degrees) compensates for 1800 carrier vs. 1800 baud. - // 16 is to set threshold between constellation points. - demod_phase_shift = ((idelta - 32 - 16) >> 5) & 0x7; - } - - nudge_pll (chan, subchan, slice, demod_phase_shift, D); - } - -#if DEBUG4 - - if (chan == 0) { - - if (1) { - //if (hdlc_rec_gathering (chan, subchan, slice)) { - char fname[30]; - - - if (demod_log_fp == NULL) { - log_file_seq++; - snprintf (fname, sizeof(fname), "demod/%04d.csv", log_file_seq); - //if (log_file_seq == 1) mkdir ("demod", 0777); - if (log_file_seq == 1) mkdir ("demod"); - - demod_log_fp = fopen (fname, "w"); - text_color_set(DW_COLOR_DEBUG); - dw_printf ("Starting demodulator log file %s\n", fname); - fprintf (demod_log_fp, "Audio, sin, cos, *cos, *sin, I, Q, phase, Clock\n"); + gray = phase_shift_to_symbol (delta + (float)(3*M_PI/4), 2, bit_quality); // Classic } - - fprintf (demod_log_fp, "%.3f, %.3f, %.3f, %.3f, %.3f, %.3f, %.2f, %.2f, %.2f\n", - fsam + 2, - - D->ms_in_cb[D->soffs] + 6, - - D->ms_in_cb[D->coffs] + 6, - sam_x_cos + 8, - sam_x_sin + 10, - 2 * I + 12, - 2 * Q + 12, - demod_phase_shift * 2. / 3. + 14., - (D->slicer[slice].data_clock_pll & 0x80000000) ? .5 : .0); - - fflush (demod_log_fp); } else { - if (demod_log_fp != NULL) { - fclose (demod_log_fp); - demod_log_fp = NULL; - } + gray = phase_shift_to_symbol (delta + (float)(3*M_PI/2), 3, bit_quality); } + nudge_pll (chan, subchan, slice, gray, D, bit_quality); } -#endif - } /* end demod_psk_process_sample */ -static const int phase_to_gray_v26[4] = {0, 1, 3, 2}; -static const int phase_to_gray_v27[8] = {1, 0, 2, 3, 7, 6, 4, 5}; - __attribute__((hot)) -inline static void nudge_pll (int chan, int subchan, int slice, int demod_bits, struct demodulator_state_s *D) +static void nudge_pll (int chan, int subchan, int slice, int demod_bits, struct demodulator_state_s *D, int *bit_quality) { /* * Finally, a PLL is used to sample near the centers of the data bits. * - * D points to a demodulator for a channel/subchannel pair so we don't - * have to keep recalculating it. + * D points to a demodulator for a channel/subchannel pair. * * D->data_clock_pll is a SIGNED 32 bit variable. * When it overflows from a large positive value to a negative value, we @@ -783,18 +781,15 @@ inline static void nudge_pll (int chan, int subchan, int slice, int demod_bits, * If we adjust it too quickly, the clock will have too much jitter. * If we adjust it too slowly, it will take too long to lock on to a new signal. * - * Be a little more agressive about adjusting the PLL + * Be a little more aggressive about adjusting the PLL * phase when searching for a signal. * Don't change it as much when locked on to a signal. - * - * I don't think the optimal value will depend on the audio sample rate - * because this happens for each transition from the demodulator. */ - D->slicer[slice].prev_d_c_pll = D->slicer[slice].data_clock_pll; - D->slicer[slice].data_clock_pll += D->pll_step_per_sample; + // Perform the add as unsigned to avoid signed overflow error. + D->slicer[slice].data_clock_pll = (signed)((unsigned)(D->slicer[slice].data_clock_pll) + (unsigned)(D->pll_step_per_sample)); if (D->slicer[slice].data_clock_pll < 0 && D->slicer[slice].prev_d_c_pll >= 0) { @@ -803,26 +798,19 @@ inline static void nudge_pll (int chan, int subchan, int slice, int demod_bits, if (D->modem_type == MODEM_QPSK) { - int gray = phase_to_gray_v26[ demod_bits ]; - -#if DEBUG4 - text_color_set(DW_COLOR_DEBUG); - - dw_printf ("a=%.2f deg, delta=%.2f deg, phaseshift=%d, bits= %d %d \n", - a * 360 / (2*M_PI), delta * 360 / (2*M_PI), demod_bits, (gray >> 1) & 1, gray & 1); + int gray = demod_bits; - //dw_printf ("phaseshift=%d, bits= %d %d \n", demod_bits, (gray >> 1) & 1, gray & 1); -#endif - hdlc_rec_bit (chan, subchan, slice, (gray >> 1) & 1, 0, -1); - hdlc_rec_bit (chan, subchan, slice, gray & 1, 0, -1); + hdlc_rec_bit (chan, subchan, slice, (gray >> 1) & 1, 0, bit_quality[1]); + hdlc_rec_bit (chan, subchan, slice, gray & 1, 0, bit_quality[0]); } else { - int gray = phase_to_gray_v27[ demod_bits ]; + int gray = demod_bits; - hdlc_rec_bit (chan, subchan, slice, (gray >> 2) & 1, 0, -1); - hdlc_rec_bit (chan, subchan, slice, (gray >> 1) & 1, 0, -1); - hdlc_rec_bit (chan, subchan, slice, gray & 1, 0, -1); + hdlc_rec_bit (chan, subchan, slice, (gray >> 2) & 1, 0, bit_quality[2]); + hdlc_rec_bit (chan, subchan, slice, (gray >> 1) & 1, 0, bit_quality[1]); + hdlc_rec_bit (chan, subchan, slice, gray & 1, 0, bit_quality[0]); } + pll_dcd_each_symbol2 (D, chan, subchan, slice); } /* @@ -836,7 +824,9 @@ inline static void nudge_pll (int chan, int subchan, int slice, int demod_bits, if (demod_bits != D->slicer[slice].prev_demod_data) { - if (hdlc_rec_gathering (chan, subchan, slice)) { + pll_dcd_signal_transition2 (D, slice, D->slicer[slice].data_clock_pll); + + if (D->slicer[slice].data_detect) { D->slicer[slice].data_clock_pll = (int)floorf((float)(D->slicer[slice].data_clock_pll) * D->pll_locked_inertia); } else { diff --git a/src/demod_psk.h b/src/demod_psk.h new file mode 100644 index 00000000..134b1996 --- /dev/null +++ b/src/demod_psk.h @@ -0,0 +1,7 @@ + +/* demod_psk.h */ + + +void demod_psk_init (enum modem_t modem_type, enum v26_e v26_alt, int samples_per_sec, int bps, char profile, struct demodulator_state_s *D); + +void demod_psk_process_sample (int chan, int subchan, int sam, struct demodulator_state_s *D); diff --git a/digipeater.c b/src/digipeater.c similarity index 74% rename from digipeater.c rename to src/digipeater.c index 1900d784..fbe89370 100644 --- a/digipeater.c +++ b/src/digipeater.c @@ -49,7 +49,8 @@ * Preemptive Digipeating (new in version 0.8) * * http://www.aprs.org/aprs12/preemptive-digipeating.txt - * + * I ignored the part about the RR bits. + * *------------------------------------------------------------------*/ #define DIGIPEATER_C @@ -62,7 +63,7 @@ #include #include /* for isdigit, isupper */ #include "regex.h" -#include +#include #include "ax25_pad.h" #include "digipeater.h" @@ -73,7 +74,7 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, char *mycall_xmit, - regex_t *uidigi, regex_t *uitrace, int to_chan, enum preempt_e preempt, char *type_filter); + regex_t *uidigi, regex_t *uitrace, int to_chan, enum preempt_e preempt, char *atgp, char *type_filter); /* @@ -149,18 +150,49 @@ void digipeater (int from_chan, packet_t pp) // dw_printf ("digipeater()\n"); - assert (from_chan >= 0 && from_chan < MAX_CHANS); - if ( ! save_audio_config_p->achan[from_chan].valid) { + + // Network TNC is OK for UI frames where we don't care about timing. + + if ( from_chan < 0 || from_chan >= MAX_CHANS || + (save_audio_config_p->chan_medium[from_chan] != MEDIUM_RADIO && + save_audio_config_p->chan_medium[from_chan] != MEDIUM_NETTNC)) { text_color_set(DW_COLOR_ERROR); - dw_printf ("digipeater: Did not expect to receive on invalid channel %d.\n", from_chan); + dw_printf ("APRS digipeater: Did not expect to receive on invalid channel %d.\n", from_chan); } /* * First pass: Look at packets being digipeated to same channel. * - * We want these to get out quickly. + * We want these to get out quickly, bypassing the usual random wait time. + * + * Some may disagree but I followed what WB4APR had to say about it. + * + * http://www.aprs.org/balloons.html + * + * APRS NETWORK FRATRICIDE: Generally, all APRS digipeaters are supposed to transmit + * immediately and all at the same time. They should NOT wait long enough for each + * one to QRM the channel with the same copy of each packet. NO, APRS digipeaters + * are all supposed to STEP ON EACH OTHER with every packet. This makes sure that + * everyone in range of a digi will hear one and only one copy of each packet. + * and that the packet will digipeat OUTWARD and not backward. The goal is that a + * digipeated packet is cleared out of the local area in ONE packet time and not + * N packet times for every N digipeaters that heard the packet. This means no + * PERSIST times, no DWAIT times and no UIDWAIT times. Notice, this is contrary + * to other packet systems that might want to guarantee delivery (but at the + * expense of throughput). APRS wants to clear the channel quickly to maximize throughput. + * + * http://www.aprs.org/kpc3/kpc3+WIDEn.txt + * + * THIRD: Eliminate the settings that are detrimental to the network. + * + * * UIDWAIT should be OFF. (the default). With it on, your digi is not doing the + * fundamental APRS fratricide that is the primary mechanism for minimizing channel + * loading. All digis that hear the same packet are supposed to DIGI it at the SAME + * time so that all those copies only take up one additional time slot. (but outward + * located digs will hear it without collision (and continue outward propagation) + * */ for (to_chan=0; to_chanachan[to_chan].mycall, &save_digi_config_p->alias[from_chan][to_chan], &save_digi_config_p->wide[from_chan][to_chan], to_chan, save_digi_config_p->preempt[from_chan][to_chan], + save_digi_config_p->atgp[from_chan][to_chan], save_digi_config_p->filter_str[from_chan][to_chan]); if (result != NULL) { dedupe_remember (pp, to_chan); - tq_append (to_chan, TQ_PRIO_0_HI, result); + tq_append (to_chan, TQ_PRIO_0_HI, result); // High priority queue. digi_count[from_chan][to_chan]++; } } @@ -198,10 +231,11 @@ void digipeater (int from_chan, packet_t pp) save_audio_config_p->achan[to_chan].mycall, &save_digi_config_p->alias[from_chan][to_chan], &save_digi_config_p->wide[from_chan][to_chan], to_chan, save_digi_config_p->preempt[from_chan][to_chan], + save_digi_config_p->atgp[from_chan][to_chan], save_digi_config_p->filter_str[from_chan][to_chan]); if (result != NULL) { dedupe_remember (pp, to_chan); - tq_append (to_chan, TQ_PRIO_1_LO, result); + tq_append (to_chan, TQ_PRIO_1_LO, result); // Low priority queue. digi_count[from_chan][to_chan]++; } } @@ -240,6 +274,9 @@ void digipeater (int from_chan, packet_t pp) * * preempt - Option for "preemptive" digipeating. * + * atgp - No tracing if this matches alias prefix. + * Hack added for special needs of ATGP. + * * filter_str - Filter expression string or NULL. * * Returns: Packet object for transmission or NULL. @@ -256,32 +293,9 @@ void digipeater (int from_chan, packet_t pp) * *------------------------------------------------------------------------------*/ -#define OBSOLETE14 1 - - -#ifndef OBSOLETE14 -static char *dest_ssid_path[16] = { - "", /* Use VIA path */ - "WIDE1-1", - "WIDE2-2", - "WIDE3-3", - "WIDE4-4", - "WIDE5-5", - "WIDE6-6", - "WIDE7-7", - "WIDE1-1", /* North */ - "WIDE1-1", /* South */ - "WIDE1-1", /* East */ - "WIDE1-1", /* West */ - "WIDE2-2", /* North */ - "WIDE2-2", /* South */ - "WIDE2-2", /* East */ - "WIDE2-2" }; /* West */ -#endif - static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, char *mycall_xmit, - regex_t *alias, regex_t *wide, int to_chan, enum preempt_e preempt, char *filter_str) + regex_t *alias, regex_t *wide, int to_chan, enum preempt_e preempt, char *atgp, char *filter_str) { char source[AX25_MAX_ADDR_LEN]; int ssid; @@ -319,15 +333,6 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, ch * Otherwise we don't want to modify the input because this could be called multiple times. */ -#ifndef OBSOLETE14 // Took it out in 1.4 - - if (ax25_get_num_repeaters(pp) == 0 && (ssid = ax25_get_ssid(pp, AX25_DESTINATION)) > 0) { - ax25_set_addr(pp, AX25_REPEATER_1, dest_ssid_path[ssid]); - ax25_set_ssid(pp, AX25_DESTINATION, 0); - /* Continue with general case, below. */ - } -#endif - /* * Find the first repeater station which doesn't have "has been repeated" set. @@ -436,6 +441,10 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, ch /* * If preemptive digipeating is enabled, try matching my call * and aliases against all remaining unused digipeaters. + * + * Bob says: "GENERIC XXXXn-N DIGIPEATING should not do preemptive digipeating." + * + * But consider this case: https://github.com/wb2osz/direwolf/issues/488 */ if (preempt != PREEMPT_OFF) { @@ -461,13 +470,22 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, ch switch (preempt) { case PREEMPT_DROP: /* remove all prior */ + // TODO: deprecate this option. Result is misleading. + + text_color_set (DW_COLOR_ERROR); + dw_printf ("The digipeat DROP option will be removed in a future release. Use PREEMPT for preemptive digipeating.\n"); + while (r2 > AX25_REPEATER_1) { ax25_remove_addr (result, r2-1); r2--; } break; - case PREEMPT_MARK: + case PREEMPT_MARK: // TODO: deprecate this option. Result is misleading. + + text_color_set (DW_COLOR_ERROR); + dw_printf ("The digipeat MARK option will be removed in a future release. Use PREEMPT for preemptive digipeating.\n"); + r2--; while (r2 >= AX25_REPEATER_1 && ax25_get_h(result,r2) == 0) { ax25_set_h (result, r2); @@ -475,7 +493,12 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, ch } break; - case PREEMPT_TRACE: /* remove prior unused */ + case PREEMPT_TRACE: /* My enhancement - remove prior unused digis. */ + /* this provides an accurate path of where packet traveled. */ + + // Uh oh. It looks like sample config files went out + // with this option. Should it be renamed as + // PREEMPT which is more descriptive? default: while (r2 > AX25_REPEATER_1 && ax25_get_h(result,r2-1) == 0) { ax25_remove_addr (result, r2-1); @@ -484,6 +507,16 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, ch break; } +// Idea: Here is an interesting idea for a new option. REORDER? +// The preemptive digipeater could move its call after the (formerly) last used digi field +// and preserve all the unused fields after that. The list of used addresses would +// accurately record the journey taken by the packet. + +// https://groups.yahoo.com/neo/groups/aprsisce/conversations/topics/31935 + +// > I was wishing for a non-marking preemptive digipeat so that the original packet would be left intact +// > or maybe something like WIDE1-1,WIDE2-1,KJ4OVQ-9 becoming KJ4OVQ-9*,WIDE1-1,WIDE2-1. + return (result); } } @@ -496,6 +529,40 @@ static packet_t digipeat_match (int from_chan, packet_t pp, char *mycall_rec, ch err = regexec(wide,repeater,0,NULL,0); if (err == 0) { +// Special hack added for ATGP to behave like some combination of options in some old TNC +// so the via path does not continue to grow and exceed the 8 available positions. +// The strange thing about this is that the used up digipeater is left there but +// removed by the next digipeater. + + if (strlen(atgp) > 0 && strncasecmp(repeater, atgp, strlen(atgp)) == 0) { + + if (ssid >= 1 && ssid <= 7) { + packet_t result; + + result = ax25_dup (pp); + assert (result != NULL); + + // First, remove any already used digipeaters. + + while (ax25_get_num_addr(result) >= 3 && ax25_get_h(result,AX25_REPEATER_1) == 1) { + ax25_remove_addr (result, AX25_REPEATER_1); + r--; + } + + ssid = ssid - 1; + ax25_set_ssid(result, r, ssid); // could be zero. + if (ssid == 0) { + ax25_set_h (result, r); + } + + // Insert own call at beginning and mark it used. + + ax25_insert_addr (result, AX25_REPEATER_1, mycall_xmit); + ax25_set_h (result, AX25_REPEATER_1); + return (result); + } + } + /* * If ssid == 1, we simply replace the repeater with my call and * mark it as being used. @@ -594,7 +661,7 @@ void digi_regen (int from_chan, packet_t pp) * * Name: main * - * Purpose: Standalone test case for this funtionality. + * Purpose: Standalone test case for this functionality. * * Usage: make -f Makefile. dtest * ./dtest @@ -603,7 +670,7 @@ void digi_regen (int from_chan, packet_t pp) #if DIGITEST -static char mycall[] = "WB2OSZ-9"; +static char mycall[12]; static regex_t alias_re; @@ -613,6 +680,7 @@ static int failed; static enum preempt_e preempt = PREEMPT_OFF; +static char config_atgp[AX25_MAX_ADDR_LEN] = "HOP"; static void test (char *in, char *out) @@ -626,6 +694,7 @@ static void test (char *in, char *out) int frame_len; alevel_t alevel; + dw_printf ("\n"); /* @@ -675,9 +744,9 @@ static void test (char *in, char *out) text_color_set(DW_COLOR_REC); dw_printf ("Rec\t%s\n", rec); -//TODO: Add filtering to test. -// V - result = digipeat_match (0, pp, mycall, mycall, &alias_re, &wide_re, 0, preempt, NULL); +//TODO: Add filtering to test. +// V + result = digipeat_match (0, pp, mycall, mycall, &alias_re, &wide_re, 0, preempt, config_atgp, NULL); if (result != NULL) { @@ -712,6 +781,7 @@ int main (int argc, char *argv[]) int e; failed = 0; char message[256]; + strlcpy(mycall, "WB2OSZ-9", sizeof(mycall)); dedupe_init (4); @@ -726,7 +796,7 @@ int main (int argc, char *argv[]) exit (1); } - e = regcomp (&wide_re, "^WIDE[1-7]-[1-7]$|^TRACE[1-7]-[1-7]$|^MA[1-7]-[1-7]$", REG_EXTENDED|REG_NOSUB); + e = regcomp (&wide_re, "^WIDE[1-7]-[1-7]$|^TRACE[1-7]-[1-7]$|^MA[1-7]-[1-7]$|^HOP[1-7]-[1-7]$", REG_EXTENDED|REG_NOSUB); if (e != 0) { regerror (e, &wide_re, message, sizeof(message)); text_color_set(DW_COLOR_ERROR); @@ -816,11 +886,8 @@ int main (int argc, char *argv[]) */ test ( "W1ABC>TEST-3:", -#ifndef OBSOLETE14 - "W1ABC>TEST,WB2OSZ-9*,WIDE3-2:"); -#else ""); -#endif + test ( "W1DEF>TEST-3,WIDE2-2:", "W1DEF>TEST-3,WB2OSZ-9*,WIDE2-1:"); @@ -830,17 +897,17 @@ int main (int argc, char *argv[]) * The 4th case might be controversial. */ - test ( "W1XYZ>TEST,R1*,WIDE3-2:info1", - "W1XYZ>TEST,R1,WB2OSZ-9*,WIDE3-1:info1"); + test ( "W1XYZ>TESTD,R1*,WIDE3-2:info1", + "W1XYZ>TESTD,R1,WB2OSZ-9*,WIDE3-1:info1"); - test ( "W1XYZ>TEST,R2*,WIDE3-2:info1", + test ( "W1XYZ>TESTD,R2*,WIDE3-2:info1", ""); - test ( "W1XYZ>TEST,R3*,WIDE3-2:info1", + test ( "W1XYZ>TESTD,R3*,WIDE3-2:info1", ""); - test ( "W1XYZ>TEST,R1*,WB2OSZ-9:has explicit routing", - "W1XYZ>TEST,R1,WB2OSZ-9*:has explicit routing"); + test ( "W1XYZ>TESTD,R1*,WB2OSZ-9:has explicit routing", + "W1XYZ>TESTD,R1,WB2OSZ-9*:has explicit routing"); /* @@ -917,6 +984,54 @@ int main (int argc, char *argv[]) test ( "WB2OSZ-15>TEST14,WIDE1-1,WIDE1-1:stuff", "WB2OSZ-15>TEST14,WB2OSZ-9*,WIDE1-1:stuff"); +// New in 1.7 - ATGP Hack + + preempt = PREEMPT_OFF; // Shouldn't make a difference here. + + test ( "W1ABC>TEST51,HOP7-7,HOP7-7:stuff1", + "W1ABC>TEST51,WB2OSZ-9*,HOP7-6,HOP7-7:stuff1"); + + test ( "W1ABC>TEST52,ABCD*,HOP7-1,HOP7-7:stuff2", + "W1ABC>TEST52,WB2OSZ-9,HOP7*,HOP7-7:stuff2"); // Used up address remains. + + test ( "W1ABC>TEST53,HOP7*,HOP7-7:stuff3", + "W1ABC>TEST53,WB2OSZ-9*,HOP7-6:stuff3"); // But it gets removed here. + + test ( "W1ABC>TEST54,HOP7*,HOP7-1:stuff4", + "W1ABC>TEST54,WB2OSZ-9,HOP7*:stuff4"); // Remains again here. + + test ( "W1ABC>TEST55,HOP7,HOP7*:stuff5", + ""); + +// Examples given for desired result. + + strlcpy (mycall, "CLNGMN-1", sizeof(mycall)); + test ( "W1ABC>TEST60,HOP7-7,HOP7-7:", + "W1ABC>TEST60,CLNGMN-1*,HOP7-6,HOP7-7:"); + test ( "W1ABC>TEST61,ROAN-3*,HOP7-6,HOP7-7:", + "W1ABC>TEST61,CLNGMN-1*,HOP7-5,HOP7-7:"); + + strlcpy (mycall, "GDHILL-8", sizeof(mycall)); + test ( "W1ABC>TEST62,MDMTNS-7*,HOP7-1,HOP7-7:", + "W1ABC>TEST62,GDHILL-8,HOP7*,HOP7-7:"); + test ( "W1ABC>TEST63,CAMLBK-9*,HOP7-1,HOP7-7:", + "W1ABC>TEST63,GDHILL-8,HOP7*,HOP7-7:"); + + strlcpy (mycall, "MDMTNS-7", sizeof(mycall)); + test ( "W1ABC>TEST64,GDHILL-8*,HOP7*,HOP7-7:", + "W1ABC>TEST64,MDMTNS-7*,HOP7-6:"); + + strlcpy (mycall, "CAMLBK-9", sizeof(mycall)); + test ( "W1ABC>TEST65,GDHILL-8,HOP7*,HOP7-7:", + "W1ABC>TEST65,CAMLBK-9*,HOP7-6:"); + + strlcpy (mycall, "KATHDN-15", sizeof(mycall)); + test ( "W1ABC>TEST66,MTWASH-14*,HOP7-1:", + "W1ABC>TEST66,KATHDN-15,HOP7*:"); + + strlcpy (mycall, "SPRNGR-1", sizeof(mycall)); + test ( "W1ABC>TEST67,CLNGMN-1*,HOP7-1:", + "W1ABC>TEST67,SPRNGR-1,HOP7*:"); if (failed == 0) { diff --git a/digipeater.h b/src/digipeater.h similarity index 88% rename from digipeater.h rename to src/digipeater.h index d1ad3b6c..5c849769 100644 --- a/digipeater.h +++ b/src/digipeater.h @@ -1,5 +1,4 @@ - #ifndef DIGIPEATER_H #define DIGIPEATER_H 1 @@ -38,7 +37,10 @@ struct digi_config_s { enum preempt_e { PREEMPT_OFF, PREEMPT_DROP, PREEMPT_MARK, PREEMPT_TRACE } preempt[MAX_CHANS][MAX_CHANS]; - //char type_filter[MAX_CHANS][MAX_CHANS][20]; // TODO1.2: remove this + // ATGP is an ugly hack for the specific need of ATGP which needs more that 8 digipeaters. + // DO NOT put this in the User Guide. On a need to know basis. + + char atgp[MAX_CHANS][MAX_CHANS][AX25_MAX_ADDR_LEN]; char *filter_str[MAX_CHANS+1][MAX_CHANS+1]; // NULL or optional Packet Filter strings such as "t/m". diff --git a/direwolf.c b/src/direwolf.c similarity index 56% rename from direwolf.c rename to src/direwolf.c index 1a562b87..e23aecb4 100644 --- a/direwolf.c +++ b/src/direwolf.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2019, 2020, 2021, 2023 John Langner, WB2OSZ // // 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 @@ -24,14 +24,18 @@ * * Purpose: Main program for "Dire Wolf" which includes: * - * AFSK modem using the "sound card." + * Various DSP modems using the "sound card." * AX.25 encoder/decoder. * APRS data encoder / decoder. * APRS digipeater. * KISS TNC emulator. * APRStt (touch tone input) gateway * Internet Gateway (IGate) - * + * Ham Radio of Things - IoT with Ham Radio + * FX.25 Forward Error Correction. + * IL2P Forward Error Correction. + * Emergency Alert System (EAS) Specific Area Message Encoding (SAME) receiver. + * AIS receiver for tracking ships. * *---------------------------------------------------------------*/ @@ -58,14 +62,15 @@ #endif #if __WIN32__ +#include +#include #else #include #include #include #include -#ifdef __OpenBSD__ -#include -#elif __APPLE__ +#if USE_SNDIO || __APPLE__ +// no need to include #else #include #endif @@ -90,10 +95,12 @@ #include "ax25_pad.h" #include "xid.h" #include "decode_aprs.h" +#include "encode_aprs.h" #include "textcolor.h" #include "server.h" #include "kiss.h" #include "kissnet.h" +#include "kissserial.h" #include "kiss_frame.h" #include "waypoint.h" #include "gen_tone.h" @@ -116,6 +123,12 @@ #include "morse.h" #include "mheard.h" #include "ax25_link.h" +#include "dtime_now.h" +#include "fx25.h" +#include "il2p.h" +#include "dwsock.h" +#include "dns_sd_dw.h" +#include "dlq.h" // for fec_type_t definition. //static int idx_decoded = 0; @@ -126,7 +139,7 @@ static BOOL cleanup_win (int); static void cleanup_linux (int); #endif -static void usage (char **argv); +static void usage (); #if defined(__SSE__) && !defined(__APPLE__) @@ -162,8 +175,8 @@ static void __cpuid(int cpuinfo[4], int infotype){ static struct audio_s audio_config; static struct tt_config_s tt_config; -//struct digi_config_s digi_config; -//struct cdigi_config_s cdigi_config; +static struct misc_config_s misc_config; + static const int audio_amplitude = 100; /* % of audio sample range. */ /* This translates to +-32k for 16 bit samples. */ @@ -175,9 +188,7 @@ static int d_p_opt = 0; /* "-d p" option for dumping packets over radio. */ static int q_h_opt = 0; /* "-q h" Quiet, suppress the "heard" line with audio level. */ static int q_d_opt = 0; /* "-q d" Quiet, suppress the printing of decoded of APRS packets. */ - - -static struct misc_config_s misc_config; +static int A_opt_ais_to_obj = 0; /* "-A" Convert received AIS to APRS "Object Report." */ int main (int argc, char *argv[]) @@ -186,18 +197,22 @@ int main (int argc, char *argv[]) //int eof; int j; char config_file[100]; - int xmit_calibrate_option = 0; int enable_pseudo_terminal = 0; struct digi_config_s digi_config; struct cdigi_config_s cdigi_config; struct igate_config_s igate_config; - int r_opt = 0, n_opt = 0, b_opt = 0, B_opt = 0, D_opt = 0; /* Command line options. */ + int r_opt = 0, n_opt = 0, b_opt = 0, B_opt = 0, D_opt = 0, U_opt = 0; /* Command line options. */ char P_opt[16]; - char l_opt[80]; + char l_opt_logdir[80]; + char L_opt_logfile[80]; char input_file[80]; + char T_opt_timestamp[40]; int t_opt = 1; /* Text color option. */ int a_opt = 0; /* "-a n" interval, in seconds, for audio statistics report. 0 for none. */ + int g_opt = 0; /* G3RUH mode, ignoring default for speed. */ + int j_opt = 0; /* 2400 bps PSK compatible with direwolf <= 1.5 */ + int J_opt = 0; /* 2400 bps PSK compatible MFJ-2400 and maybe others. */ int d_k_opt = 0; /* "-d k" option for serial port KISS. Can be repeated for more detail. */ int d_n_opt = 0; /* "-d n" option for Network KISS. Can be repeated for more detail. */ @@ -210,11 +225,27 @@ int main (int argc, char *argv[]) #if USE_HAMLIB int d_h_opt = 0; /* "-d h" option for hamlib debugging. Repeat for more detail */ #endif - int E_tx_opt = 0; /* "-E n" Error rate % for clobbering trasmit frames. */ + int d_x_opt = 1; /* "-d x" option for FX.25. Default minimal. Repeat for more detail. -qx to silence. */ + int d_2_opt = 0; /* "-d 2" option for IL2P. Default minimal. Repeat for more detail. */ + + int aprstt_debug = 0; /* "-d d" option for APRStt (think Dtmf) debug. */ + + int E_tx_opt = 0; /* "-E n" Error rate % for clobbering transmit frames. */ int E_rx_opt = 0; /* "-E Rn" Error rate % for clobbering receive frames. */ - strlcpy(l_opt, "", sizeof(l_opt)); + float e_recv_ber = 0.0; /* Receive Bit Error Rate (BER). */ + int X_fx25_xmit_enable = 0; /* FX.25 transmit enable. */ + + int I_opt = -1; /* IL2P transmit, normal polarity, arg is max_fec. */ + int i_opt = -1; /* IL2P transmit, inverted polarity, arg is max_fec. */ + + char x_opt_mode = ' '; /* "-x N" option for transmitting calibration tones. */ + int x_opt_chan = 0; /* Split into 2 parts. Mode e.g. m, a, and optional channel. */ + + strlcpy(l_opt_logdir, "", sizeof(l_opt_logdir)); + strlcpy(L_opt_logfile, "", sizeof(L_opt_logfile)); strlcpy(P_opt, "", sizeof(P_opt)); + strlcpy(T_opt_timestamp, "", sizeof(T_opt_timestamp)); #if __WIN32__ @@ -241,9 +272,20 @@ int main (int argc, char *argv[]) /* * Pre-scan the command line options for the text color option. * We need to set this before any text output. + * Default will be no colors if stdout is not a terminal (i.e. piped into + * something else such as "tee") but command line can override this. */ - t_opt = 1; /* 1 = normal, 0 = no text colors. */ +#if __WIN32__ + t_opt = _isatty(_fileno(stdout)) > 0; +#else + t_opt = isatty(fileno(stdout)); +#endif + /* 1 = normal, 0 = no text colors. */ + /* 2, 3, ... alternate escape sequences for different terminals. */ + +// FIXME: consider case of no space between t and number. + for (j=1; j= 1) { __cpuid (cpuinfo, 1); //dw_printf ("debug: cpuinfo = %x, %x, %x, %x\n", cpuinfo[0], cpuinfo[1], cpuinfo[2], cpuinfo[3]); + // https://en.wikipedia.org/wiki/CPUID if ( ! ( cpuinfo[3] & (1 << 25))) { text_color_set(DW_COLOR_ERROR); dw_printf ("------------------------------------------------------------------\n"); dw_printf ("This version requires a minimum of a Pentium 3 or equivalent.\n"); dw_printf ("If you are seeing this message, you are probably using a computer\n"); - dw_printf ("from the previous century. See comments in Makefile.win for\n"); - dw_printf ("information on how you can recompile it for use with your antique.\n"); + dw_printf ("from the previous Century. See instructions in User Guide for\n"); + dw_printf ("information on how you can compile it for use with your antique.\n"); dw_printf ("------------------------------------------------------------------\n"); } } text_color_set(DW_COLOR_INFO); #endif +// I've seen many references to people running this as root. +// There is no reason to do that. +// Ordinary users can access audio, gpio, etc. if they are in the correct groups. +// Giving an applications permission to do things it does not need to do +// is a huge security risk. +#ifndef __WIN32__ + if (getuid() == 0 || geteuid() == 0) { + text_color_set(DW_COLOR_ERROR); + for (int n=0; n<15; n++) { + dw_printf ("\n"); + dw_printf ("Dire Wolf requires only privileges available to ordinary users.\n"); + dw_printf ("Running this as root is an unnecessary security risk.\n"); + //SLEEP_SEC(1); + } + } +#endif /* * Default location of configuration file is current directory. @@ -345,7 +421,7 @@ int main (int argc, char *argv[]) /* ':' following option character means arg is required. */ - c = getopt_long(argc, argv, "P:B:D:c:pxr:b:n:d:q:t:Ul:Sa:E:", + c = getopt_long(argc, argv, "hP:B:gjJD:U:c:px:r:b:n:d:q:t:ul:L:Sa:E:T:e:X:AI:i:", long_options, &option_index); if (c == -1) break; @@ -388,8 +464,17 @@ int main (int argc, char *argv[]) #endif case 'B': /* -B baud rate and modem properties. */ - - B_opt = atoi(optarg); + /* Also implies modem type based on speed. */ + /* Special case "AIS" rather than number. */ + if (strcasecmp(optarg, "AIS") == 0) { + B_opt = 12345; // See special case below. + } + else if (strcasecmp(optarg, "EAS") == 0) { + B_opt = 23456; // See special case below. + } + else { + B_opt = atoi(optarg); + } if (B_opt < MIN_BAUD || B_opt > MAX_BAUD) { text_color_set(DW_COLOR_ERROR); dw_printf ("Use a more reasonable data baud rate in range of %d - %d.\n", MIN_BAUD, MAX_BAUD); @@ -397,13 +482,28 @@ int main (int argc, char *argv[]) } break; + case 'g': /* -g G3RUH modem, overriding default mode for speed. */ + + g_opt = 1; + break; + + case 'j': /* -j V.26 compatible with earlier direwolf. */ + + j_opt = 1; + break; + + case 'J': /* -J V.26 compatible with MFJ-2400. */ + + J_opt = 1; + break; + case 'P': /* -P for modem profile. */ //debug: dw_printf ("Demodulator profile set to \"%s\"\n", optarg); strlcpy (P_opt, optarg, sizeof(P_opt)); break; - case 'D': /* -D decrease AFSK demodulator sample rate */ + case 'D': /* -D divide AFSK demodulator sample rate */ D_opt = atoi(optarg); if (D_opt < 1 || D_opt > 8) { @@ -413,9 +513,53 @@ int main (int argc, char *argv[]) } break; - case 'x': /* -x for transmit calibration tones. */ + case 'U': /* -U multiply G3RUH demodulator sample rate (upsample) */ + + U_opt = atoi(optarg); + if (U_opt < 1 || U_opt > 4) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Crazy value for -U. \n"); + exit (EXIT_FAILURE); + } + break; - xmit_calibrate_option = 1; + case 'x': /* -x N for transmit calibration tones. */ + /* N is composed of a channel number and/or one letter */ + /* for the mode: mark, space, alternate, ptt-only. */ + + for (char *p = optarg; *p != '\0'; p++ ) { + switch (*p) { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + x_opt_chan = x_opt_chan * 10 + *p - '0'; + if (x_opt_mode == ' ') x_opt_mode = 'a'; + break; + case 'a': x_opt_mode = *p; break; // Alternating tones + case 'm': x_opt_mode = *p; break; // Mark tone + case 's': x_opt_mode = *p; break; // Space tone + case 'p': x_opt_mode = *p; break; // Set PTT only + default: + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid option '%c' for -x. Must be a, m, s, or p.\n", *p); + text_color_set(DW_COLOR_INFO); + exit (EXIT_FAILURE); + break; + } + } + if (x_opt_chan < 0 || x_opt_chan >= MAX_CHANS) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid channel %d for -x. \n", x_opt_chan); + text_color_set(DW_COLOR_INFO); + exit (EXIT_FAILURE); + } break; case 'r': /* -r audio samples/sec. e.g. 44100 */ @@ -451,10 +595,11 @@ int main (int argc, char *argv[]) } break; + case 'h': // -h for help case '?': - /* Unknown option message was already printed. */ - usage (argv); + /* For '?' unknown option message was already printed. */ + usage (); break; case 'd': /* Set debug option. */ @@ -466,7 +611,7 @@ int main (int argc, char *argv[]) case 'a': server_set_debug(1); break; - case 'k': d_k_opt++; kiss_serial_set_debug (d_k_opt); break; + case 'k': d_k_opt++; kissserial_set_debug (d_k_opt); kisspt_set_debug (d_k_opt); break; case 'n': d_n_opt++; kiss_net_set_debug (d_n_opt); break; case 'u': d_u_opt = 1; break; @@ -488,6 +633,9 @@ int main (int argc, char *argv[]) #if USE_HAMLIB case 'h': d_h_opt++; break; // Hamlib verbose level. #endif + case 'x': d_x_opt++; break; // FX.25 + case '2': d_2_opt++; break; // IL2P + case 'd': aprstt_debug++; break; // APRStt (mnemonic Dtmf) default: break; } } @@ -502,6 +650,7 @@ int main (int argc, char *argv[]) switch (*p) { case 'h': q_h_opt = 1; break; case 'd': q_d_opt = 1; break; + case 'x': d_x_opt = 0; break; // Defaults to minimal info. This silences. default: break; } } @@ -511,7 +660,7 @@ int main (int argc, char *argv[]) break; - case 'U': /* Print UTF-8 test and exit. */ + case 'u': /* Print UTF-8 test and exit. */ dw_printf ("\n UTF-8 test string: ma%c%cana %c%c F%c%c%c%ce\n\n", 0xc3, 0xb1, @@ -521,11 +670,17 @@ int main (int argc, char *argv[]) exit (0); break; - case 'l': /* -l for log file directory name */ + case 'l': /* -l for log directory with daily files */ - strlcpy (l_opt, optarg, sizeof(l_opt)); + strlcpy (l_opt_logdir, optarg, sizeof(l_opt_logdir)); break; + case 'L': /* -L for log file name with full path */ + + strlcpy (L_opt_logfile, optarg, sizeof(L_opt_logfile)); + break; + + case 'S': /* Print symbol tables and exit. */ symbols_init (); @@ -554,12 +709,41 @@ int main (int argc, char *argv[]) } break; + case 'T': /* -T for receive timestamp. */ + strlcpy (T_opt_timestamp, optarg, sizeof(T_opt_timestamp)); + break; + + case 'e': /* -e Receive Bit Error Rate (BER). */ + + e_recv_ber = atof(optarg); + break; + + case 'X': + + X_fx25_xmit_enable = atoi(optarg); + break; + + case 'I': // IL2P, normal polarity + + I_opt = atoi(optarg); + break; + + case 'i': // IL2P, inverted polarity + + i_opt = atoi(optarg); + break; + + case 'A': // -A convert AIS to APRS object + + A_opt_ais_to_obj = 1; + break; + default: /* Should not be here. */ text_color_set(DW_COLOR_DEBUG); dw_printf("?? getopt returned character code 0%o ??\n", c); - usage (argv); + usage (); } } /* end while(1) for options */ @@ -588,20 +772,25 @@ int main (int argc, char *argv[]) symbols_init (); + (void)dwsock_init(); + config_init (config_file, &audio_config, &digi_config, &cdigi_config, &tt_config, &igate_config, &misc_config); if (r_opt != 0) { audio_config.adev[0].samples_per_sec = r_opt; } + if (n_opt != 0) { audio_config.adev[0].num_channels = n_opt; if (n_opt == 2) { - audio_config.achan[1].valid = 1; + audio_config.chan_medium[1] = MEDIUM_RADIO; } } + if (b_opt != 0) { audio_config.adev[0].bits_per_sample = b_opt; } + if (B_opt != 0) { audio_config.achan[0].baud = B_opt; @@ -637,6 +826,20 @@ int main (int argc, char *argv[]) dw_printf ("Bit rate should be standard 4800 rather than specified %d.\n", audio_config.achan[0].baud); } } + else if (audio_config.achan[0].baud == 12345) { + audio_config.achan[0].modem_type = MODEM_AIS; + audio_config.achan[0].baud = 9600; + audio_config.achan[0].mark_freq = 0; + audio_config.achan[0].space_freq = 0; + } + else if (audio_config.achan[0].baud == 23456) { + audio_config.achan[0].modem_type = MODEM_EAS; + audio_config.achan[0].baud = 521; // Actually 520.83 but we have an integer field here. + // Will make more precise in afsk demod init. + audio_config.achan[0].mark_freq = 2083; // Actually 2083.3 - logic 1. + audio_config.achan[0].space_freq = 1563; // Actually 1562.5 - logic 0. + strlcpy (audio_config.achan[0].profiles, "A", sizeof(audio_config.achan[0].profiles)); + } else { audio_config.achan[0].modem_type = MODEM_SCRAMBLE; audio_config.achan[0].mark_freq = 0; @@ -644,11 +847,44 @@ int main (int argc, char *argv[]) } } + if (g_opt) { + + // Force G3RUH mode, overriding default for speed. + // Example: -B 2400 -g + + audio_config.achan[0].modem_type = MODEM_SCRAMBLE; + audio_config.achan[0].mark_freq = 0; + audio_config.achan[0].space_freq = 0; + } + + if (j_opt) { + + // V.26 compatible with earlier versions of direwolf. + // Example: -B 2400 -j or simply -j + + audio_config.achan[0].v26_alternative = V26_A; + audio_config.achan[0].modem_type = MODEM_QPSK; + audio_config.achan[0].mark_freq = 0; + audio_config.achan[0].space_freq = 0; + audio_config.achan[0].baud = 2400; + } + if (J_opt) { + + // V.26 compatible with MFJ and maybe others. + // Example: -B 2400 -J or simply -J + + audio_config.achan[0].v26_alternative = V26_B; + audio_config.achan[0].modem_type = MODEM_QPSK; + audio_config.achan[0].mark_freq = 0; + audio_config.achan[0].space_freq = 0; + audio_config.achan[0].baud = 2400; + } + + audio_config.statistics_interval = a_opt; if (strlen(P_opt) > 0) { /* -P for modem profile. */ - /* TODO: Not yet documented. Should probably since it is consistent with atest. */ strlcpy (audio_config.achan[0].profiles, P_opt, sizeof(audio_config.achan[0].profiles)); } @@ -657,14 +893,34 @@ int main (int argc, char *argv[]) audio_config.achan[0].decimate = D_opt; } + if (U_opt != 0) { + // Increase G3RUH audio sampling rate to improve performance. + // The value is normally determined automatically based on audio + // sample rate and baud. This allows override for experimentation. + audio_config.achan[0].upsample = U_opt; + } + + strlcpy(audio_config.timestamp_format, T_opt_timestamp, sizeof(audio_config.timestamp_format)); + // temp - only xmit errors. audio_config.xmit_error_rate = E_tx_opt; audio_config.recv_error_rate = E_rx_opt; - if (strlen(l_opt) > 0) { - strlcpy (misc_config.logdir, l_opt, sizeof(misc_config.logdir)); + if (strlen(l_opt_logdir) > 0 && strlen(L_opt_logfile) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Logging options -l and -L can't be used together. Pick one or the other.\n"); + exit(1); + } + + if (strlen(L_opt_logfile) > 0) { + misc_config.log_daily_names = 0; + strlcpy (misc_config.log_path, L_opt_logfile, sizeof(misc_config.log_path)); + } + else if (strlen(l_opt_logdir) > 0) { + misc_config.log_daily_names = 1; + strlcpy (misc_config.log_path, l_opt_logdir, sizeof(misc_config.log_path)); } misc_config.enable_kiss_pt = enable_pseudo_terminal; @@ -675,6 +931,48 @@ int main (int argc, char *argv[]) } + audio_config.recv_ber = e_recv_ber; + + if (X_fx25_xmit_enable > 0) { + if (I_opt != -1 || i_opt != -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't mix -X with -I or -i.\n"); + exit (EXIT_FAILURE); + } + audio_config.achan[0].fx25_strength = X_fx25_xmit_enable; + audio_config.achan[0].layer2_xmit = LAYER2_FX25; + } + + if (I_opt != -1 && i_opt != -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't use both -I and -i at the same time.\n"); + exit (EXIT_FAILURE); + } + + if (I_opt >= 0) { + audio_config.achan[0].layer2_xmit = LAYER2_IL2P; + audio_config.achan[0].il2p_max_fec = (I_opt > 0); + if (audio_config.achan[0].il2p_max_fec == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("It is highly recommended that 1, rather than 0, is used with -I for best results.\n"); + } + audio_config.achan[0].il2p_invert_polarity = 0; // normal + } + + if (i_opt >= 0) { + audio_config.achan[0].layer2_xmit = LAYER2_IL2P; + audio_config.achan[0].il2p_max_fec = (i_opt > 0); + if (audio_config.achan[0].il2p_max_fec == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("It is highly recommended that 1, rather than 0, is used with -i for best results.\n"); + } + audio_config.achan[0].il2p_invert_polarity = 1; // invert for transmit + if (audio_config.achan[0].baud == 1200) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Using -i with 1200 bps is a bad idea. Use -I instead.\n"); + } + } + /* * Open the audio source @@ -690,19 +988,22 @@ int main (int argc, char *argv[]) text_color_set(DW_COLOR_ERROR); dw_printf ("Pointless to continue without audio device.\n"); SLEEP_SEC(5); + usage (); exit (1); } /* - * Initialize the demodulator(s) and HDLC decoder. + * Initialize the demodulator(s) and layer 2 decoder (HDLC, IL2P). */ multi_modem_init (&audio_config); + fx25_init (d_x_opt); + il2p_init (d_2_opt); /* * Initialize the touch tone decoder & APRStt gateway. */ dtmf_init (&audio_config, audio_amplitude); - aprs_tt_init (&tt_config); + aprs_tt_init (&tt_config, aprstt_debug); tt_user_init (&audio_config, &tt_config); /* @@ -724,30 +1025,77 @@ int main (int argc, char *argv[]) xmit_init (&audio_config, d_p_opt); /* - * If -x option specified, transmit alternating tones for transmitter + * If -x N option specified, transmit calibration tones for transmitter * audio level adjustment, up to 1 minute then quit. - * TODO: enhance for more than one channel. + * a: Alternating mark/space tones + * m: Mark tone (e.g. 1200Hz) + * s: Space tone (e.g. 2200Hz) + * p: Set PTT only. + * A leading or trailing number is the channel. */ - if (xmit_calibrate_option) { - - int max_duration = 60; /* seconds */ - int n = audio_config.achan[0].baud * max_duration; - int chan = 0; - - text_color_set(DW_COLOR_INFO); - dw_printf ("\nSending transmit calibration tones. Press control-C to terminate.\n"); - - ptt_set (OCTYPE_PTT, chan, 1); - while (n-- > 0) { - - tone_gen_put_bit (chan, n & 1); - - } - ptt_set (OCTYPE_PTT, chan, 0); - exit (0); + if (x_opt_mode != ' ') { + if (audio_config.chan_medium[x_opt_chan] == MEDIUM_RADIO) { + if (audio_config.achan[x_opt_chan].mark_freq + && audio_config.achan[x_opt_chan].space_freq) { + int max_duration = 60; + int n = audio_config.achan[x_opt_chan].baud * max_duration; + + text_color_set(DW_COLOR_INFO); + ptt_set(OCTYPE_PTT, x_opt_chan, 1); + + switch (x_opt_mode) { + default: + case 'a': // Alternating tones: -x a + dw_printf("\nSending alternating mark/space calibration tones (%d/%dHz) on channel %d.\nPress control-C to terminate.\n", + audio_config.achan[x_opt_chan].mark_freq, + audio_config.achan[x_opt_chan].space_freq, + x_opt_chan); + while (n-- > 0) { + tone_gen_put_bit(x_opt_chan, n & 1); + } + break; + case 'm': // "Mark" tone: -x m + dw_printf("\nSending mark calibration tone (%dHz) on channel %d.\nPress control-C to terminate.\n", + audio_config.achan[x_opt_chan].mark_freq, + x_opt_chan); + while (n-- > 0) { + tone_gen_put_bit(x_opt_chan, 1); + } + break; + case 's': // "Space" tone: -x s + dw_printf("\nSending space calibration tone (%dHz) on channel %d.\nPress control-C to terminate.\n", + audio_config.achan[x_opt_chan].space_freq, + x_opt_chan); + while (n-- > 0) { + tone_gen_put_bit(x_opt_chan, 0); + } + break; + case 'p': // Silence - set PTT only: -x p + dw_printf("\nSending silence (Set PTT only) on channel %d.\nPress control-C to terminate.\n", x_opt_chan); + SLEEP_SEC(max_duration); + break; + } + + ptt_set(OCTYPE_PTT, x_opt_chan, 0); + text_color_set(DW_COLOR_INFO); + exit(EXIT_SUCCESS); + + } else { + text_color_set(DW_COLOR_ERROR); + dw_printf("\nMark/Space frequencies not defined for channel %d. Cannot calibrate using this modem type.\n", x_opt_chan); + text_color_set(DW_COLOR_INFO); + exit(EXIT_FAILURE); + } + } else { + text_color_set(DW_COLOR_ERROR); + dw_printf("\nChannel %d is not configured as a radio channel.\n", x_opt_chan); + text_color_set(DW_COLOR_INFO); + exit(EXIT_FAILURE); + } } + /* * Initialize the digipeater and IGate functions. */ @@ -763,10 +1111,16 @@ int main (int argc, char *argv[]) server_init (&audio_config, &misc_config); kissnet_init (&misc_config); +#if (USE_AVAHI_CLIENT|USE_MACOS_DNSSD) + if (misc_config.kiss_port > 0 && misc_config.dns_sd_enabled) + dns_sd_announce(&misc_config); +#endif + /* * Create a pseudo terminal and KISS TNC emulator. */ - kiss_init (&misc_config); + kisspt_init (&misc_config); + kissserial_init (&misc_config); kiss_frame_init (&audio_config); /* @@ -782,7 +1136,7 @@ int main (int argc, char *argv[]) * log the tracker beacon transmissions with fake channel 999. */ - log_init(misc_config.logdir); + log_init(misc_config.log_daily_names, misc_config.log_path); mheard_init (d_m_opt); beacon_init (&audio_config, &misc_config, &igate_config); @@ -826,7 +1180,7 @@ int main (int argc, char *argv[]) // TODO: Use only one printf per line so output doesn't get jumbled up with stuff from other threads. -void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, char *spectrum) +void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, fec_type_t fec_type, retry_t retries, char *spectrum) { char stemp[500]; @@ -835,16 +1189,31 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev char heard[AX25_MAX_ADDR_LEN]; //int j; int h; - char display_retries[32]; + char display_retries[32]; // Extra stuff before slice indicators. + // Can indicate FX.25/IL2P or fix_bits. - assert (chan >= 0 && chan < MAX_CHANS); - assert (subchan >= -1 && subchan < MAX_SUBCHANS); + assert (chan >= 0 && chan < MAX_TOTAL_CHANS); // TOTAL for virtual channels + assert (subchan >= -2 && subchan < MAX_SUBCHANS); assert (slice >= 0 && slice < MAX_SLICERS); assert (pp != NULL); // 1.1J+ strlcpy (display_retries, "", sizeof(display_retries)); - if (audio_config.achan[chan].fix_bits != RETRY_NONE || audio_config.achan[chan].passall) { - snprintf (display_retries, sizeof(display_retries), " [%s] ", retry_text[(int)retries]); + + switch (fec_type) { + case fec_type_fx25: + strlcpy (display_retries, " FX.25 ", sizeof(display_retries)); + break; + case fec_type_il2p: + strlcpy (display_retries, " IL2P ", sizeof(display_retries)); + break; + case fec_type_none: + default: + // Possible fix_bits indication. + if (audio_config.achan[chan].fix_bits != RETRY_NONE || audio_config.achan[chan].passall) { + assert (retries >= RETRY_NONE && retries <= RETRY_MAX); + snprintf (display_retries, sizeof(display_retries), " [%s] ", retry_text[(int)retries]); + } + break; } ax25_format_addrs (pp, stemp); @@ -869,8 +1238,11 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev text_color_set(DW_COLOR_DEBUG); dw_printf ("\n"); - if (( ! q_h_opt ) && alevel.rec >= 0) { /* suppress if "-q h" option */ +// The HEARD line. + if (( ! q_h_opt ) && alevel.rec >= 0) { /* suppress if "-q h" option */ +// FIXME: rather than checking for ichannel, how about checking medium==radio + if (chan != audio_config.igate_vchannel) { // suppress if from ICHANNEL if (h != -1 && h != AX25_SOURCE) { dw_printf ("Digipeater "); } @@ -879,6 +1251,14 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev ax25_alevel_to_text (alevel, alevel_text); +// Experiment: try displaying the DC bias. +// Should be 0 for soundcard but could show mistuning with SDR. + +#if 0 + char bias[16]; + snprintf (bias, sizeof(bias), " DC%+d", multi_modem_get_dc_average (chan)); + strlcat (alevel_text, bias, sizeof(alevel_text)); +#endif /* As suggested by KJ4ERJ, if we are receiving from */ /* WIDEn-0, it is quite likely (but not guaranteed), that */ @@ -905,6 +1285,7 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev dw_printf ("%s audio level = %s %s %s\n", heard, alevel_text, display_retries, spectrum); } + } } /* Version 1.2: Cranking the input level way up produces 199. */ @@ -918,6 +1299,12 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev text_color_set(DW_COLOR_ERROR); dw_printf ("Audio input level is too high. Reduce so most stations are around 50.\n"); } +// FIXME: rather than checking for ichannel, how about checking medium==radio + else if (alevel.rec < 5 && chan != audio_config.igate_vchannel) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Audio input level is too low. Increase so most stations are around 50.\n"); + } // Display non-APRS packets in a different color. @@ -926,9 +1313,25 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev // -1 for APRStt DTMF decoder. + char ts[100]; // optional time stamp + + if (strlen(audio_config.timestamp_format) > 0) { + char tstmp[100]; + timestamp_user_format (tstmp, sizeof(tstmp), audio_config.timestamp_format); + strlcpy (ts, " ", sizeof(ts)); // space after channel. + strlcat (ts, tstmp, sizeof(ts)); + } + else { + strlcpy (ts, "", sizeof(ts)); + } + if (subchan == -1) { text_color_set(DW_COLOR_REC); - dw_printf ("[%d.dtmf] ", chan); + dw_printf ("[%d.dtmf%s] ", chan, ts); + } + else if (subchan == -2) { + text_color_set(DW_COLOR_REC); + dw_printf ("[%d.is%s] ", chan, ts); } else { if (ax25_is_aprs(pp)) { @@ -939,16 +1342,16 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev } if (audio_config.achan[chan].num_subchan > 1 && audio_config.achan[chan].num_slicers == 1) { - dw_printf ("[%d.%d] ", chan, subchan); + dw_printf ("[%d.%d%s] ", chan, subchan, ts); } else if (audio_config.achan[chan].num_subchan == 1 && audio_config.achan[chan].num_slicers > 1) { - dw_printf ("[%d.%d] ", chan, slice); + dw_printf ("[%d.%d%s] ", chan, slice, ts); } else if (audio_config.achan[chan].num_subchan > 1 && audio_config.achan[chan].num_slicers > 1) { - dw_printf ("[%d.%d.%d] ", chan, subchan, slice); + dw_printf ("[%d.%d.%d%s] ", chan, subchan, slice, ts); } else { - dw_printf ("[%d] ", chan); + dw_printf ("[%d%s] ", chan, ts); } } @@ -972,7 +1375,7 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev dw_printf ("(%s)", desc); if (ftype == frame_type_U_XID) { struct xid_param_s param; - char info2text[100]; + char info2text[150]; xid_parse (pinfo, info_len, ¶m, info2text, sizeof(info2text)); dw_printf (" %s\n", info2text); @@ -1030,6 +1433,8 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev * * Suppress printed decoding if "-q d" option used. */ + char ais_obj_packet[300]; + strcpy (ais_obj_packet, ""); if (ax25_is_aprs(pp)) { @@ -1038,7 +1443,7 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev // we still want to decode it for logging and other processing. // Just be quiet about errors if "-qd" is set. - decode_aprs (&A, pp, q_d_opt); + decode_aprs (&A, pp, q_d_opt, NULL); if ( ! q_d_opt ) { @@ -1064,6 +1469,37 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev mheard_save_rf (chan, &A, pp, alevel, retries); +// For AIS, we have an option to convert the NMEA format, in User Defined data, +// into an APRS "Object Report" and send that to the clients as well. + +// FIXME: partial implementation. + + static const char user_def_da[4] = { '{', USER_DEF_USER_ID, USER_DEF_TYPE_AIS, '\0' }; + + if (strncmp((char*)pinfo, user_def_da, 3) == 0) { + + waypoint_send_ais((char*)pinfo + 3); + + if (A_opt_ais_to_obj && A.g_lat != G_UNKNOWN && A.g_lon != G_UNKNOWN) { + + char ais_obj_info[256]; + (void)encode_object (A.g_name, 0, time(NULL), + A.g_lat, A.g_lon, 0, // no ambiguity + A.g_symbol_table, A.g_symbol_code, + 0, 0, 0, "", // power, height, gain, direction. + // Unknown not handled properly. + // Should encode_object take floating point here? + (int)(A.g_course+0.5), (int)(DW_MPH_TO_KNOTS(A.g_speed_mph)+0.5), + 0, 0, 0, A.g_comment, // freq, tone, offset + ais_obj_info, sizeof(ais_obj_info)); + + snprintf (ais_obj_packet, sizeof(ais_obj_packet), "%s>%s%1d%1d:%s", A.g_src, APP_TOCALL, MAJOR_VERSION, MINOR_VERSION, ais_obj_info); + + dw_printf ("[%d.AIS] %s\n", chan, ais_obj_packet); + + // This will be sent to client apps after the User Defined Data representation. + } + } // Convert to NMEA waypoint sentence if we have a location. @@ -1078,34 +1514,71 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev /* Send to another application if connected. */ // TODO: Put a wrapper around this so we only call one function to send by all methods. +// We see the same sequence in tt_user.c. int flen; unsigned char fbuf[AX25_MAX_PACKET_LEN]; flen = ax25_pack(pp, fbuf); - server_send_rec_packet (chan, pp, fbuf, flen); - kissnet_send_rec_packet (chan, fbuf, flen); - kiss_send_rec_packet (chan, fbuf, flen); + server_send_rec_packet (chan, pp, fbuf, flen); // AGW net protocol + kissnet_send_rec_packet (chan, KISS_CMD_DATA_FRAME, fbuf, flen, NULL, -1); // KISS TCP + kissserial_send_rec_packet (chan, KISS_CMD_DATA_FRAME, fbuf, flen, NULL, -1); // KISS serial port + kisspt_send_rec_packet (chan, KISS_CMD_DATA_FRAME, fbuf, flen, NULL, -1); // KISS pseudo terminal + + if (A_opt_ais_to_obj && strlen(ais_obj_packet) != 0) { + packet_t ao_pp = ax25_from_text (ais_obj_packet, 1); + if (ao_pp != NULL) { + unsigned char ao_fbuf[AX25_MAX_PACKET_LEN]; + int ao_flen = ax25_pack(ao_pp, ao_fbuf); + + server_send_rec_packet (chan, ao_pp, ao_fbuf, ao_flen); + kissnet_send_rec_packet (chan, KISS_CMD_DATA_FRAME, ao_fbuf, ao_flen, NULL, -1); + kissserial_send_rec_packet (chan, KISS_CMD_DATA_FRAME, ao_fbuf, ao_flen, NULL, -1); + kisspt_send_rec_packet (chan, KISS_CMD_DATA_FRAME, ao_fbuf, ao_flen, NULL, -1); + ax25_delete (ao_pp); + } + } + +/* + * If it is from the ICHANNEL, we are done. + * Don't digipeat. Don't IGate. + * Don't do anything with it after printing and sending to client apps. + */ + + if (chan == audio_config.igate_vchannel) { + return; + } /* - * If it came from DTMF decoder, send it to APRStt gateway. + * If it came from DTMF decoder (subchan == -1), send it to APRStt gateway. * Otherwise, it is a candidate for IGate and digipeater. * - * TODO: It might be useful to have some way to simulate touch tone - * sequences with BEACON sendto=R... for testing. + * It is also useful to have some way to simulate touch tone + * sequences with BEACON sendto=R0 for testing. */ - if (subchan == -1) { + + if (subchan == -1) { // from DTMF decoder if (tt_config.gateway_enabled && info_len >= 2) { aprs_tt_sequence (chan, (char*)(pinfo+1)); } } + else if (*pinfo == 't' && info_len >= 2 && tt_config.gateway_enabled) { + // For testing. + // Would be nice to verify it was generated locally, + // not received over the air. + aprs_tt_sequence (chan, (char*)(pinfo+1)); + } else { -/* Send to Internet server if option is enabled. */ -/* Consider only those with correct CRC. */ - - if (ax25_is_aprs(pp) && retries == RETRY_NONE) { +/* + * Send to the IGate processing. + * Use only those with correct CRC; We don't want to spread corrupted data! + * Our earlier "fix bits" hack could allow corrupted information to get thru. + * However, if it used FEC mode (FX.25. IL2P), we have much higher level of + * confidence that it is correct. + */ + if (ax25_is_aprs(pp) && ( retries == RETRY_NONE || fec_type == fec_type_fx25 || fec_type == fec_type_il2p) ) { igate_send_rec_packet (chan, pp); } @@ -1120,21 +1593,23 @@ void app_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alev /* - * APRS digipeater. + * Send to APRS digipeater. * Use only those with correct CRC; We don't want to spread corrupted data! + * Our earlier "fix bits" hack could allow corrupted information to get thru. + * However, if it used FEC mode (FX.25. IL2P), we have much higher level of + * confidence that it is correct. */ - - if (ax25_is_aprs(pp) && retries == RETRY_NONE) { + if (ax25_is_aprs(pp) && ( retries == RETRY_NONE || fec_type == fec_type_fx25 || fec_type == fec_type_il2p) ) { digipeater (chan, pp); } /* * Connected mode digipeater. - * Use only those with correct CRC. + * Use only those with correct CRC (or using FEC.) */ - if (retries == RETRY_NONE) { + if (retries == RETRY_NONE || fec_type == fec_type_fx25 || fec_type == fec_type_il2p) { cdigipeater (chan, pp); } @@ -1201,10 +1676,20 @@ static void usage (char **argv) dw_printf (" 2400 bps uses QPSK based on V.26 standard.\n"); dw_printf (" 4800 bps uses 8PSK based on V.27 standard.\n"); dw_printf (" 9600 bps and up uses K9NG/G3RUH standard.\n"); + dw_printf (" AIS for ship Automatic Identification System.\n"); + dw_printf (" EAS for Emergency Alert System (EAS) Specific Area Message Encoding (SAME).\n"); + dw_printf (" -g Force G3RUH modem regardless of speed.\n"); + dw_printf (" -j 2400 bps QPSK compatible with direwolf <= 1.5.\n"); + dw_printf (" -J 2400 bps QPSK compatible with MFJ-2400.\n"); + dw_printf (" -P xxx Modem Profiles.\n"); + dw_printf (" -A Convert AIS positions to APRS Object Reports.\n"); dw_printf (" -D n Divide audio sample rate by n for channel 0.\n"); + dw_printf (" -X n 1 to enable FX.25 transmit. 16, 32, 64 for specific number of check bytes.\n"); + dw_printf (" -I n Enable IL2P transmit. n=1 is recommended. 0 uses weaker FEC.\n"); + dw_printf (" -i n Enable IL2P transmit, inverted polarity. n=1 is recommended. 0 uses weaker FEC.\n"); dw_printf (" -d Debug options:\n"); dw_printf (" a a = AGWPE network protocol client.\n"); - dw_printf (" k k = KISS serial port client.\n"); + dw_printf (" k k = KISS serial port or pseudo terminal client.\n"); dw_printf (" n n = KISS network client.\n"); dw_printf (" u u = Display non-ASCII text in hexadecimal.\n"); dw_printf (" p p = dump Packets in hexadecimal.\n"); @@ -1218,31 +1703,47 @@ static void usage (char **argv) #if USE_HAMLIB dw_printf (" h h = hamlib increase verbose level.\n"); #endif + dw_printf (" x x = FX.25 increase verbose level.\n"); + dw_printf (" 2 2 = IL2P.\n"); + dw_printf (" d d = APRStt (DTMF to APRS object translation).\n"); dw_printf (" -q Quiet (suppress output) options:\n"); dw_printf (" h h = Heard line with the audio level.\n"); dw_printf (" d d = Decoding of APRS packets.\n"); - dw_printf (" -t n Text colors. 1=normal, 0=disabled.\n"); + dw_printf (" x x = Silence FX.25 information.\n"); + dw_printf (" -t n Text colors. 0=disabled. 1=default. 2,3,4,... alternatives.\n"); + dw_printf (" Use 9 to test compatibility with your terminal.\n"); dw_printf (" -a n Audio statistics interval in seconds. 0 to disable.\n"); #if __WIN32__ #else dw_printf (" -p Enable pseudo terminal for KISS protocol.\n"); #endif dw_printf (" -x Send Xmit level calibration tones.\n"); - dw_printf (" -U Print UTF-8 test string and exit.\n"); + dw_printf (" a a = Alternating mark/space tones.\n"); + dw_printf (" m m = Steady mark tone (e.g. 1200Hz).\n"); + dw_printf (" s s = Steady space tone (e.g. 2200Hz).\n"); + dw_printf (" p p = Silence (Set PTT only).\n"); + dw_printf (" chan Optionally add a number to specify radio channel.\n"); + dw_printf (" -u Print UTF-8 test string and exit.\n"); dw_printf (" -S Print symbol tables and exit.\n"); + dw_printf (" -T fmt Time stamp format for sent and received frames.\n"); + dw_printf (" -e ber Receive Bit Error Rate (BER), e.g. 1e-5\n"); dw_printf ("\n"); dw_printf ("After any options, there can be a single command line argument for the source of\n"); - dw_printf ("received audio. This can overrides the audio input specified in the configuration file.\n"); + dw_printf ("received audio. This can override the audio input specified in the configuration file.\n"); dw_printf ("\n"); #if __WIN32__ + dw_printf ("Documentation can be found in the 'doc' folder\n"); #else - dw_printf ("Complete documentation can be found in /usr/local/share/doc/direwolf.\n"); + // TODO: Could vary by platform and build options. + dw_printf ("Documentation can be found in /usr/local/share/doc/direwolf\n"); #endif + dw_printf ("or online at https://github.com/wb2osz/direwolf/tree/master/doc\n"); + dw_printf ("additional topics: https://github.com/wb2osz/direwolf-doc\n"); + text_color_set(DW_COLOR_INFO); exit (EXIT_FAILURE); } - /* end direwolf.c */ diff --git a/direwolf.h b/src/direwolf.h similarity index 56% rename from direwolf.h rename to src/direwolf.h index a647b600..69b09529 100644 --- a/direwolf.h +++ b/src/direwolf.h @@ -4,6 +4,11 @@ // TODO: include this file first before anything else in each .c file. +#ifdef NDEBUG +#undef NDEBUG // Because it would disable assert(). +#endif + + #ifndef DIREWOLF_H #define DIREWOLF_H 1 @@ -27,16 +32,13 @@ #define _WIN32_WINNT 0x0501 /* Minimum OS version is XP. */ #define WINVER 0x0501 /* Minimum OS version is XP. */ +#include #include #endif - - /* - * Previously, we could handle only a single audio device. - * This meant we could have only two radio channels. - * In version 1.2, we relax this restriction and allow more audio devices. + * Maximum number of audio devices. * Three is probably adequate for standard version. * Larger reasonable numbers should also be fine. * @@ -60,7 +62,15 @@ * and make sure they handle undefined channels correctly. */ -#define MAX_CHANS ((MAX_ADEVS) * 2) +#define MAX_RADIO_CHANS ((MAX_ADEVS) * 2) + +#define MAX_CHANS MAX_RADIO_CHANS // TODO: Replace all former with latter to avoid confusion with following. + +#define MAX_TOTAL_CHANS 16 // v1.7 allows additional virtual channels which are connected + // to something other than radio modems. + // Total maximum channels is based on the 4 bit KISS field. + // Someone with very unusual requirements could increase this and + // use only the AGW network protocol. /* * Maximum number of rigs. @@ -92,7 +102,7 @@ * Each one of these can have multiple slicers, at * different levels, to compensate for different * amplitudes of the AFSK tones. - * Intially used same number as subchannels but + * Initially used same number as subchannels but * we could probably trim this down a little * without impacting performance. */ @@ -108,15 +118,54 @@ #define SLEEP_MS(n) usleep((n)*1000) #endif - #if __WIN32__ + #define PTW32_STATIC_LIB //#include "pthreads/pthread.h" -#define gmtime_r( _clock, _result ) \ - ( *(_result) = *gmtime( (_clock) ), \ - (_result) ) + +// This enables definitions of localtime_r and gmtime_r in system time.h. +//#define _POSIX_THREAD_SAFE_FUNCTIONS 1 +#define _POSIX_C_SOURCE 1 + #else -#include + #include +#endif + + +#ifdef __APPLE__ + +// https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/messages/2072 + +// The original suggestion was to add this to only ptt.c. +// I thought it would make sense to put it here, so it will apply to all files, +// consistently, rather than only one file ptt.c. + +// The placement of this is critical. Putting it earlier was a problem. +// https://github.com/wb2osz/direwolf/issues/113 + +// It needs to be after the include pthread.h because +// pthread.h pulls in , which redefines __DARWIN_C_LEVEL back to ansi, +// which breaks things. +// Maybe it should just go in ptt.c as originally suggested. + +// #define __DARWIN_C_LEVEL __DARWIN_C_FULL + +// There is a more involved patch here: +// https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/messages/2458 + +#ifndef _DARWIN_C_SOURCE +#define _DARWIN_C_SOURCE +#endif + +// Defining _DARWIN_C_SOURCE ensures that the definition for the cfmakeraw function (or similar) +// are pulled in through the include file . + +#ifdef __DARWIN_C_LEVEL +#undef __DARWIN_C_LEVEL +#endif + +#define __DARWIN_C_LEVEL __DARWIN_C_FULL + #endif @@ -132,6 +181,7 @@ #define DW_METERS_TO_FEET(x) ((x) == G_UNKNOWN ? G_UNKNOWN : (x) * 3.2808399) #define DW_FEET_TO_METERS(x) ((x) == G_UNKNOWN ? G_UNKNOWN : (x) * 0.3048) #define DW_KM_TO_MILES(x) ((x) == G_UNKNOWN ? G_UNKNOWN : (x) * 0.621371192) +#define DW_MILES_TO_KM(x) ((x) == G_UNKNOWN ? G_UNKNOWN : (x) * 1.609344) #define DW_KNOTS_TO_MPH(x) ((x) == G_UNKNOWN ? G_UNKNOWN : (x) * 1.15077945) #define DW_KNOTS_TO_METERS_PER_SEC(x) ((x) == G_UNKNOWN ? G_UNKNOWN : (x) * 0.51444444444) @@ -213,52 +263,93 @@ typedef pthread_mutex_t dw_mutex_t; +// Formerly used write/read on Linux, for some forgotten reason, +// but always using send/recv makes more sense. +// Need option to prevent a SIGPIPE signal on Linux. (added for 1.5 beta 2) + +#if __WIN32__ || __APPLE__ +#define SOCK_SEND(s,data,size) send(s,data,size,0) +#else +#define SOCK_SEND(s,data,size) send(s,data,size, MSG_NOSIGNAL) +#endif +#define SOCK_RECV(s,data,size) recv(s,data,size,0) + + /* Platform differences for string functions. */ +// Windows is missing a few which are available on Unix/Linux platforms. +// We provide our own copies when building on Windows. #if __WIN32__ char *strsep(char **stringp, const char *delim); char *strtok_r(char *str, const char *delim, char **saveptr); #endif -// Don't recall why for everyone. +// Don't recall why I added this for everyone rather than only for Windows. +// Potential problem if some C library declares it a little differently. char *strcasestr(const char *S, const char *FIND); -#if defined(__OpenBSD__) || defined(__FreeBSD__) || defined(__APPLE__) - -// strlcpy and strlcat should be in string.h and the C library. - -#else // Use our own copy - - -// These prevent /usr/include/gps.h from providing its own definition. -#define HAVE_STRLCAT 1 +// cmake tries to determine whether strlcpy and strlcat are provided by the C runtime library. +// +// ../CMakeLists.txt:check_symbol_exists(strlcpy string.h HAVE_STRLCPY) +// +// It sets HAVE_STRLCPY and HAVE_STRLCAT if the corresponding functions are declared. +// Unfortunately this does not work right for glibc 2.38 which declares the functions +// like this: +// +// extern __typeof (strlcpy) __strlcpy; +// libc_hidden_proto (__strlcpy) +// extern __typeof (strlcat) __strlcat; +// libc_hidden_proto (__strlcat) +// +// Rather than the normal way found in earlier versions: +// +// extern char *strcpy (char *__restrict __dest, const char *__restrict __src) +// +// Perhaps a later version of cmake will recognize this form but the version I'm +// using does not. +// So, our work around is to assume these functions are available for glibc >= 2.38. +// +// In theory, cmake should be able to find the version of the C runtime library, +// but I could not get it to work. So we have the test here. We will still build +// own library with the strl... functions but this does not cause a problem +// because they have special debug names which will not cause a conflict. + +#ifdef __GLIBC__ +#if (__GLIBC__ > 2) || ((__GLIBC__ == 2) && (__GLIBC_MINOR__ >= 38)) +// These functions first added in 2.38. +//#warning "DEBUG - glibc >= 2.38" #define HAVE_STRLCPY 1 +#define HAVE_STRLCAT 1 +#else +//#warning "DEBUG - glibc < 2.38" +#endif +#endif +#define DEBUG_STRL 1 // Extra Debug version when using our own strlcpy, strlcat. + // Should be ignored if not supplying our own. -#define DEBUG_STRL 1 - +#ifndef HAVE_STRLCPY // Need to supply our own. #if DEBUG_STRL - #define strlcpy(dst,src,siz) strlcpy_debug(dst,src,siz,__FILE__,__func__,__LINE__) -#define strlcat(dst,src,siz) strlcat_debug(dst,src,siz,__FILE__,__func__,__LINE__) - size_t strlcpy_debug(char *__restrict__ dst, const char *__restrict__ src, size_t siz, const char *file, const char *func, int line); -size_t strlcat_debug(char *__restrict__ dst, const char *__restrict__ src, size_t siz, const char *file, const char *func, int line); - #else - #define strlcpy(dst,src,siz) strlcpy_debug(dst,src,siz) -#define strlcat(dst,src,siz) strlcat_debug(dst,src,siz) - size_t strlcpy_debug(char *__restrict__ dst, const char *__restrict__ src, size_t siz); -size_t strlcat_debug(char *__restrict__ dst, const char *__restrict__ src, size_t siz); - #endif /* DEBUG_STRL */ +#endif -#endif /* BSD or Apple */ +#ifndef HAVE_STRLCAT // Need to supply our own. +#if DEBUG_STRL +#define strlcat(dst,src,siz) strlcat_debug(dst,src,siz,__FILE__,__func__,__LINE__) +size_t strlcat_debug(char *__restrict__ dst, const char *__restrict__ src, size_t siz, const char *file, const char *func, int line); +#else +#define strlcat(dst,src,siz) strlcat_debug(dst,src,siz) +size_t strlcat_debug(char *__restrict__ dst, const char *__restrict__ src, size_t siz); +#endif /* DEBUG_STRL */ +#endif #endif /* ifndef DIREWOLF_H */ diff --git a/dlq.c b/src/dlq.c similarity index 85% rename from dlq.c rename to src/dlq.c index 6a4236f6..f56b8649 100644 --- a/dlq.c +++ b/src/dlq.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2014, 2015, 2016, 2018 John Langner, WB2OSZ // // 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 @@ -130,12 +130,18 @@ void dlq_init (void) #else int err; err = pthread_mutex_init (&wake_up_mutex, NULL); + if (err != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("dlq_init: pthread_mutex_init err=%d", err); + perror (""); + exit (EXIT_FAILURE); + } err = pthread_mutex_init (&dlq_mutex, NULL); if (err != 0) { text_color_set(DW_COLOR_ERROR); dw_printf ("dlq_init: pthread_mutex_init err=%d", err); perror (""); - exit (1); + exit (EXIT_FAILURE); } #endif @@ -209,7 +215,10 @@ void dlq_init (void) * display of audio level line. * Use -2 to indicate DTMF message.) * - * retries - Level of bit correction used. + * fec_type - Was it from FX.25 or IL2P? Need to know because + * meaning of retries is different. + * + * retries - Level of correction used. * * spectrum - Display of how well multiple decoders did. * @@ -219,7 +228,7 @@ void dlq_init (void) * *--------------------------------------------------------------------*/ -void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, char *spectrum) +void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, fec_type_t fec_type, retry_t retries, char *spectrum) { struct dlq_item_s *pnew; @@ -230,7 +239,7 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev dw_printf ("dlq_rec_frame (chan=%d, pp=%p, ...)\n", chan, pp); #endif - assert (chan >= 0 && chan < MAX_CHANS); + assert (chan >= 0 && chan < MAX_TOTAL_CHANS); // TOTAL to include virtual channels. if (pp == NULL) { text_color_set(DW_COLOR_ERROR); @@ -250,6 +259,11 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; if (s_new_count > s_delete_count + 50) { @@ -264,6 +278,7 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev pnew->subchan = subchan; pnew->pp = pp; pnew->alevel = alevel; + pnew->fec_type = fec_type; pnew->retries = retries; if (spectrum == NULL) strlcpy(pnew->spectrum, "", sizeof(pnew->spectrum)); @@ -488,6 +503,11 @@ void dlq_connect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; pnew->type = DLQ_CONNECT_REQUEST; @@ -541,6 +561,11 @@ void dlq_disconnect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; pnew->type = DLQ_DISCONNECT_REQUEST; @@ -553,7 +578,74 @@ void dlq_disconnect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int append_to_queue (pnew); -} /* end dlq_connect_request */ +} /* end dlq_disconnect_request */ + + + +/*------------------------------------------------------------------- + * + * Name: dlq_outstanding_frames_request + * + * Purpose: Client application wants to know number of outstanding information + * frames supplied, supplied by the client, that have not yet been + * delivered to the remote station. + * + * Inputs: addrs - Source (owncall), destination (peercall) + * + * num_addr - Number of addresses. Should be 2. + * If more they will be ignored. + * + * chan - Channel, 0 is first. + * + * client - Client application instance. We could have multiple + * applications, all on the same channel, connecting + * to different stations. We need to know which one + * should get the results. + * + * Outputs: Request is appended to queue for processing by + * the data link state machine. + * + * Description: The data link state machine will count up all information frames + * for the given source(mycall) / destination(remote) / channel link. + * A 'Y' response will be sent back to the client application. + * + *--------------------------------------------------------------------*/ + +void dlq_outstanding_frames_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client) +{ + struct dlq_item_s *pnew; +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("dlq_outstanding_frames_request (...)\n"); +#endif + + assert (chan >= 0 && chan < MAX_CHANS); + +/* Allocate a new queue item. */ + + pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } + s_new_count++; + + pnew->type = DLQ_OUTSTANDING_FRAMES_REQUEST; + pnew->chan = chan; + memcpy (pnew->addrs, addrs, sizeof(pnew->addrs)); + pnew->num_addr = num_addr; + pnew->client = client; + +/* Put it into queue. */ + + append_to_queue (pnew); + +} /* end dlq_outstanding_frames_request */ + + + + /*------------------------------------------------------------------- @@ -604,6 +696,11 @@ void dlq_xmit_data_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int n /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; pnew->type = DLQ_XMIT_DATA_REQUEST; @@ -631,8 +728,7 @@ void dlq_xmit_data_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int n * * Purpose: Register callsigns that we will recognize for incoming connection requests. * - * Inputs: addrs - Source (owncall), destination (peercall), - * and possibly digipeaters. + * Inputs: addr - Callsign to [un]register. * * chan - Channel, 0 is first. * @@ -652,7 +748,7 @@ void dlq_xmit_data_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int n *--------------------------------------------------------------------*/ -void dlq_register_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client) +void dlq_register_callsign (char *addr, int chan, int client) { struct dlq_item_s *pnew; @@ -667,11 +763,16 @@ void dlq_register_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client) /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; pnew->type = DLQ_REGISTER_CALLSIGN; pnew->chan = chan; - strlcpy (pnew->addrs[0], addr, AX25_MAX_ADDR_LEN); + strlcpy (pnew->addrs[0], addr, sizeof(pnew->addrs[0])); pnew->num_addr = 1; pnew->client = client; @@ -682,7 +783,7 @@ void dlq_register_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client) } /* end dlq_register_callsign */ -void dlq_unregister_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client) +void dlq_unregister_callsign (char *addr, int chan, int client) { struct dlq_item_s *pnew; @@ -697,11 +798,16 @@ void dlq_unregister_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; pnew->type = DLQ_UNREGISTER_CALLSIGN; pnew->chan = chan; - strlcpy (pnew->addrs[0], addr, AX25_MAX_ADDR_LEN); + strlcpy (pnew->addrs[0], addr, sizeof(pnew->addrs[0])); pnew->num_addr = 1; pnew->client = client; @@ -751,6 +857,11 @@ void dlq_channel_busy (int chan, int activity, int status) /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; pnew->type = DLQ_CHANNEL_BUSY; @@ -767,6 +878,58 @@ void dlq_channel_busy (int chan, int activity, int status) + + +/*------------------------------------------------------------------- + * + * Name: dlq_seize_confirm + * + * Purpose: Inform data link state machine that the transmitter is on. + * This is in response to lm_seize_request. + * + * Inputs: chan - Radio channel number. + * + * Outputs: Request is appended to queue for processing by + * the data link state machine. + * + * Description: When removed from the data link state machine queue, this + * becomes lm_seize_confirm. + * + *--------------------------------------------------------------------*/ + +void dlq_seize_confirm (int chan) +{ + struct dlq_item_s *pnew; + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("dlq_seize_confirm (chan=%d)\n", chan); +#endif + + +/* Allocate a new queue item. */ + + pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } + s_new_count++; + + pnew->type = DLQ_SEIZE_CONFIRM; + pnew->chan = chan; + +/* Put it into queue. */ + + append_to_queue (pnew); + + +} /* end dlq_seize_confirm */ + + + + /*------------------------------------------------------------------- * * Name: dlq_client_cleanup @@ -797,6 +960,11 @@ void dlq_client_cleanup (int client) /* Allocate a new queue item. */ pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1); + if (pnew == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } s_new_count++; // All we care about is the client number. @@ -1079,6 +1247,11 @@ cdata_t *cdata_new (int pid, char *data, int len) size = ( len + 127 ) & ~0x7f; cdata = malloc ( sizeof(cdata_t) + size ); + if (cdata == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } cdata->magic = TXDATA_MAGIC; cdata->next = NULL; diff --git a/dlq.h b/src/dlq.h similarity index 81% rename from dlq.h rename to src/dlq.h index 6874fb06..fdac1c0c 100644 --- a/dlq.h +++ b/src/dlq.h @@ -33,9 +33,12 @@ typedef struct cdata_s { + /* Types of things that can be in queue. */ -typedef enum dlq_type_e {DLQ_REC_FRAME, DLQ_CONNECT_REQUEST, DLQ_DISCONNECT_REQUEST, DLQ_XMIT_DATA_REQUEST, DLQ_REGISTER_CALLSIGN, DLQ_UNREGISTER_CALLSIGN, DLQ_CHANNEL_BUSY, DLQ_CLIENT_CLEANUP} dlq_type_t; +typedef enum dlq_type_e {DLQ_REC_FRAME, DLQ_CONNECT_REQUEST, DLQ_DISCONNECT_REQUEST, DLQ_XMIT_DATA_REQUEST, DLQ_REGISTER_CALLSIGN, DLQ_UNREGISTER_CALLSIGN, DLQ_OUTSTANDING_FRAMES_REQUEST, DLQ_CHANNEL_BUSY, DLQ_SEIZE_CONFIRM, DLQ_CLIENT_CLEANUP} dlq_type_t; + +typedef enum fec_type_e {fec_type_none=0, fec_type_fx25=1, fec_type_il2p=2} fec_type_t; /* A queue item. */ @@ -68,7 +71,11 @@ typedef struct dlq_item_s { alevel_t alevel; /* Audio level. */ + fec_type_t fec_type; // Type of FEC for received signal: none, FX.25, or IL2P. + retry_t retries; /* Effort expended to get a valid CRC. */ + /* Bits changed for regular AX.25. */ + /* Number of bytes fixed for FX.25. */ char spectrum[MAX_SUBCHANS*MAX_SLICERS+1]; /* "Spectrum" display for multi-decoders. */ @@ -102,20 +109,24 @@ void dlq_init (void); -void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, char *spectrum); +void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, fec_type_t fec_type, retry_t retries, char *spectrum); void dlq_connect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client, int pid); void dlq_disconnect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client); +void dlq_outstanding_frames_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client); + void dlq_xmit_data_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client, int pid, char *xdata_ptr, int xdata_len); -void dlq_register_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client); +void dlq_register_callsign (char *addr, int chan, int client); -void dlq_unregister_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client); +void dlq_unregister_callsign (char *addr, int chan, int client); void dlq_channel_busy (int chan, int activity, int status); +void dlq_seize_confirm (int chan); + void dlq_client_cleanup (int client); diff --git a/src/dns_sd_avahi.c b/src/dns_sd_avahi.c new file mode 100644 index 00000000..63ce0b66 --- /dev/null +++ b/src/dns_sd_avahi.c @@ -0,0 +1,260 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2020 Heikki Hannikainen, OH7LZB +// +// +// 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, see . + +/*------------------------------------------------------------------ + * + * Module: dns_sd_avahi.c + * + * Purpose: Announce the KISS over TCP service using DNS-SD via Avahi + * + * Description: + * + * Most people have typed in enough IP addresses and ports by now, and + * would rather just select an available TNC that is automatically + * discovered on the local network. Even more so on a mobile device + * such an Android or iOS phone or tablet. + * + * On Linux, the announcement can be made through Avahi, the mDNS + * framework commonly deployed on Linux systems. + * + * This is largely based on the publishing example of the Avahi library. + */ + +#ifdef USE_AVAHI_CLIENT + +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include "dns_sd_dw.h" +#include "dns_sd_common.h" +#include "textcolor.h" + +static AvahiEntryGroup *group = NULL; +static AvahiSimplePoll *simple_poll = NULL; +static AvahiClient *client = NULL; +static char *name = NULL; +static int kiss_port = 0; + +pthread_t avahi_thread; + +static void create_services(AvahiClient *c); + +#define PRINT_PREFIX "DNS-SD: Avahi: " + +static void entry_group_callback(AvahiEntryGroup *g, AvahiEntryGroupState state, AVAHI_GCC_UNUSED void *userdata) +{ + assert(g == group || group == NULL); + group = g; + + /* Called whenever the entry group state changes */ + switch (state) { + case AVAHI_ENTRY_GROUP_ESTABLISHED : + /* The entry group has been established successfully */ + text_color_set(DW_COLOR_INFO); + dw_printf(PRINT_PREFIX "Service '%s' successfully registered.\n", name); + break; + case AVAHI_ENTRY_GROUP_COLLISION: { + char *n; + /* A service name collision with a remote service + * happened. Let's pick a new name. */ + n = avahi_alternative_service_name(name); + avahi_free(name); + name = n; + text_color_set(DW_COLOR_INFO); + dw_printf(PRINT_PREFIX "Service name collision, renaming service to '%s'\n", name); + /* And recreate the services */ + create_services(avahi_entry_group_get_client(g)); + break; + } + case AVAHI_ENTRY_GROUP_FAILURE: + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "Entry group failure: %s\n", avahi_strerror(avahi_client_errno(avahi_entry_group_get_client(g)))); + /* Some kind of failure happened while we were registering our services */ + avahi_simple_poll_quit(simple_poll); + break; + case AVAHI_ENTRY_GROUP_UNCOMMITED: + case AVAHI_ENTRY_GROUP_REGISTERING: + ; + } +} + +static void create_services(AvahiClient *c) +{ + char *n; + int ret; + assert(c); + /* If this is the first time we're called, let's create a new + * entry group if necessary */ + if (!group) { + if (!(group = avahi_entry_group_new(c, entry_group_callback, NULL))) { + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "avahi_entry_group_new() failed: %s\n", avahi_strerror(avahi_client_errno(c))); + goto fail; + } + } else { + avahi_entry_group_reset(group); + } + + /* If the group is empty (either because it was just created, or + * because it was reset previously, add our entries. */ + if (avahi_entry_group_is_empty(group)) { + text_color_set(DW_COLOR_INFO); + dw_printf(PRINT_PREFIX "Announcing KISS TCP on port %d as '%s'\n", kiss_port, name); + + /* Announce with AVAHI_PROTO_INET instead of AVAHI_PROTO_UNSPEC, since Dire Wolf currently + * only listens on IPv4. + */ + + if ((ret = avahi_entry_group_add_service(group, AVAHI_IF_UNSPEC, AVAHI_PROTO_INET, 0, name, DNS_SD_SERVICE, NULL, NULL, kiss_port, NULL)) < 0) { + if (ret == AVAHI_ERR_COLLISION) + goto collision; + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "Failed to add _kiss-tnc._tcp service: %s\n", avahi_strerror(ret)); + goto fail; + } + + /* Tell the server to register the service */ + if ((ret = avahi_entry_group_commit(group)) < 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "Failed to commit entry group: %s\n", avahi_strerror(ret)); + goto fail; + } + } + return; + +collision: + /* A service name collision with a local service happened. Let's + * pick a new name */ + n = avahi_alternative_service_name(name); + avahi_free(name); + name = n; + text_color_set(DW_COLOR_INFO); + dw_printf(PRINT_PREFIX "Service name collision, renaming service to '%s'\n", name); + avahi_entry_group_reset(group); + create_services(c); + return; + +fail: + avahi_simple_poll_quit(simple_poll); +} + +static void client_callback(AvahiClient *c, AvahiClientState state, AVAHI_GCC_UNUSED void * userdata) +{ + assert(c); + /* Called whenever the client or server state changes */ + switch (state) { + case AVAHI_CLIENT_S_RUNNING: + /* The server has startup successfully and registered its host + * name on the network, so it's time to create our services */ + create_services(c); + break; + case AVAHI_CLIENT_FAILURE: + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "Client failure: %s\n", avahi_strerror(avahi_client_errno(c))); + avahi_simple_poll_quit(simple_poll); + break; + case AVAHI_CLIENT_S_COLLISION: + /* Let's drop our registered services. When the server is back + * in AVAHI_SERVER_RUNNING state we will register them + * again with the new host name. */ + case AVAHI_CLIENT_S_REGISTERING: + /* The server records are now being established. This + * might be caused by a host name change. We need to wait + * for our own records to register until the host name is + * properly esatblished. */ + if (group) + avahi_entry_group_reset(group); + break; + case AVAHI_CLIENT_CONNECTING: + ; + } +} + +static void cleanup(void) +{ + /* Cleanup things */ + if (client) + avahi_client_free(client); + + if (simple_poll) + avahi_simple_poll_free(simple_poll); + + avahi_free(name); +} + + +static void *avahi_mainloop(void *arg) +{ + /* Run the main loop */ + avahi_simple_poll_loop(simple_poll); + + cleanup(); + + return NULL; +} + +void dns_sd_announce (struct misc_config_s *mc) +{ + text_color_set(DW_COLOR_DEBUG); + //kiss_port = mc->kiss_port; // now an array. + kiss_port = mc->kiss_port[0]; // FIXME: Quick hack until I can handle multiple TCP ports properly. + + int error; + + /* Allocate main loop object */ + if (!(simple_poll = avahi_simple_poll_new())) { + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "Failed to create Avahi simple poll object.\n"); + goto fail; + } + + if (mc->dns_sd_name[0]) { + name = avahi_strdup(mc->dns_sd_name); + } else { + name = dns_sd_default_service_name(); + } + + /* Allocate a new client */ + client = avahi_client_new(avahi_simple_poll_get(simple_poll), 0, client_callback, NULL, &error); + + /* Check whether creating the client object succeeded */ + if (!client) { + text_color_set(DW_COLOR_ERROR); + dw_printf(PRINT_PREFIX "Failed to create Avahi client: %s\n", avahi_strerror(error)); + goto fail; + } + + pthread_create(&avahi_thread, NULL, &avahi_mainloop, NULL); + + return; + +fail: + cleanup(); +} + +#endif // USE_AVAHI_CLIENT + diff --git a/src/dns_sd_common.c b/src/dns_sd_common.c new file mode 100644 index 00000000..65a1cbf7 --- /dev/null +++ b/src/dns_sd_common.c @@ -0,0 +1,65 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2020 Heikki Hannikainen, OH7LZB +// +// +// 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, see . + +/*------------------------------------------------------------------ + * + * Module: dns_sd_common.c + * + * Purpose: Announce the KISS over TCP service using DNS-SD, common functions + * + * Description: + * + * Most people have typed in enough IP addresses and ports by now, and + * would rather just select an available TNC that is automatically + * discovered on the local network. Even more so on a mobile device + * such an Android or iOS phone or tablet. + * + * This module contains common functions needed on Linux and MacOS. + */ + + +#include +#include +#include + +/* Get a default service name to publish. By default, + * "Dire Wolf on ", or just "Dire Wolf" if hostname cannot + * be obtained. + */ +char *dns_sd_default_service_name(void) +{ + char hostname[51]; + char sname[64]; + + int i = gethostname(hostname, sizeof(hostname)); + if (i == 0) { + hostname[sizeof(hostname)-1] = 0; + + // on some systems, an FQDN is returned; remove domain part + char *dot = strchr(hostname, '.'); + if (dot) + *dot = 0; + + snprintf(sname, sizeof(sname), "Dire Wolf on %s", hostname); + return strdup(sname); + } + + return strdup("Dire Wolf"); +} + diff --git a/src/dns_sd_common.h b/src/dns_sd_common.h new file mode 100644 index 00000000..f104bf83 --- /dev/null +++ b/src/dns_sd_common.h @@ -0,0 +1,7 @@ + +#if (USE_AVAHI_CLIENT|USE_MACOS_DNSSD) + +char *dns_sd_default_service_name(void); + +#endif + diff --git a/src/dns_sd_dw.h b/src/dns_sd_dw.h new file mode 100644 index 00000000..79f4b868 --- /dev/null +++ b/src/dns_sd_dw.h @@ -0,0 +1,10 @@ + +#if (USE_AVAHI_CLIENT|USE_MACOS_DNSSD) + +#include "config.h" + +#define DNS_SD_SERVICE "_kiss-tnc._tcp" + +void dns_sd_announce (struct misc_config_s *mc); + +#endif // USE_AVAHI_CLIENT diff --git a/src/dns_sd_macos.c b/src/dns_sd_macos.c new file mode 100644 index 00000000..d733a412 --- /dev/null +++ b/src/dns_sd_macos.c @@ -0,0 +1,89 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2020 Heikki Hannikainen, OH7LZB +// +// +// 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, see . + +/*------------------------------------------------------------------ + * + * Module: dns_sd_macos.c + * + * Purpose: Announce the KISS over TCP service using MacOS dns-sd + * + * Description: + * + * Most people have typed in enough IP addresses and ports by now, and + * would rather just select an available TNC that is automatically + * discovered on the local network. Even more so on a mobile device + * such an Android or iOS phone or tablet. + * + * On MacOs, the announcement can be made through dns-sd. + */ + +#ifdef USE_MACOS_DNSSD + +#include +#include +#include + +#include "dns_sd_dw.h" +#include "dns_sd_common.h" +#include "textcolor.h" + +static char *name = NULL; + +static void registerServiceCallBack(DNSServiceRef sdRef, DNSServiceFlags flags, DNSServiceErrorType errorCode, + const char* name, const char* regType, const char* domain, void* context) +{ + if (errorCode == kDNSServiceErr_NoError) { + text_color_set(DW_COLOR_INFO); + dw_printf("DNS-SD: Successfully registered '%s'\n", name); + } else { + text_color_set(DW_COLOR_ERROR); + dw_printf("DNS-SD: Failed to register '%s': %d\n", name, errorCode); + } +} + +void dns_sd_announce (struct misc_config_s *mc) +{ + //int kiss_port = mc->kiss_port; // now an array. + int kiss_port = mc->kiss_port[0]; // FIXME: Quick hack until I can handle multiple TCP ports properly. + + if (mc->dns_sd_name[0]) { + name = strdup(mc->dns_sd_name); + } else { + name = dns_sd_default_service_name(); + } + + uint16_t port_nw = htons(kiss_port); + + DNSServiceRef registerRef; + DNSServiceErrorType err = DNSServiceRegister( + ®isterRef, 0, 0, name, DNS_SD_SERVICE, NULL, NULL, + port_nw, 0, NULL, registerServiceCallBack, NULL); + + if (err == kDNSServiceErr_NoError) { + text_color_set(DW_COLOR_INFO); + dw_printf("DNS-SD: Announcing KISS TCP on port %d as '%s'\n", kiss_port, name); + } else { + text_color_set(DW_COLOR_ERROR); + dw_printf("DNS-SD: Failed to announce '%s': %d\n", name, err); + } +} + +#endif // USE_MACOS_DNSSD + + diff --git a/dsp.c b/src/dsp.c similarity index 59% rename from dsp.c rename to src/dsp.c index fa163053..4a5f4a88 100644 --- a/dsp.c +++ b/src/dsp.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2015 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2015, 2019 John Langner, WB2OSZ // // 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 @@ -43,7 +43,6 @@ #include "dsp.h" -//#include "fsk_demod_agc.h" /* for M_FILTER_SIZE, etc. */ #define MIN(a,b) ((a)<(b)?(a):(b)) #define MAX(a,b) ((a)>(b)?(a):(b)) @@ -51,7 +50,7 @@ // Don't remove this. It serves as a reminder that an experiment is underway. -#if defined(TUNE_MS_FILTER_SIZE) || defined(TUNE_AGC_FAST) || defined(TUNE_LPF_BAUD) || defined(TUNE_PLL_LOCKED) || defined(TUNE_PROFILE) +#if defined(TUNE_MS_FILTER_SIZE) || defined(TUNE_MS2_FILTER_SIZE) || defined(TUNE_AGC_FAST) || defined(TUNE_LPF_BAUD) || defined(TUNE_PLL_LOCKED) || defined(TUNE_PROFILE) #define DEBUG1 1 // Don't remove this. #endif @@ -118,9 +117,12 @@ float window (bp_window_t type, int size, int j) * Inputs: fc - Cutoff frequency as fraction of sampling frequency. * filter_size - Number of filter taps. * wtype - Window type, BP_WINDOW_HAMMING, etc. + * lp_delay_fract - Fudge factor for the delay value. * * Outputs: lp_filter * + * Returns: Signal delay thru the filter in number of audio samples. + * *----------------------------------------------------------------*/ @@ -171,14 +173,22 @@ void gen_lowpass (float fc, float *lp_filter, int filter_size, bp_window_t wtype for (j=0; j -0.001 && t < 0.001) { + sinc = 1; + } + else { + sinc = sinf(M_PI * t) / (M_PI * t); + } + + if (fabsf(a * t) > 0.499 && fabsf(a * t) < 0.501) { + window = M_PI / 4; + } + else { + window = cos(M_PI * a * t) / ( 1 - powf(2 * a * t, 2)); + // This made nicer looking waveforms for generating signal. + //window = cos(M_PI * a * t); + // Do we want to let it go negative? + // I think this would happen when a > 0.5 / (filter width in symbol times) + if (window < 0) { + //printf ("'a' is too large for range of 't'.\n"); + //window = 0; + } + } + + result = sinc * window; + +#if DEBUGRRC + // t should vary from - to + half of filter size in symbols. + // Result should be 1 at t=0 and 0 at all other integer values of t. + + printf ("%.3f, %.3f, %.3f, %.3f\n", t, sinc, window, result); +#endif + return (result); +} + +// The Root Raised Cosine (RRC) low pass filter is suppposed to minimize Intersymbol Interference (ISI). + +void gen_rrc_lowpass (float *pfilter, int filter_taps, float rolloff, float samples_per_symbol) +{ + int k; + float t; + + for (k = 0; k < filter_taps; k++) { + t = (k - ((filter_taps - 1.0) / 2.0)) / samples_per_symbol; + pfilter[k] = rrc (t, rolloff); + } + + // Scale it for unity gain. + + t = 0; + for (k = 0; k < filter_taps; k++) { + t += pfilter[k]; + } + for (k = 0; k < filter_taps; k++) { + pfilter[k] = pfilter[k] / t; + } } -/* end dsp.c */ \ No newline at end of file +/* end dsp.c */ diff --git a/src/dsp.h b/src/dsp.h new file mode 100644 index 00000000..e0dbd248 --- /dev/null +++ b/src/dsp.h @@ -0,0 +1,17 @@ + +/* dsp.h */ + +// TODO: put prefixes on these names. + +float window (bp_window_t type, int size, int j); + +void gen_lowpass (float fc, float *lp_filter, int filter_size, bp_window_t wtype); + +void gen_bandpass (float f1, float f2, float *bp_filter, int filter_size, bp_window_t wtype); + +void gen_ms (int fc, int samples_per_sec, float *sin_table, float *cos_table, int filter_size, int wtype); + + +__attribute__((const)) float rrc (float t, float a); + +void gen_rrc_lowpass (float *pfilter, int filter_taps, float rolloff, float samples_per_symbol); diff --git a/src/dtime_now.c b/src/dtime_now.c new file mode 100644 index 00000000..e5b40c6c --- /dev/null +++ b/src/dtime_now.c @@ -0,0 +1,304 @@ + +#include "direwolf.h" + +#include + +#include "textcolor.h" +#include "dtime_now.h" + + +/* Current time in seconds but more resolution than time(). */ + +/* We don't care what date a 0 value represents because we */ +/* only use this to calculate elapsed real time. */ + + + +#include + +#ifdef __APPLE__ +#include +#endif + +#include // needed for Mac. + + +/*------------------------------------------------------------------ + * + * Name: dtime_realtime + * + * Purpose: Return current wall clock time as double precision. + * + * Input: none + * + * Returns: Unix time, as double precision, so we can get resolution + * finer than one second. + * + * Description: Normal unix time is in seconds since 1/1/1970 00:00:00 UTC. + * Sometimes we want resolution finer than a second. + * Rather than having a separate variable for the fractional + * part of a second, and having extra calculations everywhere, + * simply use double precision floating point to make usage + * easier. + * + * NOTE: This is not a good way to calculate elapsed time because + * it can jump forward or backware via NTP or other manual setting. + * + * Use the monotonic version for measuring elapsed time. + * + * History: Originally I called this dtime_now. We ran into issues where + * we really cared about elapsed time, rather than wall clock time. + * The wall clock time could be wrong at start up time if there + * is no realtime clock or Internet access. It can then jump + * when GPS time or Internet access becomes available. + * All instances of dtime_now should be replaced by dtime_realtime + * if we want wall clock time, or dtime_monotonic if it is to be + * used for measuring elapsed time, such as between becons. + * + *---------------------------------------------------------------*/ + +double dtime_realtime (void) +{ + double result; + +#if __WIN32__ + /* 64 bit integer is number of 100 nanosecond intervals from Jan 1, 1601. */ + + FILETIME ft; + + GetSystemTimeAsFileTime (&ft); + + result = ((( (double)ft.dwHighDateTime * (256. * 256. * 256. * 256.) + + (double)ft.dwLowDateTime ) / 10000000.) - 11644473600.); +#else + /* tv_sec is seconds from Jan 1, 1970. */ + + struct timespec ts; + +#ifdef __APPLE__ + +// Why didn't I use clock_gettime? +// Not available before Max OSX 10.12? https://github.com/gambit/gambit/issues/293 + + struct timeval tp; + gettimeofday(&tp, NULL); + ts.tv_nsec = tp.tv_usec * 1000; + ts.tv_sec = tp.tv_sec; +#else + clock_gettime (CLOCK_REALTIME, &ts); +#endif + + result = ((double)(ts.tv_sec) + (double)(ts.tv_nsec) * 0.000000001); + +#endif + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("dtime_realtime() returns %.3f\n", result ); +#endif + + return (result); +} + + +/*------------------------------------------------------------------ + * + * Name: dtime_monotonic + * + * Purpose: Return montonically increasing time, which is not influenced + * by the wall clock changing. e.g. leap seconds, NTP adjustments. + * + * Input: none + * + * Returns: Time as double precision, so we can get resolution + * finer than one second. + * + * Description: Use this when calculating elapsed time. + * + *---------------------------------------------------------------*/ + +double dtime_monotonic (void) +{ + double result; + +#if __WIN32__ + +// FIXME: +// This is still returning wall clock time. +// https://docs.microsoft.com/en-us/windows/win32/api/sysinfoapi/nf-sysinfoapi-gettickcount64 +// GetTickCount64 would be ideal but it requires Vista or Server 2008. +// As far as I know, the current version of direwolf still works on XP. +// +// As a work-around, GetTickCount could be used if we add extra code to deal +// with the wrap around after about 49.7 days. +// Resolution is only about 10 or 16 milliseconds. Is that good enough? + + /* 64 bit integer is number of 100 nanosecond intervals from Jan 1, 1601. */ + + FILETIME ft; + + GetSystemTimeAsFileTime (&ft); + + result = ((( (double)ft.dwHighDateTime * (256. * 256. * 256. * 256.) + + (double)ft.dwLowDateTime ) / 10000000.) - 11644473600.); +#else + /* tv_sec is seconds from Jan 1, 1970. */ + + struct timespec ts; + +#ifdef __APPLE__ + +// FIXME: Does MacOS have a monotonically increasing time? +// https://stackoverflow.com/questions/41509505/clock-gettime-on-macos + + struct timeval tp; + gettimeofday(&tp, NULL); + ts.tv_nsec = tp.tv_usec * 1000; + ts.tv_sec = tp.tv_sec; +#else + +// This is the only case handled properly. +// Probably the only one that matters. +// It is common to have a Raspberry Pi, without Internet, +// starting up direwolf before GPS/NTP adjusts the time. + + clock_gettime (CLOCK_MONOTONIC, &ts); +#endif + + result = ((double)(ts.tv_sec) + (double)(ts.tv_nsec) * 0.000000001); + +#endif + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("dtime_now() returns %.3f\n", result ); +#endif + + return (result); +} + + + +/*------------------------------------------------------------------ + * + * Name: timestamp_now + * + * Purpose: Convert local time to one of these formats for debug output. + * + * HH:MM:SS + * HH:MM:SS.mmm + * + * Input: result_size - Size of result location. + * Should be at least 9 or 13. + * + * show_ms - True to display milliseconds. + * + * Output: result - Result is placed here. + * + *---------------------------------------------------------------*/ + +void timestamp_now (char *result, int result_size, int show_ms) +{ + double now = dtime_realtime(); + time_t t = (int)now; + struct tm tm; + + localtime_r (&t, &tm); + strftime (result, result_size, "%H:%M:%S", &tm); + + if (show_ms) { + int ms = (now - (int)t) * 1000; + char strms[16]; + + if (ms == 1000) ms = 999; + sprintf (strms, ".%03d", ms); + strlcat (result, strms, result_size); + } + +} /* end timestamp_now */ + + + +/*------------------------------------------------------------------ + * + * Name: timestamp_user_format + * + * Purpose: Convert local time user-specified format. e.g. + * + * HH:MM:SS + * mm/dd/YYYY HH:MM:SS + * dd/mm/YYYY HH:MM:SS + * + * Input: result_size - Size of result location. + * + * user_format - See strftime documentation. + * + * https://linux.die.net/man/3/strftime + * https://msdn.microsoft.com/en-us/library/aa272978(v=vs.60).aspx + * + * Note that Windows does not support all of the Linux formats. + * For example, Linux has %T which is equivalent to %H:%M:%S + * + * Output: result - Result is placed here. + * + *---------------------------------------------------------------*/ + +void timestamp_user_format (char *result, int result_size, char *user_format) +{ + double now = dtime_realtime(); + time_t t = (int)now; + struct tm tm; + + localtime_r (&t, &tm); + strftime (result, result_size, user_format, &tm); + +} /* end timestamp_user_format */ + + +/*------------------------------------------------------------------ + * + * Name: timestamp_filename + * + * Purpose: Generate unique file name based on the current time. + * The format will be: + * + * YYYYMMDD-HHMMSS-mmm + * + * Input: result_size - Size of result location. + * Should be at least 20. + * + * Output: result - Result is placed here. + * + * Description: This is for the kissutil "-r" option which places + * each received frame in a new file. It is possible to + * have two packets arrive in less than a second so we + * need more than one second resolution. + * + * What if someone wants UTC, rather than local time? + * You can simply set an environment variable like this: + * + * TZ=UTC direwolf + * + * so it's probably not worth the effort to add another + * option. + * + *---------------------------------------------------------------*/ + +void timestamp_filename (char *result, int result_size) +{ + double now = dtime_realtime(); + time_t t = (int)now; + struct tm tm; + + localtime_r (&t, &tm); + strftime (result, result_size, "%Y%m%d-%H%M%S", &tm); + + int ms = (now - (int)t) * 1000; + char strms[16]; + + if (ms == 1000) ms = 999; + sprintf (strms, "-%03d", ms); + strlcat (result, strms, result_size); + +} /* end timestamp_filename */ + diff --git a/src/dtime_now.h b/src/dtime_now.h new file mode 100644 index 00000000..411534bf --- /dev/null +++ b/src/dtime_now.h @@ -0,0 +1,18 @@ + + +extern double dtime_realtime (void); + +extern double dtime_monotonic (void); + + +void timestamp_now (char *result, int result_size, int show_ms); + +void timestamp_user_format (char *result, int result_size, char *user_format); + +void timestamp_filename (char *result, int result_size); + + +// FIXME: remove temp workaround. +// Needs many scattered updates. + +#define dtime_now dtime_realtime diff --git a/dtmf.c b/src/dtmf.c similarity index 99% rename from dtmf.c rename to src/dtmf.c index 031946e4..953b0f70 100644 --- a/dtmf.c +++ b/src/dtmf.c @@ -245,7 +245,7 @@ char dtmf_sample (int c, float input) * others in the same group multiplied by some factor. * * For perfect synthetic signals this needs to be in - * the range of about 1.33 (very senstive) to 2.15 (very fussy). + * the range of about 1.33 (very sensitive) to 2.15 (very fussy). * * Too low will cause false triggers on random noise. * Too high will won't decode less than perfect signals. @@ -529,7 +529,7 @@ int main () memset (&my_audio_config, 0, sizeof(my_audio_config)); my_audio_config.adev[ACHAN2ADEV(c)].defined = 1; my_audio_config.adev[ACHAN2ADEV(c)].samples_per_sec = 44100; - my_audio_config.achan[c].valid = 1; + my_audio_config.chan_medium[c] = MEDIUM_RADIO; my_audio_config.achan[c].dtmf_decode = DTMF_DECODE_ON; dtmf_init(&my_audio_config, 50); diff --git a/dtmf.h b/src/dtmf.h similarity index 100% rename from dtmf.h rename to src/dtmf.h diff --git a/dwgps.c b/src/dwgps.c similarity index 99% rename from dwgps.c rename to src/dwgps.c index 83e92444..ccf24b0b 100644 --- a/dwgps.c +++ b/src/dwgps.c @@ -90,7 +90,7 @@ static dw_mutex_t s_gps_mutex; * * Name: dwgps_init * - * Purpose: Intialize the GPS interface. + * Purpose: Initialize the GPS interface. * * Inputs: pconfig Configuration settings. This might include * serial port name for direct connect and host diff --git a/dwgps.h b/src/dwgps.h similarity index 100% rename from dwgps.h rename to src/dwgps.h diff --git a/dwgpsd.c b/src/dwgpsd.c similarity index 59% rename from dwgpsd.c rename to src/dwgpsd.c index 57094978..1bc9d47a 100644 --- a/dwgpsd.c +++ b/src/dwgpsd.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2013, 2014, 2015 John Langner, WB2OSZ +// Copyright (C) 2013, 2014, 2015, 2020, 2022 John Langner, WB2OSZ // // 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 @@ -20,7 +20,7 @@ /*------------------------------------------------------------------ * - * Module: dwgps.c + * Module: dwgpsd.c * * Purpose: Interface to location data, i.e. GPS receiver. * @@ -45,6 +45,7 @@ #include #include #include +#include #if __WIN32__ #error Not for Windows @@ -54,13 +55,38 @@ #include -// Debian bug report: direwolf (1.2-1) FTBFS with libgps22 as part of the gpsd transition (#803605): -// dwgps.c claims to only support GPSD_API_MAJOR_VERSION 5, but also builds successfully with -// GPSD_API_MAJOR_VERSION 6 provided by libgps22 when the attached patch is applied. -#if GPSD_API_MAJOR_VERSION < 5 || GPSD_API_MAJOR_VERSION > 6 -#error libgps API version might be incompatible. + + +// An API incompatibility was introduced with API version 7. +// and again with 9. +// and again with 10. +// We deal with it by using a bunch of conditional code such as: +// #if GPSD_API_MAJOR_VERSION >= 9 + + +// release lib version API Raspberry Pi OS Testing status +// 3.22 28 11 bullseye OK. +// 3.23 29 12 OK. +// 3.25 30 14 OK, Jan. 2023 + + +// Previously the compilation would fail if the API version was later +// than the last one tested. Now it is just a warning because it changes so +// often but more recent versions have not broken backward compatibility. + +#define MAX_TESTED_VERSION 14 + +#if (GPSD_API_MAJOR_VERSION < 5) || (GPSD_API_MAJOR_VERSION > MAX_TESTED_VERSION) +#pragma message "Your version of gpsd might be incompatible with this application." +#pragma message "The libgps application program interface (API) often" +#pragma message "changes to be incompatible with earlier versions." +// I could not figure out how to do value substitution here. +#pragma message "You have libgpsd API version GPSD_API_MAJOR_VERSION." +#pragma message "The last that has been tested is MAX_TESTED_VERSION." +#pragma message "Even if this builds successfully, it might not run properly." #endif + /* * Information for interface to gpsd daemon. */ @@ -91,7 +117,7 @@ static void * read_gpsd_thread (void *arg); * * Name: dwgpsd_init * - * Purpose: Intialize the GPS interface. + * Purpose: Initialize the GPS interface. * * Inputs: pconfig Configuration settings. This includes * host name or address for network connection. @@ -112,7 +138,7 @@ static void * read_gpsd_thread (void *arg); * shared region via dwgps_put_data. * * The application calls dwgps_read to get the most - 8 recent information. + * recent information. * *--------------------------------------------------------------------*/ @@ -122,7 +148,7 @@ static void * read_gpsd_thread (void *arg); * Originally, I wanted to use the shared memory interface to gpsd * because it is simpler and more efficient. Just access it when we * actually need the data and we don't have a lot of extra unnecessary - * busy work going on. + * busy work going on constantly polling it when we don't need the information. * * The current version of gpsd, supplied with Raspian (Wheezy), is 3.6 from back in * May 2012, is missing support for the shared memory interface. @@ -138,13 +164,44 @@ static void * read_gpsd_thread (void *arg); * cd gpsd-3.11 * scons prefix=/usr libdir=lib/arm-linux-gnueabihf shm_export=True python=False * sudo scons udev-install - * + * * For now, we will use the socket interface. Maybe get back to this again someday. * * Update: January 2016. * * I'm told that the shared memory interface might work in Raspian, Jessie version. * Haven't tried it yet. + * + * June 2020: This is how to build the most recent. + * + * Based on https://www.satsignal.eu/raspberry-pi/UpdatingGPSD.html + * + * git clone https://gitlab.com/gpsd/gpsd.git gpsd-gitlab + * cd gpsd-gitlab + * scons --config=force + * scons + * sudo scons install + * + * The problem we have here is that the library is put in /usr/local/lib and direwolf + * can't find it there. Solution is to define environment variable: + * + * export LD_LIBRARY_PATH=/use/local/lib + * + * January 2023: Now using 64 bit Raspberry Pi OS, bullseye. + * See https://gitlab.com/gpsd/gpsd/-/blob/master/build.adoc + * Try to install in proper library place so we don't have to mess with LD_LIBRARY_PATH. + * + * (Remove any existing gpsd first so we are not mixing mismatched pieces.) + * + * sudo apt-get install libncurses5-dev + * sudo apt-get install gtk+-3.0 + * + * git clone https://gitlab.com/gpsd/gpsd.git gpsd-gitlab + * cd gpsd-gitlab + * scons prefix=/usr libdir=lib/aarch64-linux-gnu + * [ scons check ] + * sudo scons udev-install + * */ @@ -173,7 +230,7 @@ int dwgpsd_init (struct misc_config_s *pconfig, int debug) if (strlen(pconfig->gpsd_host) == 0) { - /* Nothing to do. Leave initial fix value of errror. */ + /* Nothing to do. Leave initial fix value of error. */ return (0); } @@ -190,7 +247,7 @@ int dwgpsd_init (struct misc_config_s *pconfig, int debug) gps_stream(&gpsdata, WATCH_ENABLE | WATCH_JSON, NULL); - e = pthread_create (&read_gps_tid, NULL, read_gpsd_thread, (void *)(long)arg); + e = pthread_create (&read_gps_tid, NULL, read_gpsd_thread, (void *)(ptrdiff_t)arg); if (e != 0) { text_color_set(DW_COLOR_ERROR); perror("Could not create GPS reader thread"); @@ -228,7 +285,7 @@ int dwgpsd_init (struct misc_config_s *pconfig, int debug) * *--------------------------------------------------------------------*/ -#define TIMEOUT 30 +#define TIMEOUT 15 #if ENABLE_GPSD @@ -251,13 +308,34 @@ static void * read_gpsd_thread (void *arg) while (1) { +// Example code found here: +// https://lists.nongnu.org/archive/html/gpsd-dev/2017-11/msg00001.html + if ( ! gps_waiting(&gpsdata, TIMEOUT * 1000000)) { text_color_set(DW_COLOR_ERROR); - dw_printf ("GPSD: Timeout waiting for GPS data.\n"); - /* Fall thru to read which should get error and bail out. */ + dw_printf ("------------------------------------------\n"); + dw_printf ("dwgpsd: Timeout waiting for GPS data.\n"); + dw_printf ("Is GPSD daemon running?\n"); + dw_printf ("Troubleshooting tip: Try running cgps or xgps.\n"); + dw_printf ("------------------------------------------\n"); + info.fix = DWFIX_ERROR; + SLEEP_MS(5000); + continue; } +// https://github.com/wb2osz/direwolf/issues/196 +// https://bugzilla.redhat.com/show_bug.cgi?id=1674812 + +// gps_read has two new parameters in API version 7. +// It looks like this could be used to obtain the JSON message from the daemon. +// Specify NULL, instead of message buffer space, if this is not desired. +// Why couldn't they add a new function instead of introducing incompatibility? + +#if GPSD_API_MAJOR_VERSION >= 7 + if (gps_read (&gpsdata, NULL, 0) == -1) { +#else if (gps_read (&gpsdata) == -1) { +#endif text_color_set(DW_COLOR_ERROR); dw_printf ("------------------------------------------\n"); @@ -274,17 +352,61 @@ static void * read_gpsd_thread (void *arg) break; // Jump out of loop and terminate thread. } +#if GPSD_API_MAJOR_VERSION >= 9 + +// The gps.h revision history says: +// * mark altitude in gps_fix_t as deprecated and undefined +// This seems really stupid to me. +// If it is deprecated and undefined then take it out. Someone trying to use +// it would get a compile error and know that something needs to be done. +// Instead we all just go merrily on our way using a field that is [allegedly] undefined. +// Why not simply add more variables with different definitions of altitude +// and keep the original variable working as it always did? +// If it is truly undefined, as the comment would have us believe, numerous +// people will WASTE VAST AMOUNTS OF TIME pondering why altitude is now broken in +// their applications. + +#define stupid_altitude altMSL +#else +#define stupid_altitude altitude +#endif + +#if GPSD_API_MAJOR_VERSION >= 10 + +// They did it again. Whimsical incompatibilities that cause +// pain and aggravation for everyone trying to use this library. +// +// error: ‘struct gps_data_t’ has no member named ‘status’ +// +// Yes, I can understand that it is a more logical place but it breaks +// all existing code that uses this. +// I'm really getting annoyed about wasting so much time on keeping up with all +// of these incompatibilities that are completely unnecessary. + +#define stupid_status fix.status +#else +#define stupid_status status +#endif + + + if (s_debug >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("gpsdata: status=%d, mode=%d, lat=%.6f, lon=%.6f, track=%.1f, speed=%.1f, alt=%.0f\n", + gpsdata.stupid_status, gpsdata.fix.mode, + gpsdata.fix.latitude, gpsdata.fix.longitude, + gpsdata.fix.track, gpsdata.fix.speed, gpsdata.fix.stupid_altitude); + } + + // Inform user about change in fix status. + switch (gpsdata.fix.mode) { default: case MODE_NOT_SEEN: - if (info.fix >= DWFIX_2D) { + case MODE_NO_FIX: + if (info.fix <= DWFIX_NOT_SEEN) { text_color_set(DW_COLOR_INFO); - dw_printf ("GPSD: Lost location fix.\n"); + dw_printf ("GPSD: No location fix.\n"); } - info.fix = DWFIX_NOT_SEEN; - break; - - case MODE_NO_FIX: if (info.fix >= DWFIX_2D) { text_color_set(DW_COLOR_INFO); dw_printf ("GPSD: Lost location fix.\n"); @@ -309,21 +431,27 @@ static void * read_gpsd_thread (void *arg) break; } - /* Data is available. */ - // TODO: what is gpsdata.status? +// Oct. 2020 - 'status' is always zero for latest version of libgps so we can't use that anymore. - if (gpsdata.status >= STATUS_FIX && gpsdata.fix.mode >= MODE_2D) { + if (/*gpsdata.stupid_status >= STATUS_FIX &&*/ gpsdata.fix.mode >= MODE_2D) { - info.dlat = isnan(gpsdata.fix.latitude) ? G_UNKNOWN : gpsdata.fix.latitude; - info.dlon = isnan(gpsdata.fix.longitude) ? G_UNKNOWN : gpsdata.fix.longitude; - info.track = isnan(gpsdata.fix.track) ? G_UNKNOWN : gpsdata.fix.track; - info.speed_knots = isnan(gpsdata.fix.speed) ? G_UNKNOWN : (MPS_TO_KNOTS * gpsdata.fix.speed); +#define GOOD(x) (isfinite(x) && ! isnan(x)) + info.dlat = GOOD(gpsdata.fix.latitude) ? gpsdata.fix.latitude : G_UNKNOWN; + info.dlon = GOOD(gpsdata.fix.longitude) ? gpsdata.fix.longitude : G_UNKNOWN; + // When stationary, track is NaN which is not finite. + info.track = GOOD(gpsdata.fix.track) ? gpsdata.fix.track : G_UNKNOWN; + info.speed_knots = GOOD(gpsdata.fix.speed) ? (MPS_TO_KNOTS * gpsdata.fix.speed) : G_UNKNOWN; if (gpsdata.fix.mode >= MODE_3D) { - info.altitude = isnan(gpsdata.fix.altitude) ? G_UNKNOWN : gpsdata.fix.altitude; + info.altitude = GOOD(gpsdata.fix.stupid_altitude) ? gpsdata.fix.stupid_altitude : G_UNKNOWN; } + // Otherwise keep last known altitude when we downgrade from 3D to 2D fix. + // Caller knows altitude is outdated if info.fix == DWFIX_2D. } + // Otherwise keep the last known location which is better than totally lost. + // Caller knows location is outdated if info.fix == DWFIX_NO_FIX. + info.timestamp = time(NULL); if (s_debug >= 2) { @@ -357,6 +485,7 @@ void dwgpsd_term (void) { #if ENABLE_GPSD + gps_stream (&gpsdata, WATCH_DISABLE, NULL); gps_close (&gpsdata); #endif diff --git a/dwgpsd.h b/src/dwgpsd.h similarity index 100% rename from dwgpsd.h rename to src/dwgpsd.h diff --git a/dwgpsnmea.c b/src/dwgpsnmea.c similarity index 88% rename from dwgpsnmea.c rename to src/dwgpsnmea.c index 6ce6963b..14cda77e 100644 --- a/dwgpsnmea.c +++ b/src/dwgpsnmea.c @@ -26,6 +26,21 @@ * * Description: This version is available for all operating systems. * + * + * TODO: GPS is no longer the only game in town. + * "GNSS" is often seen as a more general term to include + * other similar systems. Some receivers will receive + * multiple types at the same time and combine them + * for greater accuracy and reliability. + * + * We can now see NMEA sentences with other "Talker IDs." + * + * $GPxxx = GPS + * $GLxxx = GLONASS + * $GAxxx = Galileo + * $GBxxx = BeiDou + * $GNxxx = Any combination + * *---------------------------------------------------------------*/ @@ -39,6 +54,7 @@ #include #include #include +#include #include "textcolor.h" @@ -128,15 +144,13 @@ int dwgpsnmea_init (struct misc_config_s *pconfig, int debug) /* * Open serial port connection. - * 4800 baud is standard for GPS. - * Should add an option to allow changing someday. */ - s_gpsnmea_port_fd = serial_port_open (pconfig->gpsnmea_port, 4800); + s_gpsnmea_port_fd = serial_port_open (pconfig->gpsnmea_port, pconfig->gpsnmea_speed); if (s_gpsnmea_port_fd != MYFDERROR) { #if __WIN32__ - read_gps_th = (HANDLE)_beginthreadex (NULL, 0, read_gpsnmea_thread, (void*)(long)s_gpsnmea_port_fd, 0, NULL); + read_gps_th = (HANDLE)_beginthreadex (NULL, 0, read_gpsnmea_thread, (void*)(ptrdiff_t)s_gpsnmea_port_fd, 0, NULL); if (read_gps_th == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Could not create GPS NMEA listening thread.\n"); @@ -144,7 +158,7 @@ int dwgpsnmea_init (struct misc_config_s *pconfig, int debug) } #else int e; - e = pthread_create (&read_gps_tid, NULL, read_gpsnmea_thread, (void*)(long)s_gpsnmea_port_fd); + e = pthread_create (&read_gps_tid, NULL, read_gpsnmea_thread, (void*)(ptrdiff_t)s_gpsnmea_port_fd); if (e != 0) { text_color_set(DW_COLOR_ERROR); perror("Could not create GPS NMEA listening thread."); @@ -166,12 +180,10 @@ int dwgpsnmea_init (struct misc_config_s *pconfig, int debug) /* Return fd to share if waypoint wants same device. */ -/* Currently both are fixed speed at 4800. */ -/* If that ever becomes configurable, that needs to be compared too. */ MYFDTYPE dwgpsnmea_get_fd(char *wp_port_name, int speed) { - if (strcmp(s_save_configp->gpsnmea_port, wp_port_name) == 0 && speed == 4800) { + if (strcmp(s_save_configp->gpsnmea_port, wp_port_name) == 0 && speed == s_save_configp->gpsnmea_speed) { return (s_gpsnmea_port_fd); } return (MYFDERROR); @@ -201,7 +213,7 @@ static unsigned __stdcall read_gpsnmea_thread (void *arg) static void * read_gpsnmea_thread (void *arg) #endif { - MYFDTYPE fd = (MYFDTYPE)(long)arg; + MYFDTYPE fd = (MYFDTYPE)(ptrdiff_t)arg; // Maximum length of message from GPS receiver is 82 according to some people. // Make buffer considerably larger to be safe. @@ -215,7 +227,7 @@ static void * read_gpsnmea_thread (void *arg) if (s_debug >= 2) { text_color_set(DW_COLOR_DEBUG); - dw_printf ("read_gpsnmea_thread (%d)\n", (int)(long)arg); + dw_printf ("read_gpsnmea_thread (%d)\n", (int)(ptrdiff_t)arg); } dwgps_clear (&info); @@ -250,9 +262,12 @@ static void * read_gpsnmea_thread (void *arg) } dwgps_set_data (&info); - // TODO: doesn't exist yet - serial_port_close(fd); + serial_port_close(s_gpsnmea_port_fd); s_gpsnmea_port_fd = MYFDERROR; + // TODO: If the open() was in this thread, we could wait a while and + // try to open again. That would allow recovery if the USB GPS device + // is unplugged and plugged in again. break; /* terminate thread. */ } @@ -273,71 +288,53 @@ static void * read_gpsnmea_thread (void *arg) } /* Process sentence. */ +// TODO: More general: Ignore the second letter rather than recognizing only GP... and GN... - if (strncmp(gps_msg, "$GPRMC", 6) == 0) { + if (strncmp(gps_msg, "$GPRMC", 6) == 0 || + strncmp(gps_msg, "$GNRMC", 6) == 0) { - f = dwgpsnmea_gprmc (gps_msg, 0, &info.dlat, &info.dlon, &info.speed_knots, &info.track); + // Here we just tuck away the course and speed. + // Fix and location will be updated by GxGGA. - if (f == DWFIX_ERROR) { + double ignore_dlat; + double ignore_dlon; + + f = dwgpsnmea_gprmc (gps_msg, 0, &ignore_dlat, &ignore_dlon, &info.speed_knots, &info.track); - /* Parse error. Shouldn't happen. Better luck next time. */ - text_color_set(DW_COLOR_INFO); + if (f == DWFIX_ERROR) { + /* Parse error. Shouldn't happen. Better luck next time. */ + text_color_set(DW_COLOR_ERROR); dw_printf ("GPSNMEA: Error parsing $GPRMC sentence.\n"); dw_printf ("%s\n", gps_msg); } - else if (f == DWFIX_2D) { - - if (info.fix != DWFIX_2D && info.fix != DWFIX_3D) { - - text_color_set(DW_COLOR_INFO); - dw_printf ("GPSNMEA: Location fix is now available.\n"); - - info.fix = DWFIX_2D; // Don't know if 2D or 3D. Take minimum. - } - info.timestamp = time(NULL); - if (s_debug >= 2) { - text_color_set(DW_COLOR_DEBUG); - dwgps_print ("GPSNMEA: ", &info); - } - dwgps_set_data (&info); - } - else { - - if (info.fix == DWFIX_2D || info.fix == DWFIX_3D) { - - text_color_set(DW_COLOR_INFO); - dw_printf ("GPSNMEA: Lost location fix.\n"); - } - info.fix = f; /* lost it. */ - info.timestamp = time(NULL); - if (s_debug >= 2) { - text_color_set(DW_COLOR_DEBUG); - dwgps_print ("GPSNMEA: ", &info); - } - dwgps_set_data (&info); - } - } - else if (strncmp(gps_msg, "$GPGGA", 6) == 0) { + + else if (strncmp(gps_msg, "$GPGGA", 6) == 0 || + strncmp(gps_msg, "$GNGGA", 6) == 0) { int nsat; f = dwgpsnmea_gpgga (gps_msg, 0, &info.dlat, &info.dlon, &info.altitude, &nsat); - /* Only switch between 2D & 3D. */ - /* Let GPRMC handle other changes in fix state and data transfer. */ - if (f == DWFIX_ERROR) { - /* Parse error. Shouldn't happen. Better luck next time. */ - text_color_set(DW_COLOR_INFO); + text_color_set(DW_COLOR_ERROR); dw_printf ("GPSNMEA: Error parsing $GPGGA sentence.\n"); dw_printf ("%s\n", gps_msg); } - else if ((f == DWFIX_3D && info.fix == DWFIX_2D) || - (f == DWFIX_2D && info.fix == DWFIX_3D)) { - text_color_set(DW_COLOR_INFO); - dw_printf ("GPSNMEA: Location fix is now %dD.\n", (int)f); - info.fix = f; + else { + if (f != info.fix) { // Print change in location fix. + text_color_set(DW_COLOR_INFO); + if (f == DWFIX_NO_FIX) dw_printf ("GPSNMEA: Location fix has been lost.\n"); + if (f == DWFIX_2D) dw_printf ("GPSNMEA: Location fix is now 2D.\n"); + if (f == DWFIX_3D) dw_printf ("GPSNMEA: Location fix is now 3D.\n"); + info.fix = f; + } + info.timestamp = time(NULL); + if (s_debug >= 2) { + text_color_set(DW_COLOR_DEBUG); + dwgps_print ("GPSNMEA: ", &info); + } + dwgps_set_data (&info); } } } @@ -418,7 +415,7 @@ static int remove_checksum (char *sent, int quiet) * * Name: dwgpsnmea_gprmc * - * Purpose: Parse $GPRMC sentence and extract interesing parts. + * Purpose: Parse $GPRMC sentence and extract interesting parts. * * Inputs: sentence NMEA sentence. * @@ -428,9 +425,11 @@ static int remove_checksum (char *sent, int quiet) * odlon longitude * oknots speed * ocourse direction of travel. - * + * * Left undefined if not valid. * + * Note: RMC does not contain altitude. + * * Returns: DWFIX_ERROR Parse error. * DWFIX_NO_FIX GPS is there but Position unknown. Could be temporary. * DWFIX_2D Valid position. We don't know if it is really 2D or 3D. @@ -567,7 +566,7 @@ dwfix_t dwgpsnmea_gprmc (char *sentence, int quiet, double *odlat, double *odlon * * Name: dwgpsnmea_gpgga * - * Purpose: Parse $GPGGA sentence and extract interesing parts. + * Purpose: Parse $GPGGA sentence and extract interesting parts. * * Inputs: sentence NMEA sentence. * @@ -580,10 +579,13 @@ dwfix_t dwgpsnmea_gprmc (char *sentence, int quiet, double *odlat, double *odlon * * Left undefined if not valid. * + * Note: GGA has altitude but not course and speed so we need to use both. + * * Returns: DWFIX_ERROR Parse error. * DWFIX_NO_FIX GPS is there but Position unknown. Could be temporary. * DWFIX_2D Valid position. We don't know if it is really 2D or 3D. * Take more cautious value so we don't try using altitude. + * DWFIX_3D Valid 3D position. * * Examples: $GPGGA,001429.00,,,,,0,00,99.99,,,,,,*68 * $GPGGA,212407.000,4237.1505,N,07120.8602,W,0,00,,,M,,M,,*58 @@ -592,9 +594,6 @@ dwfix_t dwgpsnmea_gprmc (char *sentence, int quiet, double *odlat, double *odlon * *--------------------------------------------------------------------*/ - -// TODO: in progress... - dwfix_t dwgpsnmea_gpgga (char *sentence, int quiet, double *odlat, double *odlon, float *oalt, int *onsat) { char stemp[NMEA_MAX_LEN]; /* Make copy because parsing is destructive. */ @@ -691,7 +690,7 @@ dwfix_t dwgpsnmea_gpgga (char *sentence, int quiet, double *odlat, double *odlon return (DWFIX_ERROR); } - // TODO: num sat... + // TODO: num sat... Why would we care? /* * We can distinguish between 2D & 3D fix by presence diff --git a/dwgpsnmea.h b/src/dwgpsnmea.h similarity index 100% rename from dwgpsnmea.h rename to src/dwgpsnmea.h diff --git a/src/dwsock.c b/src/dwsock.c new file mode 100644 index 00000000..6324a2fc --- /dev/null +++ b/src/dwsock.c @@ -0,0 +1,451 @@ + +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2017 John Langner, WB2OSZ +// +// 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, see . +// + + +/*------------------------------------------------------------------ + * + * Module: dwsock.c + * + * Purpose: Functions for TCP sockets. + * + * Description: These are used for connecting between different applications, + * possibly on different hosts. + * + * New in version 1.5: + * Duplicate code already exists in multiple places and I was about + * to add another one. Instead, we will gather the common code here + * instead of having yet another copy. + * + *---------------------------------------------------------------*/ + +#include "direwolf.h" // Sets _WIN32_WINNT for XP API level needed by ws2tcpip.h + +#if __WIN32__ + +#include +#include // _WIN32_WINNT must be set to 0x0501 before including this + +#else + +#include +#include +#include +#include +#include +#include +#include +#include +#include +//#include +#include + +#endif + +#include +#include +#include +#include +#include +#include + +#include "textcolor.h" +#include "dwsock.h" + +static void shuffle (struct addrinfo *host[], int nhosts); + + +/*------------------------------------------------------------------- + * + * Name: dwsock_init + * + * Purpose: Preparation before using socket interface. + * + * Inputs: none + * + * Returns: 0 for success, -1 for error. + * + * Errors: Message is printed. I've never seen it fail. + * + * Description: Doesn't do anything for Linux. + * + * TODO: Use this instead of own copy in aclients.c + * TODO: Use this instead of own copy in appserver.c + * TODO: Use this instead of own copy in audio_win.c + * TODO: Use this instead of own copy in igate.c + * TODO: Use this instead of own copy in kissnet.c + * TODO: Use this instead of own copy in kissutil.c + * TODO: Use this instead of own copy in server.c + * TODO: Use this instead of own copy in tnctest.c + * TODO: Use this instead of own copy in ttcalc.c + * + *--------------------------------------------------------------------*/ + +int dwsock_init(void) +{ +#if __WIN32__ + WSADATA wsadata; + int err; + + err = WSAStartup (MAKEWORD(2,2), &wsadata); + if (err != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("WSAStartup failed, error: %d\n", err); + return (-1); + } + + if (LOBYTE(wsadata.wVersion) != 2 || HIBYTE(wsadata.wVersion) != 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Could not find a usable version of Winsock.dll\n"); + WSACleanup(); + return (-1); + } +#endif + return (0); + +} /* end dwsock_init */ + + + +/*------------------------------------------------------------------- + * + * Name: sock_connect + * + * Purpose: Connect to given host / port. + * + * Inputs: hostname - Host name or IP address. + * + * port - TCP port as text string. + * + * description - Description of the remote server to be used in error message. + * e.g. "APRS-IS (Igate) Server" or "TCP KISS TNC". + * + * allow_ipv6 - True to allow IPv6. Otherwise only IPv4. + * + * debug - Print debugging information. + * + * Outputs: ipaddr_str - The IP address, in text form, is placed here in case + * the caller wants it. Should be DWSOCK_IPADDR_LEN bytes. + * + * Returns: Socket Handle / file descriptor or -1 for error. + * + * Errors: (1) Can't find address for given host name. + * + * Print error and return -1. + * + * (2) Can't connect to one of the address(es). + * + * Silently try the next one. + * + * (3) Can't connect to any of the address(es). + * + * Nothing is printed for success. The caller might do that + * to provide confirmation on what is happening. + * + *--------------------------------------------------------------------*/ + +int dwsock_connect (char *hostname, char *port, char *description, int allow_ipv6, int debug, char ipaddr_str[DWSOCK_IPADDR_LEN]) +{ +#define MAX_HOSTS 50 + + struct addrinfo hints; + struct addrinfo *ai_head = NULL; + struct addrinfo *ai; + struct addrinfo *hosts[MAX_HOSTS]; + int num_hosts, n; + int err; + int server_sock = -1; + + strlcpy (ipaddr_str, "???", DWSOCK_IPADDR_LEN); + memset (&hints, 0, sizeof(hints)); + + hints.ai_family = AF_UNSPEC; /* Allow either IPv4 or IPv6. */ + if ( ! allow_ipv6) { + hints.ai_family = AF_INET; /* IPv4 only. */ + } + hints.ai_socktype = SOCK_STREAM; + hints.ai_protocol = IPPROTO_TCP; + +/* + * First, we need to look up the DNS name to get IP address. + * It is possible to have multiple addresses. + */ + + ai_head = NULL; + err = getaddrinfo(hostname, port, &hints, &ai_head); + if (err != 0) { + text_color_set(DW_COLOR_ERROR); +#if __WIN32__ + dw_printf ("Can't get address for %s, %s, err=%d\n", + description, hostname, WSAGetLastError()); +#else + dw_printf ("Can't get address for %s, %s, %s\n", + description, hostname, gai_strerror(err)); +#endif + freeaddrinfo(ai_head); + return (-1); + } + + if (debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("getaddrinfo returns:\n"); + } + + num_hosts = 0; + for (ai = ai_head; ai != NULL; ai = ai->ai_next) { + + if (debug) { + text_color_set(DW_COLOR_DEBUG); + dwsock_ia_to_text (ai->ai_family, ai->ai_addr, ipaddr_str, DWSOCK_IPADDR_LEN); + dw_printf (" %s\n", ipaddr_str); + } + + hosts[num_hosts] = ai; + if (num_hosts < MAX_HOSTS) num_hosts++; + } + + shuffle (hosts, num_hosts); + + if (debug) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("addresses for hostname:\n"); + for (n=0; nai_family, hosts[n]->ai_addr, ipaddr_str, DWSOCK_IPADDR_LEN); + dw_printf (" %s\n", ipaddr_str); + } + } + +/* + * Try each address until we find one that is successful. + */ + for (n = 0; n < num_hosts; n++) { +#if __WIN32__ + SOCKET is; +#else + int is; +#endif + ai = hosts[n]; + + dwsock_ia_to_text (ai->ai_family, ai->ai_addr, ipaddr_str, DWSOCK_IPADDR_LEN); + is = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol); +#if __WIN32__ + if (is == INVALID_SOCKET) { + printf ("Socket creation failed, err=%d", WSAGetLastError()); + WSACleanup(); + is = -1; + continue; + } +#else + if (err != 0) { + printf ("Socket creation failed, err=%s", gai_strerror(err)); + (void) close (is); + is = -1; + continue; + } +#endif + +#ifndef DEBUG_DNS + err = connect(is, ai->ai_addr, (int)ai->ai_addrlen); +#if __WIN32__ + if (err == SOCKET_ERROR) { +#if DEBUGx + printf("Connect to %s on %s (%s), port %s failed.\n", + description, hostname, ipaddr_str, port); +#endif + closesocket (is); + is = -1; + continue; + } +#else + if (err != 0) { +#if DEBUGx + printf("Connect to %s on %s (%s), port %s failed.\n", + description, hostname, ipaddr_str, port); +#endif + (void) close (is); + is = -1; + continue; + } + + /* IGate documentation says to use no delay. */ + /* Does it really make a difference? */ + int flag = 1; + err = setsockopt (is, IPPROTO_TCP, TCP_NODELAY, (void*)(long)(&flag), (socklen_t)sizeof(flag)); + if (err < 0) { + printf("setsockopt TCP_NODELAY failed.\n"); + } +#endif + +/* Success. */ + + + server_sock = is; +#endif + break; + } + + freeaddrinfo(ai_head); + +// no, caller should handle this. +// function should be generally be silent unless debug option. + + if (server_sock == -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Unable to connect to %s at %s (%s), port %s\n", + description, hostname, ipaddr_str, port ); + } + + return (server_sock); + +} /* end dwsock_connect */ + + + +/*------------------------------------------------------------------- + * + * Name: dwsock_bind + * + * Purpose: We also have a bunch of duplicate code for the server side. + * + * Inputs: + * + * TODO: Use this instead of own copy in audio.c + * TODO: Use this instead of own copy in audio_portaudio.c + * TODO: Use this instead of own copy in audio_win.c + * TODO: Use this instead of own copy in kissnet.c + * TODO: Use this instead of own copy in server.c + * + *--------------------------------------------------------------------*/ + +// Not implemented yet. + + +/* + * Addresses don't get mixed up very well. + * IPv6 always shows up last so we'd probably never + * end up using any of them for APRS-IS server. + * Add our own shuffle. + */ + +static void shuffle (struct addrinfo *host[], int nhosts) +{ + int j, k; + + assert (RAND_MAX >= nhosts); /* for % to work right */ + + if (nhosts < 2) return; + + srand (time(NULL)); + + for (j=0; j=0 && ksin_addr.S_un.S_un_b.s_b1, + sa4->sin_addr.S_un.S_un_b.s_b2, + sa4->sin_addr.S_un.S_un_b.s_b3, + sa4->sin_addr.S_un.S_un_b.s_b4); +#else + inet_ntop (AF_INET, &(sa4->sin_addr), pStringBuf, StringBufSize); +#endif + break; + + case AF_INET6: + sa6 = (struct sockaddr_in6 *)pAddr; +#if __WIN32__ + snprintf (pStringBuf, StringBufSize, "%x:%x:%x:%x:%x:%x:%x:%x", + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[0]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[1]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[2]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[3]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[4]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[5]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[6]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[7])); +#else + inet_ntop (AF_INET6, &(sa6->sin6_addr), pStringBuf, StringBufSize); +#endif + break; + + default: + snprintf (pStringBuf, StringBufSize, "Invalid address family!"); + } + return pStringBuf; + +} /* end dwsock_ia_to_text */ + + +void dwsock_close (int fd) +{ +#if __WIN32__ + closesocket (fd); +#else + close (fd); +#endif +} + + + + +/* end dwsock.c */ diff --git a/src/dwsock.h b/src/dwsock.h new file mode 100644 index 00000000..986f6a29 --- /dev/null +++ b/src/dwsock.h @@ -0,0 +1,21 @@ + +/* dwsock.h - Socket helper functions. */ + +#ifndef DWSOCK_H +#define DWSOCK_H 1 + +#define DWSOCK_IPADDR_LEN 48 // Size of string to hold IPv4 or IPv6 address. + // I think 40 would be adequate but we'll make + // it a little larger just to be safe. + // Use INET6_ADDRSTRLEN (from netinet/in.h) instead? + +int dwsock_init (void); + +int dwsock_connect (char *hostname, char *port, char *description, int allow_ipv6, int debug, char ipaddr_str[DWSOCK_IPADDR_LEN]); + /* ipaddr_str needs to be at least SOCK_IPADDR_LEN bytes */ + +char *dwsock_ia_to_text (int Family, void * pAddr, char * pStringBuf, size_t StringBufSize); + +void dwsock_close (int fd); + +#endif \ No newline at end of file diff --git a/encode_aprs.c b/src/encode_aprs.c similarity index 87% rename from encode_aprs.c rename to src/encode_aprs.c index f46bca89..20992bf7 100644 --- a/encode_aprs.c +++ b/src/encode_aprs.c @@ -81,8 +81,14 @@ typedef struct position_s { static int set_norm_position (char symtab, char symbol, double dlat, double dlong, int ambiguity, position_t *presult) { + // An over zealous compiler might complain about l*itude_to_str writing + // N characters plus nul to an N character field so we stick it into a + // larger temp then copy the desired number of bytes. (Issue 296) - latitude_to_str (dlat, ambiguity, presult->lat); + char stemp[16]; + + latitude_to_str (dlat, ambiguity, stemp); + memcpy (presult->lat, stemp, sizeof(presult->lat)); if (symtab != '/' && symtab != '\\' && ! isdigit(symtab) && ! isupper(symtab)) { text_color_set(DW_COLOR_ERROR); @@ -90,7 +96,8 @@ static int set_norm_position (char symtab, char symbol, double dlat, double dlon } presult->sym_table_id = symtab; - longitude_to_str (dlong, ambiguity, presult->lon); + longitude_to_str (dlong, ambiguity, stemp); + memcpy (presult->lon, stemp, sizeof(presult->lon)); if (symbol < '!' || symbol > '~') { text_color_set(DW_COLOR_ERROR); @@ -118,7 +125,7 @@ static int set_norm_position (char symtab, char symbol, double dlat, double dlon * height - Feet. * gain - dBi. * - * course - Degress, 0 - 360 (360 equiv. to 0). + * course - Degrees, 0 - 360 (360 equiv. to 0). * Use G_UNKNOWN for none or unknown. * speed - knots. * @@ -336,7 +343,7 @@ static int phg_data_extension (int power, int height, int gain, char *dir, char * * Purpose: Fill in parts of the course & speed data extension. * - * Inputs: course - Degress, 0 - 360 (360 equiv. to 0). + * Inputs: course - Degrees, 0 - 360 (360 equiv. to 0). * Use G_UNKNOWN for none or unknown. * * speed - knots. @@ -487,7 +494,7 @@ static int frequency_spec (float freq, float tone, float offset, char *presult) * gain - dB. Not clear if it is dBi or dBd. * dir - Directivity: N, NE, etc., omni. * - * course - Degress, 0 - 360 (360 equiv. to 0). + * course - Degrees, 0 - 360 (360 equiv. to 0). * Use G_UNKNOWN for none or unknown. * speed - knots. // TODO: should distinguish unknown(not revevant) vs. known zero. * @@ -497,7 +504,7 @@ static int frequency_spec (float freq, float tone, float offset, char *presult) * * comment - Additional comment text. * - * result_size - Ammount of space for result, provideed by + * result_size - Amount of space for result, provided by * caller, to avoid buffer overflow. * * Outputs: presult - Stored here. Should be at least ??? bytes. @@ -512,7 +519,7 @@ static int frequency_spec (float freq, float tone, float offset, char *presult) * Power/height/gain/directivity or * Course/speed. * - * Afer that, + * After that, * *----------------------------------------------------------------*/ @@ -544,6 +551,15 @@ int encode_position (int messaging, int compressed, double lat, double lon, int int result_len = 0; if (compressed) { + +// Thought: +// https://groups.io/g/direwolf/topic/92718535#6886 +// When speed is zero, we could put the altitude in the compressed +// position rather than having /A=999999. +// However, the resolution would be decreased and that could be important +// when hiking in hilly terrain. It would also be confusing to +// flip back and forth between two different representations. + aprs_compressed_pos_t *p = (aprs_compressed_pos_t *)presult; p->dti = messaging ? '=' : '!'; @@ -629,7 +645,7 @@ int encode_position (int messaging, int compressed, double lat, double lon, int * gain - dB. Not clear if it is dBi or dBd. * dir - Direction: N, NE, etc., omni. * - * course - Degress, 0 - 360 (360 equiv. to 0). + * course - Degrees, 0 - 360 (360 equiv. to 0). * Use G_UNKNOWN for none or unknown. * speed - knots. * @@ -639,7 +655,7 @@ int encode_position (int messaging, int compressed, double lat, double lon, int * * comment - Additional comment text. * - * result_size - Ammount of space for result, provideed by + * result_size - Amount of space for result, provided by * caller, to avoid buffer overflow. * * Outputs: presult - Stored here. Should be at least ??? bytes. @@ -759,6 +775,63 @@ int encode_object (char *name, int compressed, time_t thyme, double lat, double } /* end encode_object */ + +/*------------------------------------------------------------------ + * + * Name: encode_message + * + * Purpose: Construct info part for APRS "message" format. + * + * Inputs: addressee - Addressed to, up to 9 characters. + * text - Text part of the message. + * id - Identifier, 0 to 5 characters. + * result_size - Amount of space for result, provided by + * caller, to avoid buffer overflow. + * + * Outputs: presult - Stored here. + * + * Returns: Number of characters in result. + * + * Description: + * + *----------------------------------------------------------------*/ + + +typedef struct aprs_message_s { + char dti; /* : Data Type Indicator */ + char addressee[9]; /* Fixed width 9 characters. */ + char sep; /* : separator */ + char text; + } aprs_message_t; + +int encode_message (char *addressee, char *text, char *id, char *presult, size_t result_size) +{ + aprs_message_t *p = (aprs_message_t *) presult; + int n; + + p->dti = ':'; + + memset (p->addressee, ' ', sizeof(p->addressee)); + n = strlen(addressee); + if (n > (int)(sizeof(p->addressee))) n = sizeof(p->addressee); + memcpy (p->addressee, addressee, n); + + p->sep = ':'; + p->text = '\0'; + + strlcat (presult, text, result_size); + if (strlen(id) > 0) { + strlcat (presult, "{", result_size); + strlcat (presult, id, result_size); + } + + return (strlen(presult)); + +} /* end encode_message */ + + + + /*------------------------------------------------------------------ * * Name: main @@ -767,7 +840,7 @@ int encode_object (char *name, int compressed, time_t thyme, double lat, double * * Description: Just a smattering, not an organized test. * - * $ rm a.exe ; gcc -DEN_MAIN encode_aprs.c latlong.c textcolor.c ; ./a.exe + * $ rm a.exe ; gcc -DEN_MAIN encode_aprs.c latlong.c textcolor.c misc.a ; ./a.exe * *----------------------------------------------------------------*/ @@ -796,7 +869,7 @@ int main (int argc, char *argv[]) dw_printf ("%s\n", result); if (strcmp(result, "!4234.61ND07126.47W&PHG7368") != 0) { dw_printf ("ERROR! line %d\n", __LINE__); errors++; } -/* with freq & tone. minus offset, no offset, explict simplex. */ +/* with freq & tone. minus offset, no offset, explicit simplex. */ encode_position (0, 0, 42+34.61/60, -(71+26.47/60), 0, G_UNKNOWN, 'D', '&', 0, 0, 0, NULL, G_UNKNOWN, 0, 146.955, 74.4, -0.6, NULL, result, sizeof(result)); @@ -881,9 +954,35 @@ int main (int argc, char *argv[]) exit (EXIT_FAILURE); } + +/*********** Message. ***********/ + + + encode_message ("N2GH", "some stuff", "", result, sizeof(result)); + dw_printf ("%s\n", result); + if (strcmp(result, ":N2GH :some stuff") != 0) { dw_printf ("ERROR! line %d\n", __LINE__); errors++; } + + + encode_message ("N2GH", "other stuff", "12345", result, sizeof(result)); + dw_printf ("%s\n", result); + if (strcmp(result, ":N2GH :other stuff{12345") != 0) { dw_printf ("ERROR! line %d\n", __LINE__); errors++; } + + + encode_message ("WB2OSZ-123", "other stuff", "12345", result, sizeof(result)); + dw_printf ("%s\n", result); + if (strcmp(result, ":WB2OSZ-12:other stuff{12345") != 0) { dw_printf ("ERROR! line %d\n", __LINE__); errors++; } + + + if (errors != 0) { + text_color_set (DW_COLOR_ERROR); + dw_printf ("Encode APRS test FAILED with %d errors.\n", errors); + exit (EXIT_FAILURE); + } + text_color_set (DW_COLOR_REC); dw_printf ("Encode APRS test PASSED with no errors.\n"); exit (EXIT_SUCCESS); + } /* end main */ diff --git a/encode_aprs.h b/src/encode_aprs.h similarity index 86% rename from encode_aprs.h rename to src/encode_aprs.h index b100e11c..dc7b8bdf 100644 --- a/encode_aprs.h +++ b/src/encode_aprs.h @@ -14,3 +14,4 @@ int encode_object (char *name, int compressed, time_t thyme, double lat, double float freq, float tone, float offset, char *comment, char *presult, size_t result_size); +int encode_message (char *addressee, char *text, char *id, char *presult, size_t result_size); diff --git a/fcs_calc.c b/src/fcs_calc.c similarity index 84% rename from fcs_calc.c rename to src/fcs_calc.c index 97e34a5c..f49a2f55 100644 --- a/fcs_calc.c +++ b/src/fcs_calc.c @@ -87,6 +87,22 @@ unsigned short fcs_calc (unsigned char *data, int len) } +/* + * CRC is also used for duplicate checking for the digipeater and IGate. + * A packet is considered a duplicate if the source, destination, and + * information parts match. In other words, we ignore the via path + * which changes along the way. + * Rather than keeping a variable length string we just keep a 16 bit + * CRC which takes less memory and processing to compare. + * + * This can result in occasional false matches. If we had a random + * 16 bit number, there is a 1/65536 ( = 0.0015 % ) chance that it will + * match and we will drop something that should be passed along. + * + * Looking at it another way, there is a 0.9999847412109375 (out of 1) + * probability of doing the right thing. + */ + /* * This can be used when we want to calculate a single CRC over disjoint data. * diff --git a/fcs_calc.h b/src/fcs_calc.h similarity index 100% rename from fcs_calc.h rename to src/fcs_calc.h diff --git a/src/fsk_demod_state.h b/src/fsk_demod_state.h new file mode 100644 index 00000000..c9b26c23 --- /dev/null +++ b/src/fsk_demod_state.h @@ -0,0 +1,558 @@ +/* fsk_demod_state.h */ + +#ifndef FSK_DEMOD_STATE_H + +#include // int64_t + +#include "rpack.h" + +#include "audio.h" // for enum modem_t + +/* + * Demodulator state. + * The name of the file is from we only had FSK. Now we have other techniques. + * Different copy is required for each channel & subchannel being processed concurrently. + */ + +// TODO1.2: change prefix from BP_ to DSP_ + +typedef enum bp_window_e { BP_WINDOW_TRUNCATED, + BP_WINDOW_COSINE, + BP_WINDOW_HAMMING, + BP_WINDOW_BLACKMAN, + BP_WINDOW_FLATTOP } bp_window_t; + +// Experimental low pass filter to detect DC bias or low frequency changes. +// IIR behaves like an analog R-C filter. +// Intuitively, it seems like FIR would be better because it is based on a finite history. +// However, it would require MANY taps and a LOT of computation for a low frequency. +// We can use a little trick here to keep a running average. +// This would be equivalent to convolving with an array of all 1 values. +// That would eliminate the need to multiply. +// We can also eliminate the need to add them all up each time by keeping a running total. +// Add a sample to the total when putting it in our array of recent samples. +// Subtract it from the total when it gets pushed off the end. +// We can also eliminate the need to shift them all down by using a circular buffer. + +#define CIC_LEN_MAX 4000 + +typedef struct cic_s { + int len; // Number of elements used. + // Might want to dynamically allocate. + short in[CIC_LEN_MAX]; // Samples coming in. + int sum; // Running sum. + int inext; // Next position to fill. +} cic_t; + + +#define MAX_FILTER_SIZE 480 /* 401 is needed for profile A, 300 baud & 44100. Revisit someday. */ + // Size comes out to 417 for 1200 bps with 48000 sample rate + // v1.7 - Was 404. Bump up to 480. + +struct demodulator_state_s +{ +/* + * These are set once during initialization. + */ + enum modem_t modem_type; // MODEM_AFSK, MODEM_8PSK, etc. + +// enum v26_e v26_alt; // Which alternative when V.26. + + char profile; // 'A', 'B', etc. Upper case. + // Only needed to see if we are using 'F' to take fast path. + +#define TICKS_PER_PLL_CYCLE ( 256.0 * 256.0 * 256.0 * 256.0 ) + + int pll_step_per_sample; // PLL is advanced by this much each audio sample. + // Data is sampled when it overflows. + + +/* + * Window type for the various filters. + */ + + bp_window_t lp_window; + +/* + * Alternate Low pass filters. + * First is arbitrary number for quick IIR. + * Second is frequency as ratio to baud rate for FIR. + */ + int lpf_use_fir; /* 0 for IIR, 1 for FIR. */ + + float lpf_iir; /* Only if using IIR. */ + + float lpf_baud; /* Cutoff frequency as fraction of baud. */ + /* Intuitively we'd expect this to be somewhere */ + /* in the range of 0.5 to 1. */ + /* In practice, it turned out a little larger */ + /* for profiles B, C, D. */ + + float lp_filter_width_sym; /* Length in number of symbol times. */ + +#define lp_filter_len_bits lp_filter_width_sym // FIXME: temp hack + + int lp_filter_taps; /* Size of Low Pass filter, in audio samples. */ + +#define lp_filter_size lp_filter_taps // FIXME: temp hack + + +/* + * Automatic gain control. Fast attack and slow decay factors. + */ + float agc_fast_attack; + float agc_slow_decay; + +/* + * Use a longer term view for reporting signal levels. + */ + float quick_attack; + float sluggish_decay; + +/* + * Hysteresis before final demodulator 0 / 1 decision. + */ + float hysteresis; + int num_slicers; /* >1 for multiple slicers. */ + +/* + * Phase Locked Loop (PLL) inertia. + * Larger number means less influence by signal transitions. + * It is more resistant to change when locked on to a signal. + */ + float pll_locked_inertia; + float pll_searching_inertia; + + +/* + * Optional band pass pre-filter before mark/space detector. + */ + int use_prefilter; /* True to enable it. */ + + float prefilter_baud; /* Cutoff frequencies, as fraction of */ + /* baud rate, beyond tones used. */ + /* Example, if we used 1600/1800 tones at */ + /* 300 baud, and this was 0.5, the cutoff */ + /* frequencies would be: */ + /* lower = min(1600,1800) - 0.5 * 300 = 1450 */ + /* upper = max(1600,1800) + 0.5 * 300 = 1950 */ + + float pre_filter_len_sym; // Length in number of symbol times. +#define pre_filter_len_bits pre_filter_len_sym // temp until all references changed. + + bp_window_t pre_window; // Window type for filter shaping. + + int pre_filter_taps; // Calculated number of filter taps. +#define pre_filter_size pre_filter_taps // temp until all references changed. + + float pre_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + float raw_cb[MAX_FILTER_SIZE] __attribute__((aligned(16))); // audio in, need better name. + +/* + * The rest are continuously updated. + */ + + unsigned int lo_phase; /* Local oscillator for PSK. */ + + +/* + * Use half of the AGC code to get a measure of input audio amplitude. + * These use "quick" attack and "sluggish" decay while the + * AGC uses "fast" attack and "slow" decay. + */ + + float alevel_rec_peak; + float alevel_rec_valley; + float alevel_mark_peak; + float alevel_space_peak; + +/* + * Outputs from the mark and space amplitude detection, + * used as inputs to the FIR lowpass filters. + * Kernel for the lowpass filters. + */ + + float lp_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + float m_peak, s_peak; + float m_valley, s_valley; + float m_amp_prev, s_amp_prev; + + +/* + * For the PLL and data bit timing. + * starting in version 1.2 we can have multiple slicers for one demodulator. + * Each slicer has its own PLL and HDLC decoder. + */ + +/* + * Version 1.3: Clean up subchan vs. slicer. + * + * Originally some number of CHANNELS (originally 2, later 6) + * which can have multiple parallel demodulators called SUB-CHANNELS. + * This was originally for staggered frequencies for HF SSB. + * It can also be used for multiple demodulators with the same + * frequency but other differing parameters. + * Each subchannel has its own demodulator and HDLC decoder. + * + * In version 1.2 we added multiple SLICERS. + * The data structure, here, has multiple slicers per + * demodulator (subchannel). Due to fuzzy thinking or + * expediency, the multiple slicers got mapped into subchannels. + * This means we can't use both multiple decoders and + * multiple slicers at the same time. + * + * Clean this up in 1.3 and keep the concepts separate. + * This means adding a third variable many places + * we are passing around the origin. + * + */ + struct { + + signed int data_clock_pll; // PLL for data clock recovery. + // It is incremented by pll_step_per_sample + // for each audio sample. + // Must be 32 bits!!! + // So far, this is the case for every compiler used. + + signed int prev_d_c_pll; // Previous value of above, before + // incrementing, to detect overflows. + + int pll_symbol_count; // Number symbols during time nudge_total is accumulated. + int64_t pll_nudge_total; // Sum of DPLL nudge amounts. + // Both of these are cleared at start of frame. + // At end of frame, we can see if incoming + // baud rate is a little off. + + int prev_demod_data; // Previous data bit detected. + // Used to look for transitions. + float prev_demod_out_f; + + /* This is used only for "9600" baud data. */ + + int lfsr; // Descrambler shift register. + + // This is for detecting phase lock to incoming signal. + + int good_flag; // Set if transition is near where expected, + // i.e. at a good time. + int bad_flag; // Set if transition is not where expected, + // i.e. at a bad time. + unsigned char good_hist; // History of good transitions for past octet. + unsigned char bad_hist; // History of bad transitions for past octet. + unsigned int score; // History of whether good triumphs over bad + // for past 32 symbols. + int data_detect; // True when locked on to signal. + + } slicer [MAX_SLICERS]; // Actual number in use is num_slicers. + // Should be in range 1 .. MAX_SLICERS, +/* + * Version 1.6: + * + * This has become quite disorganized and messy with different combinations of + * fields used for different demodulator types. Start to reorganize it into a common + * part (with things like the DPLL for clock recovery), and separate sections + * for each of the demodulator types. + * Still a lot to do here. + */ + + union { + +////////////////////////////////////////////////////////////////////////////////// +// // +// AFSK only - new method in 1.7 // +// // +////////////////////////////////////////////////////////////////////////////////// + + + struct afsk_only_s { + + unsigned int m_osc_phase; // Phase for Mark local oscillator. + unsigned int m_osc_delta; // How much to change for each audio sample. + + unsigned int s_osc_phase; // Phase for Space local oscillator. + unsigned int s_osc_delta; // How much to change for each audio sample. + + unsigned int c_osc_phase; // Phase for Center frequency local oscillator. + unsigned int c_osc_delta; // How much to change for each audio sample. + + // Need two mixers for profile "A". + + float m_I_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float m_Q_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + float s_I_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float s_Q_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + // Only need one mixer for profile "B". Reuse the same storage? + +//#define c_I_raw m_I_raw +//#define c_Q_raw m_Q_raw + float c_I_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float c_Q_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + int use_rrc; // Use RRC rather than generic low pass. + + float rrc_width_sym; /* Width of RRC filter in number of symbols. */ + + float rrc_rolloff; /* Rolloff factor for RRC. Between 0 and 1. */ + + float prev_phase; // To see phase shift between samples for FM demod. + + float normalize_rpsam; // Normalize to -1 to +1 for expected tones. + + } afsk; + +////////////////////////////////////////////////////////////////////////////////// +// // +// Baseband only, AKA G3RUH // +// // +////////////////////////////////////////////////////////////////////////////////// + +// TODO: Continue experiments with root raised cosine filter. +// Either switch to that or take out all the related stuff. + + struct bb_only_s { + + float rrc_width_sym; /* Width of RRC filter in number of symbols. */ + + float rrc_rolloff; /* Rolloff factor for RRC. Between 0 and 1. */ + + int rrc_filter_taps; // Number of elements used in the next two. + +// FIXME: TODO: reevaluate max size needed. + + float audio_in[MAX_FILTER_SIZE] __attribute__((aligned(16))); // Audio samples in. + + + float lp_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); // Low pass filter. + + // New in 1.7 - Polyphase filter to reduce CPU requirements. + + float lp_polyphase_1[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float lp_polyphase_2[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float lp_polyphase_3[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float lp_polyphase_4[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + float lp_1_iir_param; // very low pass filters to get DC offset. + float lp_1_out; + + float lp_2_iir_param; + float lp_2_out; + + float agc_1_fast_attack; // Signal envelope detection. + float agc_1_slow_decay; + float agc_1_peak; + float agc_1_valley; + + float agc_2_fast_attack; + float agc_2_slow_decay; + float agc_2_peak; + float agc_2_valley; + + float agc_3_fast_attack; + float agc_3_slow_decay; + float agc_3_peak; + float agc_3_valley; + + // CIC low pass filters to detect DC bias or low frequency changes. + // IIR behaves like an analog R-C filter. + // Intuitively, it seems like FIR would be better because it is based on a finite history. + // However, it would require MANY taps and a LOT of computation for a low frequency. + // We can use a little trick here to keep a running average. + // This would be equivalent to convolving with an array of all 1 values. + // That would eliminate the need to multiply. + // We can also eliminate the need to add them all up each time by keeping a running total. + // Add a sample to the total when putting it in our array of recent samples. + // Subtract it from the total when it gets pushed off the end. + // We can also eliminate the need to shift them all down by using a circular buffer. + // This only works with integers because float would have cumulated round off errors. + + cic_t cic_center1; + cic_t cic_above; + cic_t cic_below; + + } bb; + +////////////////////////////////////////////////////////////////////////////////// +// // +// PSK only. // +// // +////////////////////////////////////////////////////////////////////////////////// + + + struct psk_only_s { + + enum v26_e v26_alt; // Which alternative when V.26. + + float sin_table256[256]; // Precomputed sin table for speed. + + + // Optional band pass pre-filter before phase detector. + +// TODO? put back into common section? +// TODO? Why was I thinking that? + + int use_prefilter; // True to enable it. + + float prefilter_baud; // Cutoff frequencies, as fraction of baud rate, beyond tones used. + // In the case of PSK, we use only a single tone of 1800 Hz. + // If we were using 2400 bps (= 1200 baud), this would be + // the fraction of 1200 for the cutoff below and above 1800. + + + float pre_filter_width_sym; /* Length in number of symbol times. */ + + int pre_filter_taps; /* Size of pre filter, in audio samples. */ + + bp_window_t pre_window; + + float audio_in[MAX_FILTER_SIZE] __attribute__((aligned(16))); + float pre_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + // Use local oscillator or correlate with previous sample. + + int psk_use_lo; /* Use local oscillator rather than self correlation. */ + + unsigned int lo_step; /* How much to advance the local oscillator */ + /* phase for each audio sample. */ + + unsigned int lo_phase; /* Local oscillator phase accumulator for PSK. */ + + // After mixing with LO before low pass filter. + + float I_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); // signal * LO cos. + float Q_raw[MAX_FILTER_SIZE] __attribute__((aligned(16))); // signal * LO sin. + + // Number of delay line taps into previous symbol. + // They are one symbol period and + or - 45 degrees of the carrier frequency. + + int boffs; /* symbol length based on sample rate and baud. */ + int coffs; /* to get cos component of previous symbol. */ + int soffs; /* to get sin component of previous symbol. */ + + float delay_line_width_sym; + int delay_line_taps; // In audio samples. + + float delay_line[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + // Low pass filter Second is frequency as ratio to baud rate for FIR. + +// TODO? put back into common section? +// TODO? What are the tradeoffs? + float lpf_baud; /* Cutoff frequency as fraction of baud. */ + /* Intuitively we'd expect this to be somewhere */ + /* in the range of 0.5 to 1. */ + + float lp_filter_width_sym; /* Length in number of symbol times. */ + + int lp_filter_taps; /* Size of Low Pass filter, in audio samples (i.e. filter taps). */ + + bp_window_t lp_window; + + float lp_filter[MAX_FILTER_SIZE] __attribute__((aligned(16))); + + } psk; + + } u; // end of union for different demodulator types. + +}; + + +/*------------------------------------------------------------------- + * + * Name: pll_dcd_signal_transition2 + * dcd_each_symbol2 + * + * Purpose: New DCD strategy for 1.6. + * + * Inputs: D Pointer to demodulator state. + * + * chan Radio channel: 0 to MAX_CHANS - 1 + * + * subchan Which of multiple demodulators: 0 to MAX_SUBCHANS - 1 + * + * slice Slicer number: 0 to MAX_SLICERS - 1. + * + * dpll_phase Signed 32 bit counter for DPLL phase. + * Wraparound is where data is sampled. + * Ideally transitions would occur close to 0. + * + * Output: D->slicer[slice].data_detect - true when PLL is locked to incoming signal. + * + * Description: From the beginning, DCD was based on finding several flag octets + * in a row and dropping when eight bits with no transitions. + * It was less than ideal but we limped along with it all these years. + * This fell apart when FX.25 came along and a couple of the + * correlation tags have eight "1" bits in a row. + * + * Our new strategy is to keep a running score of how well demodulator + * output transitions match to where expected. + * + *--------------------------------------------------------------------*/ + +#include "hdlc_rec.h" // for dcd_change + +// These are good for 1200 bps AFSK. +// Might want to override for other modems. + +#ifndef DCD_THRESH_ON +#define DCD_THRESH_ON 30 // Hysteresis: Can miss 2 out of 32 for detecting lock. + // 31 is best for TNC Test CD. 30 almost as good. + // 30 better for 1200 regression test. +#endif + +#ifndef DCD_THRESH_OFF +#define DCD_THRESH_OFF 6 // Might want a little more fine tuning. +#endif + +#ifndef DCD_GOOD_WIDTH +#define DCD_GOOD_WIDTH 512 // No more than 1024!!! +#endif + +__attribute__((always_inline)) +inline static void pll_dcd_signal_transition2 (struct demodulator_state_s *D, int slice, int dpll_phase) +{ + if (dpll_phase > - DCD_GOOD_WIDTH * 1024 * 1024 && dpll_phase < DCD_GOOD_WIDTH * 1024 * 1024) { + D->slicer[slice].good_flag = 1; + } + else { + D->slicer[slice].bad_flag = 1; + } +} + +__attribute__((always_inline)) +inline static void pll_dcd_each_symbol2 (struct demodulator_state_s *D, int chan, int subchan, int slice) +{ + D->slicer[slice].good_hist <<= 1; + D->slicer[slice].good_hist |= D->slicer[slice].good_flag; + D->slicer[slice].good_flag = 0; + + D->slicer[slice].bad_hist <<= 1; + D->slicer[slice].bad_hist |= D->slicer[slice].bad_flag; + D->slicer[slice].bad_flag = 0; + + D->slicer[slice].score <<= 1; + // 2 is to detect 'flag' patterns with 2 transitions per octet. + D->slicer[slice].score |= (signed)__builtin_popcount(D->slicer[slice].good_hist) + - (signed)__builtin_popcount(D->slicer[slice].bad_hist) >= 2; + + int s = __builtin_popcount(D->slicer[slice].score); + if (s >= DCD_THRESH_ON) { + if (D->slicer[slice].data_detect == 0) { + D->slicer[slice].data_detect = 1; + dcd_change (chan, subchan, slice, D->slicer[slice].data_detect); + } + } + else if (s <= DCD_THRESH_OFF) { + if (D->slicer[slice].data_detect != 0) { + D->slicer[slice].data_detect = 0; + dcd_change (chan, subchan, slice, D->slicer[slice].data_detect); + } + } +} + + +#define FSK_DEMOD_STATE_H 1 +#endif diff --git a/fsk_filters.h b/src/fsk_filters.h similarity index 100% rename from fsk_filters.h rename to src/fsk_filters.h diff --git a/fsk_gen_filter.h b/src/fsk_gen_filter.h similarity index 100% rename from fsk_gen_filter.h rename to src/fsk_gen_filter.h diff --git a/src/fx25.h b/src/fx25.h new file mode 100644 index 00000000..99e9d261 --- /dev/null +++ b/src/fx25.h @@ -0,0 +1,96 @@ +#ifndef FX25_H +#define FX25_H + +#include // for uint64_t + + +/* Reed-Solomon codec control block */ +struct rs { + unsigned int mm; /* Bits per symbol */ + unsigned int nn; /* Symbols per block (= (1<mm) +#define NN (rs->nn) +#define ALPHA_TO (rs->alpha_to) +#define INDEX_OF (rs->index_of) +#define GENPOLY (rs->genpoly) +#define NROOTS (rs->nroots) +#define FCR (rs->fcr) +#define PRIM (rs->prim) +#define IPRIM (rs->iprim) +#define A0 (NN) + + + +__attribute__((always_inline)) +static inline int modnn(struct rs *rs, int x){ + while (x >= rs->nn) { + x -= rs->nn; + x = (x >> rs->mm) + (x & rs->nn); + } + return x; +} + +#define MODNN(x) modnn(rs,x) + + +#define ENCODE_RS encode_rs_char +#define DECODE_RS decode_rs_char +#define INIT_RS init_rs_char +#define FREE_RS free_rs_char + +#define DTYPE unsigned char + +void ENCODE_RS(struct rs *rs, DTYPE *data, DTYPE *bb); + +int DECODE_RS(struct rs *rs, DTYPE *data, int *eras_pos, int no_eras); + +struct rs *INIT_RS(unsigned int symsize, unsigned int gfpoly, + unsigned int fcr, unsigned int prim, unsigned int nroots); + +void FREE_RS(struct rs *rs); + + + +// These 3 are the external interface. +// Maybe these should be in a different file, separated from the internal stuff. + +void fx25_init ( int debug_level ); +int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode); +void fx25_rec_bit (int chan, int subchan, int slice, int dbit); +int fx25_rec_busy (int chan); + + +// Other functions in fx25_init.c. + +struct rs *fx25_get_rs (int ctag_num); +uint64_t fx25_get_ctag_value (int ctag_num); +int fx25_get_k_data_radio (int ctag_num); +int fx25_get_k_data_rs (int ctag_num); +int fx25_get_nroots (int ctag_num); +int fx25_get_debug (void); +int fx25_tag_find_match (uint64_t t); +int fx25_pick_mode (int fx_mode, int dlen); + +void fx_hex_dump(unsigned char *x, int len); + + + +#define CTAG_MIN 0x01 +#define CTAG_MAX 0x0B + +// Maximum sizes of "data" and "check" parts. + +#define FX25_MAX_DATA 239 // i.e. RS(255,239) +#define FX25_MAX_CHECK 64 // e.g. RS(255, 191) +#define FX25_BLOCK_SIZE 255 // Block size always 255 for 8 bit symbols. + +#endif // FX25_H \ No newline at end of file diff --git a/src/fx25_auto.c b/src/fx25_auto.c new file mode 100644 index 00000000..c84ecefc --- /dev/null +++ b/src/fx25_auto.c @@ -0,0 +1 @@ +/* */ \ No newline at end of file diff --git a/src/fx25_encode.c b/src/fx25_encode.c new file mode 100644 index 00000000..02c20a84 --- /dev/null +++ b/src/fx25_encode.c @@ -0,0 +1,84 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2019 John Langner, WB2OSZ +// +// 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, see . +// +// ----------------------------------------------------------------------- +// +// +// Most of this is based on: +// +// FX.25 Encoder +// Author: Jim McGuire KB3MPL +// Date: 23 October 2007 +// +// This program is a single-file implementation of the FX.25 encapsulation +// structure for use with AX.25 data packets. Details of the FX.25 +// specification are available at: +// http://www.stensat.org/Docs/Docs.htm +// +// This program implements a single RS(255,239) FEC structure. Future +// releases will incorporate more capabilities as accommodated in the FX.25 +// spec. +// +// The Reed Solomon encoding routines are based on work performed by +// Phil Karn. Phil was kind enough to release his code under the GPL, as +// noted below. Consequently, this FX.25 implementation is also released +// under the terms of the GPL. +// +// Phil Karn's original copyright notice: + /* Test the Reed-Solomon codecs + * for various block sizes and with random data and random error patterns + * + * Copyright 2002 Phil Karn, KA9Q + * May be used under the terms of the GNU General Public License (GPL) + * + */ + + +#include +#include +#include +#include + +#include "fx25.h" + + + +void ENCODE_RS(struct rs * restrict rs, DTYPE * restrict data, DTYPE * restrict bb) +{ + + int i, j; + DTYPE feedback; + + memset(bb,0,NROOTS*sizeof(DTYPE)); // clear out the FEC data area + + for(i=0;i. +// +// ----------------------------------------------------------------------- +// +// This is based on: +// +// +// FX25_extract.c +// Author: Jim McGuire KB3MPL +// Date: 23 October 2007 +// +// +// Accepts an FX.25 byte stream on STDIN, finds the correlation tag, stores 256 bytes, +// corrects errors with FEC, removes the bit-stuffing, and outputs the resultant AX.25 +// byte stream out STDOUT. +// +// stdout prints a bunch of status information about the packet being processed. +// +// +// Usage : FX25_extract < infile > outfile [2> logfile] +// +// +// +// This program is a single-file implementation of the FX.25 extraction/decode +// structure for use with FX.25 data frames. Details of the FX.25 +// specification are available at: +// http://www.stensat.org/Docs/Docs.htm +// +// This program implements a single RS(255,239) FEC structure. Future +// releases will incorporate more capabilities as accommodated in the FX.25 +// spec. +// +// The Reed Solomon encoding routines are based on work performed by +// Phil Karn. Phil was kind enough to release his code under the GPL, as +// noted below. Consequently, this FX.25 implementation is also released +// under the terms of the GPL. +// +// Phil Karn's original copyright notice: + /* Test the Reed-Solomon codecs + * for various block sizes and with random data and random error patterns + * + * Copyright 2002 Phil Karn, KA9Q + * May be used under the terms of the GNU General Public License (GPL) + * + */ + +#include +#include +#include + +#include "fx25.h" + + +//#define DEBUG 5 + +//----------------------------------------------------------------------- +// Revision History +//----------------------------------------------------------------------- +// 0.0.1 - initial release +// Modifications from Phil Karn's GPL source code. +// Initially added code to run with PC file +// I/O and use the (255,239) decoder exclusively. Confirmed that the +// code produces the correct results. +// +//----------------------------------------------------------------------- +// 0.0.2 - + + +#define min(a,b) ((a) < (b) ? (a) : (b)) + + + +int DECODE_RS(struct rs * restrict rs, DTYPE * restrict data, int *eras_pos, int no_eras) { + + int deg_lambda, el, deg_omega; + int i, j, r,k; + DTYPE u,q,tmp,num1,num2,den,discr_r; +// DTYPE lambda[NROOTS+1], s[NROOTS]; /* Err+Eras Locator poly and syndrome poly */ +// DTYPE b[NROOTS+1], t[NROOTS+1], omega[NROOTS+1]; +// DTYPE root[NROOTS], reg[NROOTS+1], loc[NROOTS]; + DTYPE lambda[FX25_MAX_CHECK+1], s[FX25_MAX_CHECK]; /* Err+Eras Locator poly and syndrome poly */ + DTYPE b[FX25_MAX_CHECK+1], t[FX25_MAX_CHECK+1], omega[FX25_MAX_CHECK+1]; + DTYPE root[FX25_MAX_CHECK], reg[FX25_MAX_CHECK+1], loc[FX25_MAX_CHECK]; + int syn_error, count; + + /* form the syndromes; i.e., evaluate data(x) at roots of g(x) */ + for(i=0;i 0) { + /* Init lambda to be the erasure locator polynomial */ + lambda[1] = ALPHA_TO[MODNN(PRIM*(NN-1-eras_pos[0]))]; + for (i = 1; i < no_eras; i++) { + u = MODNN(PRIM*(NN-1-eras_pos[i])); + for (j = i+1; j > 0; j--) { + tmp = INDEX_OF[lambda[j - 1]]; + if(tmp != A0) + lambda[j] ^= ALPHA_TO[MODNN(u + tmp)]; + } + } + +#if DEBUG >= 1 + /* Test code that verifies the erasure locator polynomial just constructed + Needed only for decoder debugging. */ + + /* find roots of the erasure location polynomial */ + for(i=1;i<=no_eras;i++) + reg[i] = INDEX_OF[lambda[i]]; + + count = 0; + for (i = 1,k=IPRIM-1; i <= NN; i++,k = MODNN(k+IPRIM)) { + q = 1; + for (j = 1; j <= no_eras; j++) + if (reg[j] != A0) { + reg[j] = MODNN(reg[j] + j); + q ^= ALPHA_TO[reg[j]]; + } + if (q != 0) + continue; + /* store root and error location number indices */ + root[count] = i; + loc[count] = k; + count++; + } + if (count != no_eras) { + fprintf(stderr,"count = %d no_eras = %d\n lambda(x) is WRONG\n",count,no_eras); + count = -1; + goto finish; + } +#if DEBUG >= 2 + fprintf(stderr,"\n Erasure positions as determined by roots of Eras Loc Poly:\n"); + for (i = 0; i < count; i++) + fprintf(stderr,"%d ", loc[i]); + fprintf(stderr,"\n"); +#endif +#endif + } + for(i=0;i 0; j--){ + if (reg[j] != A0) { + reg[j] = MODNN(reg[j] + j); + q ^= ALPHA_TO[reg[j]]; + } + } + if (q != 0) + continue; /* Not a root */ + /* store root (index-form) and error location number */ +#if DEBUG>=2 + fprintf(stderr,"count %d root %d loc %d\n",count,i,k); +#endif + root[count] = i; + loc[count] = k; + /* If we've already found max possible roots, + * abort the search to save time + */ + if(++count == deg_lambda) + break; + } + if (deg_lambda != count) { + /* + * deg(lambda) unequal to number of roots => uncorrectable + * error detected + */ + count = -1; + goto finish; + } + /* + * Compute err+eras evaluator poly omega(x) = s(x)*lambda(x) (modulo + * x**NROOTS). in index form. Also find deg(omega). + */ + deg_omega = 0; + for (i = 0; i < NROOTS;i++){ + tmp = 0; + j = (deg_lambda < i) ? deg_lambda : i; + for(;j >= 0; j--){ + if ((s[i - j] != A0) && (lambda[j] != A0)) + tmp ^= ALPHA_TO[MODNN(s[i - j] + lambda[j])]; + } + if(tmp != 0) + deg_omega = i; + omega[i] = INDEX_OF[tmp]; + } + omega[NROOTS] = A0; + + /* + * Compute error values in poly-form. num1 = omega(inv(X(l))), num2 = + * inv(X(l))**(FCR-1) and den = lambda_pr(inv(X(l))) all in poly-form + */ + for (j = count-1; j >=0; j--) { + num1 = 0; + for (i = deg_omega; i >= 0; i--) { + if (omega[i] != A0) + num1 ^= ALPHA_TO[MODNN(omega[i] + i * root[j])]; + } + num2 = ALPHA_TO[MODNN(root[j] * (FCR - 1) + NN)]; + den = 0; + + /* lambda[i+1] for i even is the formal derivative lambda_pr of lambda[i] */ + for (i = min(deg_lambda,NROOTS-1) & ~1; i >= 0; i -=2) { + if(lambda[i+1] != A0) + den ^= ALPHA_TO[MODNN(lambda[i+1] + i * root[j])]; + } + if (den == 0) { +#if DEBUG >= 1 + fprintf(stderr,"\n ERROR: denominator = 0\n"); +#endif + count = -1; + goto finish; + } + /* Apply error to data */ + if (num1 != 0) { + data[loc[j]] ^= ALPHA_TO[MODNN(INDEX_OF[num1] + INDEX_OF[num2] + NN - INDEX_OF[den])]; + } + } + finish: + if(eras_pos != NULL){ + for(i=0;i. +// +// ----------------------------------------------------------------------- +// +// +// Some of this is based on: +// +// FX.25 Encoder +// Author: Jim McGuire KB3MPL +// Date: 23 October 2007 +// +// This program is a single-file implementation of the FX.25 encapsulation +// structure for use with AX.25 data packets. Details of the FX.25 +// specification are available at: +// http://www.stensat.org/Docs/Docs.htm +// +// This program implements a single RS(255,239) FEC structure. Future +// releases will incorporate more capabilities as accommodated in the FX.25 +// spec. +// +// The Reed Solomon encoding routines are based on work performed by +// Phil Karn. Phil was kind enough to release his code under the GPL, as +// noted below. Consequently, this FX.25 implementation is also released +// under the terms of the GPL. +// +// Phil Karn's original copyright notice: + /* Test the Reed-Solomon codecs + * for various block sizes and with random data and random error patterns + * + * Copyright 2002 Phil Karn, KA9Q + * May be used under the terms of the GNU General Public License (GPL) + * + */ + +#include "direwolf.h" + +#include +#include +#include +#include +#include // uint64_t +#include // PRIx64 +#include + +#include "fx25.h" +#include "textcolor.h" + + +#define NTAB 3 + +static struct { + int symsize; // Symbol size, bits (1-8). Always 8 for this application. + int genpoly; // Field generator polynomial coefficients. + int fcs; // First root of RS code generator polynomial, index form. + int prim; // Primitive element to generate polynomial roots. + int nroots; // RS code generator polynomial degree (number of roots). + // Same as number of check bytes added. + struct rs *rs; // Pointer to RS codec control block. Filled in at init time. +} Tab[NTAB] = { + {8, 0x11d, 1, 1, 16, NULL }, // RS(255,239) + {8, 0x11d, 1, 1, 32, NULL }, // RS(255,223) + {8, 0x11d, 1, 1, 64, NULL }, // RS(255,191) +}; + +/* + * Reference: http://www.stensat.org/docs/FX-25_01_06.pdf + * FX.25 + * Forward Error Correction Extension to + * AX.25 Link Protocol For Amateur Packet Radio + * Version: 0.01 DRAFT + * Date: 01 September 2006 + */ + +struct correlation_tag_s { + uint64_t value; // 64 bit value, send LSB first. + int n_block_radio; // Size of transmitted block, all in bytes. + int k_data_radio; // Size of transmitted data part. + int n_block_rs; // Size of RS algorithm block. + int k_data_rs; // Size of RS algorithm data part. + int itab; // Index into Tab array. +}; + +static const struct correlation_tag_s tags[16] = { + /* Tag_00 */ { 0x566ED2717946107ELL, 0, 0, 0, 0, -1 }, // Reserved + + /* Tag_01 */ { 0xB74DB7DF8A532F3ELL, 255, 239, 255, 239, 0 }, // RS(255, 239) 16-byte check value, 239 information bytes + /* Tag_02 */ { 0x26FF60A600CC8FDELL, 144, 128, 255, 239, 0 }, // RS(144,128) - shortened RS(255, 239), 128 info bytes + /* Tag_03 */ { 0xC7DC0508F3D9B09ELL, 80, 64, 255, 239, 0 }, // RS(80,64) - shortened RS(255, 239), 64 info bytes + /* Tag_04 */ { 0x8F056EB4369660EELL, 48, 32, 255, 239, 0 }, // RS(48,32) - shortened RS(255, 239), 32 info bytes + + /* Tag_05 */ { 0x6E260B1AC5835FAELL, 255, 223, 255, 223, 1 }, // RS(255, 223) 32-byte check value, 223 information bytes + /* Tag_06 */ { 0xFF94DC634F1CFF4ELL, 160, 128, 255, 223, 1 }, // RS(160,128) - shortened RS(255, 223), 128 info bytes + /* Tag_07 */ { 0x1EB7B9CDBC09C00ELL, 96, 64, 255, 223, 1 }, // RS(96,64) - shortened RS(255, 223), 64 info bytes + /* Tag_08 */ { 0xDBF869BD2DBB1776LL, 64, 32, 255, 223, 1 }, // RS(64,32) - shortened RS(255, 223), 32 info bytes + + /* Tag_09 */ { 0x3ADB0C13DEAE2836LL, 255, 191, 255, 191, 2 }, // RS(255, 191) 64-byte check value, 191 information bytes + /* Tag_0A */ { 0xAB69DB6A543188D6LL, 192, 128, 255, 191, 2 }, // RS(192, 128) - shortened RS(255, 191), 128 info bytes + /* Tag_0B */ { 0x4A4ABEC4A724B796LL, 128, 64, 255, 191, 2 }, // RS(128, 64) - shortened RS(255, 191), 64 info bytes + + /* Tag_0C */ { 0x0293D578626B67E6LL, 0, 0, 0, 0, -1 }, // Undefined + /* Tag_0D */ { 0xE3B0B0D6917E58A6LL, 0, 0, 0, 0, -1 }, // Undefined + /* Tag_0E */ { 0x720267AF1BE1F846LL, 0, 0, 0, 0, -1 }, // Undefined + /* Tag_0F */ { 0x93210201E8F4C706LL, 0, 0, 0, 0, -1 } // Undefined +}; + + + +#define CLOSE_ENOUGH 8 // How many bits can be wrong in tag yet consider it a match? + // Needs to be large enough to match with significant errors + // but not so large to get frequent false matches. + // Probably don't want >= 16 because the hamming distance between + // any two pairs is 32. + // What is a good number? 8?? 12?? 15?? + // 12 got many false matches with random noise. + // Even 8 might be too high. We see 2 or 4 bit errors here + // at the point where decoding the block is very improbable. + // After 2 months of continuous operation as a digipeater/iGate, + // no false triggers were observed. So 8 doesn't seem to be too + // high for 1200 bps. No study has been done for 9600 bps. + +// Given a 64 bit correlation tag value, find acceptable match in table. +// Return index into table or -1 for no match. + +// Both gcc and clang have a built in function to count the number of '1' bits +// in an integer. This can result in a single machine instruction. You might need +// to supply your own popcount function if using a different compiler. + +int fx25_tag_find_match (uint64_t t) +{ + for (int c = CTAG_MIN; c <= CTAG_MAX; c++) { + if (__builtin_popcountll(t ^ tags[c].value) <= CLOSE_ENOUGH) { + //printf ("%016" PRIx64 " received\n", t); + //printf ("%016" PRIx64 " tag %d\n", tags[c].value, c); + //printf ("%016" PRIx64 " xor, popcount = %d\n", t ^ tags[c].value, __builtin_popcountll(t ^ tags[c].value)); + return (c); + } + } + return (-1); +} + + + +void free_rs_char(struct rs *rs){ + free(rs->alpha_to); + free(rs->index_of); + free(rs->genpoly); + free(rs); +} + + +/*------------------------------------------------------------- + * + * Name: fx25_init + * + * Purpose: This must be called once before any of the other fx25 functions. + * + * Inputs: debug_level - Controls level of informational / debug messages. + * + * 0 Only errors. + * 1 (default) Transmitting ctag. Currently no other way to know this. + * 2 Receive correlation tag detected. FEC decode complete. + * 3 Dump data going in and out. + * + * Use command line -dx to increase level or -qx for quiet. + * + * Description: Initialize 3 Reed-Solomon codecs, for 16, 32, and 64 check bytes. + * + *--------------------------------------------------------------*/ + +static int g_debug_level; + +void fx25_init ( int debug_level ) +{ + g_debug_level = debug_level; + + for (int i = 0 ; i < NTAB ; i++) { + Tab[i].rs = INIT_RS(Tab[i].symsize, Tab[i].genpoly, Tab[i].fcs, Tab[i].prim, Tab[i].nroots); + if (Tab[i].rs == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf("FX.25 internal error: init_rs_char failed!\n"); + exit(EXIT_FAILURE); + } + } + + // Verify integrity of tables and assumptions. + // This also does a quick check for the popcount function. + + for (int j = 0; j < 16 ; j++) { + for (int k = 0; k < 16; k++) { + if (j == k) { + assert (__builtin_popcountll(tags[j].value ^ tags[k].value) == 0); + } + else { + assert (__builtin_popcountll(tags[j].value ^ tags[k].value) == 32); + } + } + } + + for (int j = CTAG_MIN; j <= CTAG_MAX; j++) { + assert (tags[j].n_block_radio - tags[j].k_data_radio == Tab[tags[j].itab].nroots); + assert (tags[j].n_block_rs - tags[j].k_data_rs == Tab[tags[j].itab].nroots); + assert (tags[j].n_block_rs == FX25_BLOCK_SIZE); + } + + assert (fx25_pick_mode (100+1, 239) == 1); + assert (fx25_pick_mode (100+1, 240) == -1); + + assert (fx25_pick_mode (100+5, 223) == 5); + assert (fx25_pick_mode (100+5, 224) == -1); + + assert (fx25_pick_mode (100+9, 191) == 9); + assert (fx25_pick_mode (100+9, 192) == -1); + + assert (fx25_pick_mode (16, 32) == 4); + assert (fx25_pick_mode (16, 64) == 3); + assert (fx25_pick_mode (16, 128) == 2); + assert (fx25_pick_mode (16, 239) == 1); + assert (fx25_pick_mode (16, 240) == -1); + + assert (fx25_pick_mode (32, 32) == 8); + assert (fx25_pick_mode (32, 64) == 7); + assert (fx25_pick_mode (32, 128) == 6); + assert (fx25_pick_mode (32, 223) == 5); + assert (fx25_pick_mode (32, 234) == -1); + + assert (fx25_pick_mode (64, 64) == 11); + assert (fx25_pick_mode (64, 128) == 10); + assert (fx25_pick_mode (64, 191) == 9); + assert (fx25_pick_mode (64, 192) == -1); + + assert (fx25_pick_mode (1, 32) == 4); + assert (fx25_pick_mode (1, 33) == 3); + assert (fx25_pick_mode (1, 64) == 3); + assert (fx25_pick_mode (1, 65) == 6); + assert (fx25_pick_mode (1, 128) == 6); + assert (fx25_pick_mode (1, 191) == 9); + assert (fx25_pick_mode (1, 223) == 5); + assert (fx25_pick_mode (1, 239) == 1); + assert (fx25_pick_mode (1, 240) == -1); + +} // fx25_init + + +// Get properties of specified CTAG number. + +struct rs *fx25_get_rs (int ctag_num) +{ + assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); + assert (tags[ctag_num].itab >= 0 && tags[ctag_num].itab < NTAB); + assert (Tab[tags[ctag_num].itab].rs != NULL); + return (Tab[tags[ctag_num].itab].rs); +} + +uint64_t fx25_get_ctag_value (int ctag_num) +{ + assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); + return (tags[ctag_num].value); +} + +int fx25_get_k_data_radio (int ctag_num) +{ + assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); + return (tags[ctag_num].k_data_radio); +} + +int fx25_get_k_data_rs (int ctag_num) +{ + assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); + return (tags[ctag_num].k_data_rs); +} + +int fx25_get_nroots (int ctag_num) +{ + assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); + return (Tab[tags[ctag_num].itab].nroots); +} + +int fx25_get_debug (void) +{ + return (g_debug_level); +} + +/*------------------------------------------------------------- + * + * Name: fx25_pick_mode + * + * Purpose: Pick suitable transmission format based on user preference + * and size of data part required. + * + * Inputs: fx_mode - 0 = none. + * 1 = pick a tag automatically. + * 16, 32, 64 = use this many check bytes. + * 100 + n = use tag n. + * + * 0 and 1 would be the most common. + * Others are mostly for testing. + * + * dlen - Required size for transmitted "data" part, in bytes. + * This includes the AX.25 frame with bit stuffing and a flag + * pattern on each end. + * + * Returns: Correlation tag number in range of CTAG_MIN thru CTAG_MAX. + * -1 is returned for failure. + * The caller should fall back to using plain old AX.25. + * + *--------------------------------------------------------------*/ + +int fx25_pick_mode (int fx_mode, int dlen) +{ + if (fx_mode <= 0) return (-1); + +// Specify a specific tag by adding 100 to the number. +// Fails if data won't fit. + + if (fx_mode - 100 >= CTAG_MIN && fx_mode - 100 <= CTAG_MAX) { + if (dlen <= fx25_get_k_data_radio(fx_mode - 100)) { + return (fx_mode - 100); + } + else { + return (-1); // Assuming caller prints failure message. + } + } + +// Specify number of check bytes. +// Pick the shortest one that can handle the required data length. + + else if (fx_mode == 16 || fx_mode == 32 || fx_mode == 64) { + for (int k = CTAG_MAX; k >= CTAG_MIN; k--) { + if (fx_mode == fx25_get_nroots(k) && dlen <= fx25_get_k_data_radio(k)) { + return (k); + } + } + return (-1); + } + +// For any other number, [[ or if the preference was not possible, ?? ]] +// try to come up with something reasonable. For shorter frames, +// use smaller overhead. For longer frames, where an error is +// more probable, use more check bytes. When the data gets even +// larger, check bytes must be reduced to fit in block size. +// When all else fails, fall back to normal AX.25. +// Some of this is from observing UZ7HO Soundmodem behavior. +// +// Tag Data Check Max Num +// Number Bytes Bytes Repaired +// ------ ----- ----- ----- +// 0x04 32 16 8 +// 0x03 64 16 8 +// 0x06 128 32 16 +// 0x09 191 64 32 +// 0x05 223 32 16 +// 0x01 239 16 8 +// none larger +// +// The PRUG FX.25 TNC has additional modes that will handle larger frames +// by using multiple RS blocks. This is a future possibility but needs +// to be coordinated with other FX.25 developers so we maintain compatibility. +// See https://web.tapr.org/meetings/DCC_2020/JE1WAZ/DCC-2020-PRUG-FINAL.pptx + + static const int prefer[6] = { 0x04, 0x03, 0x06, 0x09, 0x05, 0x01 }; + for (int k = 0; k < 6; k++) { + int m = prefer[k]; + if (dlen <= fx25_get_k_data_radio(m)) { + return (m); + } + } + return (-1); + +// TODO: revisit error messages, produced by caller, when this returns -1. + +} + + +/* Initialize a Reed-Solomon codec + * symsize = symbol size, bits (1-8) - always 8 for this application. + * gfpoly = Field generator polynomial coefficients + * fcr = first root of RS code generator polynomial, index form + * prim = primitive element to generate polynomial roots + * nroots = RS code generator polynomial degree (number of roots) + */ + +struct rs *INIT_RS(unsigned int symsize,unsigned int gfpoly,unsigned fcr,unsigned prim, + unsigned int nroots){ + struct rs *rs; + int i, j, sr,root,iprim; + + if(symsize > 8*sizeof(DTYPE)) + return NULL; /* Need version with ints rather than chars */ + + if(fcr >= (1<= (1<= (1<mm = symsize; + rs->nn = (1<alpha_to = (DTYPE *)calloc((rs->nn+1),sizeof(DTYPE)); + if(rs->alpha_to == NULL){ + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } + rs->index_of = (DTYPE *)calloc((rs->nn+1),sizeof(DTYPE)); + if(rs->index_of == NULL){ + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } + + /* Generate Galois field lookup tables */ + rs->index_of[0] = A0; /* log(zero) = -inf */ + rs->alpha_to[A0] = 0; /* alpha**-inf = 0 */ + sr = 1; + for(i=0;inn;i++){ + rs->index_of[sr] = i; + rs->alpha_to[i] = sr; + sr <<= 1; + if(sr & (1<nn; + } + if(sr != 1){ + /* field generator polynomial is not primitive! */ + free(rs->alpha_to); + free(rs->index_of); + free(rs); + return NULL; + } + + /* Form RS code generator polynomial from its roots */ + rs->genpoly = (DTYPE *)calloc((nroots+1),sizeof(DTYPE)); + if(rs->genpoly == NULL){ + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } + rs->fcr = fcr; + rs->prim = prim; + rs->nroots = nroots; + + /* Find prim-th root of 1, used in decoding */ + for(iprim=1;(iprim % prim) != 0;iprim += rs->nn) + ; + rs->iprim = iprim / prim; + + rs->genpoly[0] = 1; + for (i = 0,root=fcr*prim; i < nroots; i++,root += prim) { + rs->genpoly[i+1] = 1; + + /* Multiply rs->genpoly[] by @**(root + x) */ + for (j = i; j > 0; j--){ + if (rs->genpoly[j] != 0) + rs->genpoly[j] = rs->genpoly[j-1] ^ rs->alpha_to[modnn(rs,rs->index_of[rs->genpoly[j]] + root)]; + else + rs->genpoly[j] = rs->genpoly[j-1]; + } + /* rs->genpoly[0] can never be zero */ + rs->genpoly[0] = rs->alpha_to[modnn(rs,rs->index_of[rs->genpoly[0]] + root)]; + } + /* convert rs->genpoly[] to index form for quicker encoding */ + for (i = 0; i <= nroots; i++) { + rs->genpoly[i] = rs->index_of[rs->genpoly[i]]; + } + +// diagnostic prints +#if 0 + printf("Alpha To:\n\r"); + for (i=0; i < sizeof(DTYPE)*(rs->nn+1); i++) + printf("0x%2x,", rs->alpha_to[i]); + printf("\n\r"); + + printf("Index Of:\n\r"); + for (i=0; i < sizeof(DTYPE)*(rs->nn+1); i++) + printf("0x%2x,", rs->index_of[i]); + printf("\n\r"); + + printf("GenPoly:\n\r"); + for (i = 0; i <= nroots; i++) + printf("0x%2x,", rs->genpoly[i]); + printf("\n\r"); +#endif + return rs; +} + + +// TEMPORARY!!! +// FIXME: We already have multiple copies of this. +// Consolidate them into one somewhere. + +void fx_hex_dump (unsigned char *p, int len) +{ + int n, i, offset; + + offset = 0; + while (len > 0) { + n = len < 16 ? len : 16; + dw_printf (" %03x: ", offset); + for (i=0; i. +// + + + +/******************************************************************************** + * + * File: fx25_rec.c + * + * Purpose: Extract FX.25 codeblocks from a stream of bits and process them. + * + *******************************************************************************/ + +#include "direwolf.h" + +#include +#include +#include +#include +#include +//#if __WIN32__ +//#include +//#endif + +#include "fx25.h" + +#include "fcs_calc.h" +#include "textcolor.h" +#include "multi_modem.h" +#include "demod.h" + +struct fx_context_s { + + enum { FX_TAG=0, FX_DATA, FX_CHECK } state; + uint64_t accum; // Accumulate bits for matching to correlation tag. + int ctag_num; // Correlation tag number, CTAG_MIN to CTAG_MAX if approx. match found. + int k_data_radio; // Expected size of "data" sent over radio. + int coffs; // Starting offset of the check part. + int nroots; // Expected number of check bytes. + int dlen; // Accumulated length in "data" below. + int clen; // Accumulated length in "check" below. + unsigned char imask; // Mask for storing a bit. + unsigned char block[FX25_BLOCK_SIZE+1]; +}; + +static struct fx_context_s *fx_context[MAX_CHANS][MAX_SUBCHANS][MAX_SLICERS]; + +static void process_rs_block (int chan, int subchan, int slice, struct fx_context_s *F); + +static int my_unstuff (int chan, int subchan, int slice, unsigned char * restrict pin, int ilen, unsigned char * restrict frame_buf); + +//#define FXTEST 1 // Define for standalone test application. + // It expects to find files fx01.dat, fx02.dat, ..., fx0b.dat/ + +#if FXTEST +static int fx25_test_count = 0; + +int main () +{ + fx25_init(3); + + for (int i = CTAG_MIN; i <= CTAG_MAX; i++) { + + char fname[32]; + snprintf (fname, sizeof(fname), "fx%02x.dat", i); + FILE *fp = fopen(fname, "rb"); + if (fp == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\n"); + dw_printf ("****** Could not open %s ******\n", fname); + dw_printf ("****** Did you generate the test files first? ******\n"); + exit (EXIT_FAILURE); + } + +//#if 0 // reminder for future if reading from stdin. +//#if __WIN32__ +// // So 0x1a byte does not signal EOF. +// _setmode(_fileno(stdin), _O_BINARY); +//#endif +// fp = stdin; +//#endif + unsigned char ch; + while (fread(&ch, 1, 1, fp) == 1) { + for (unsigned char imask = 0x01; imask != 0; imask <<=1) { + fx25_rec_bit (0, 0, 0, ch & imask); + } + } + fclose (fp); + } + + if (fx25_test_count == 11) { + text_color_set(DW_COLOR_REC); + dw_printf ("\n"); + dw_printf ("\n"); + dw_printf ("\n"); + dw_printf ("***** FX25 unit test Success - all tests passed. *****\n"); + exit (EXIT_SUCCESS); + } + text_color_set(DW_COLOR_ERROR); + dw_printf ("\n"); + dw_printf ("\n"); + dw_printf ("***** FX25 unit test FAILED. Only %d/11 tests passed. *****\n", fx25_test_count); + exit (EXIT_SUCCESS); + +} // end main + +#endif // FXTEST + + + +/*********************************************************************************** + * + * Name: fx25_rec_bit + * + * Purpose: Extract FX.25 codeblocks from a stream of bits. + * In a completely integrated AX.25 / FX.25 receive system, + * this would see the same bit stream as hdlc_rec_bit. + * + * Inputs: chan - Channel number. + * + * subchan - This allows multiple demodulators per channel. + * + * slice - Allows multiple slicers per demodulator (subchannel). + * + * dbit - Data bit after NRZI and any descrambling. + * Any non-zero value is logic '1'. + * + * Description: This is called once for each received bit. + * For each valid frame, process_rec_frame() is called for further processing. + * It can gather multiple candidates from different parallel demodulators + * ("subchannels") and slicers, then decide which one is the best. + * + ***********************************************************************************/ + +#define FENCE 0x55 // to detect buffer overflow. + +void fx25_rec_bit (int chan, int subchan, int slice, int dbit) +{ + +// Allocate context blocks only as needed. + + struct fx_context_s *F = fx_context[chan][subchan][slice]; + if (F == NULL) { + assert (chan >= 0 && chan < MAX_CHANS); + assert (subchan >= 0 && subchan < MAX_SUBCHANS); + assert (slice >= 0 && slice < MAX_SLICERS); + F = fx_context[chan][subchan][slice] = (struct fx_context_s *)malloc(sizeof (struct fx_context_s)); + assert (F != NULL); + memset (F, 0, sizeof(struct fx_context_s)); + } + +// State machine to identify correlation tag then gather appropriate number of data and check bytes. + + switch (F->state) { + case FX_TAG: + F->accum >>= 1; + if (dbit) F->accum |= 1LL << 63; + int c = fx25_tag_find_match (F->accum); + if (c >= CTAG_MIN && c <= CTAG_MAX) { + + F->ctag_num = c; + F->k_data_radio = fx25_get_k_data_radio (F->ctag_num); + F->nroots = fx25_get_nroots (F->ctag_num); + F->coffs = fx25_get_k_data_rs (F->ctag_num); + assert (F->coffs == FX25_BLOCK_SIZE - F->nroots); + + if (fx25_get_debug() >= 2) { + text_color_set(DW_COLOR_INFO); + dw_printf ("FX.25[%d.%d]: Matched correlation tag 0x%02x with %d bit errors. Expecting %d data & %d check bytes.\n", + chan, slice, // ideally subchan too only if applicable + c, + __builtin_popcountll(F->accum ^ fx25_get_ctag_value(c)), + F->k_data_radio, F->nroots); + } + + F->imask = 0x01; + F->dlen = 0; + F->clen = 0; + memset (F->block, 0, sizeof(F->block)); + F->block[FX25_BLOCK_SIZE] = FENCE; + F->state = FX_DATA; + } + break; + + case FX_DATA: + if (dbit) F->block[F->dlen] |= F->imask; + F->imask <<= 1; + if (F->imask == 0) { + F->imask = 0x01; + F->dlen++; + if (F->dlen >= F->k_data_radio) { + F->state = FX_CHECK; + } + } + break; + + case FX_CHECK: + if (dbit) F->block[F->coffs + F->clen] |= F->imask; + F->imask <<= 1; + if (F->imask == 0) { + F->imask = 0x01; + F->clen++; + if (F->clen >= F->nroots) { + + process_rs_block (chan, subchan, slice, F); // see below + + F->ctag_num = -1; + F->accum = 0; + F->state = FX_TAG; + } + } + break; + } +} + + + +/*********************************************************************************** + * + * Name: fx25_rec_busy + * + * Purpose: Is FX.25 reception currently in progress? + * + * Inputs: chan - Channel number. + * + * Returns: True if currently in progress for the specified channel. + * + * Description: This is required for duplicate removal. One channel and can have + * multiple demodulators (called subchannels) running in parallel. + * Each of them can have multiple slicers. Duplicates need to be + * removed. Normally a delay of a couple bits (or more accurately + * symbols) was fine because they all took about the same amount of time. + * Now, we can have an additional delay of up to 64 check bytes and + * some filler in the data portion. We can't simply wait that long. + * With normal AX.25 a couple frames can come and go during that time. + * We want to delay the duplicate removal while FX.25 block reception + * is going on. + * + ***********************************************************************************/ + +int fx25_rec_busy (int chan) +{ + assert (chan >= 0 && chan < MAX_CHANS); + + // This could be a little faster if we knew number of + // subchannels and slicers but it is probably insignificant. + + for (int i = 0; i < MAX_SUBCHANS; i++) { + for (int j = 0; j < MAX_SLICERS; j++) { + if (fx_context[chan][i][j] != NULL) { + if (fx_context[chan][i][j]->state != FX_TAG) { + return (1); + } + } + } + } + return (0); + +} // end fx25_rec_busy + + + +/*********************************************************************************** + * + * Name: process_rs_block + * + * Purpose: After the correlation tag was detected and the appropriate number + * of data and check bytes are accumulated, this performs the processing + * + * Inputs: chan, subchan, slice + * + * F->ctag_num - Correlation tag number (index into table) + * + * F->dlen - Number of "data" bytes. + * + * F->clen - Number of "check" bytes" + * + * F->block - Codeblock. Always 255 total bytes. + * Anything left over after data and check + * bytes is filled with zeros. + * + * <- - - - - - - - - - - 255 bytes total - - - - - - - - -> + * +-----------------------+---------------+---------------+ + * | dlen bytes "data" | zero fill | check bytes | + * +-----------------------+---------------+---------------+ + * + * Description: Use Reed-Solomon decoder to fix up any errors. + * Extract the AX.25 frame from the corrected data. + * + ***********************************************************************************/ + +static void process_rs_block (int chan, int subchan, int slice, struct fx_context_s *F) +{ + if (fx25_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("FX.25[%d.%d]: Received RS codeblock.\n", chan, slice); + fx_hex_dump (F->block, FX25_BLOCK_SIZE); + } + assert (F->block[FX25_BLOCK_SIZE] == FENCE); + + int derrlocs[FX25_MAX_CHECK]; // Half would probably be OK. + struct rs *rs = fx25_get_rs(F->ctag_num); + + int derrors = DECODE_RS(rs, F->block, derrlocs, 0); + + if (derrors >= 0) { // -1 for failure. >= 0 for success, number of bytes corrected. + + if (fx25_get_debug() >= 2) { + text_color_set(DW_COLOR_INFO); + if (derrors == 0) { + dw_printf ("FX.25[%d.%d]: FEC complete with no errors.\n", chan, slice); + } + else { + dw_printf ("FX.25[%d.%d]: FEC complete, fixed %2d errors in byte positions:", chan, slice, derrors); + for (int k = 0; k < derrors; k++) { + dw_printf (" %d", derrlocs[k]); + } + dw_printf ("\n"); + } + } + + unsigned char frame_buf[FX25_MAX_DATA+1]; // Out must be shorter than input. + int frame_len = my_unstuff (chan, subchan, slice, F->block, F->dlen, frame_buf); + + if (frame_len >= 14 + 1 + 2) { // Minimum length: Two addresses & control & FCS. + + unsigned short actual_fcs = frame_buf[frame_len-2] | (frame_buf[frame_len-1] << 8); + unsigned short expected_fcs = fcs_calc (frame_buf, frame_len - 2); + if (actual_fcs == expected_fcs) { + + if (fx25_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("FX.25[%d.%d]: Extracted AX.25 frame:\n", chan, slice); + fx_hex_dump (frame_buf, frame_len); + } + +#if FXTEST + fx25_test_count++; +#else + alevel_t alevel = demod_get_audio_level (chan, subchan); + + multi_modem_process_rec_frame (chan, subchan, slice, frame_buf, frame_len - 2, alevel, derrors, 1); /* len-2 to remove FCS. */ + +#endif + } else { + // Most likely cause is defective sender software. + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d]: Bad FCS for AX.25 frame.\n", chan, slice); + fx_hex_dump (F->block, F->dlen); + fx_hex_dump (frame_buf, frame_len); + } + } + else { + // Most likely cause is defective sender software. + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d]: AX.25 frame is shorter than minimum length.\n", chan, slice); + fx_hex_dump (F->block, F->dlen); + fx_hex_dump (frame_buf, frame_len); + } + } + else if (fx25_get_debug() >= 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d]: FEC failed. Too many errors.\n", chan, slice); + } + +} // process_rs_block + + +/*********************************************************************************** + * + * Name: my_unstuff + * + * Purpose: Remove HDLC it stuffing and surrounding flag delimiters. + * + * Inputs: chan, subchan, slice - For error messages. + * + * pin - "data" part of RS codeblock. + * First byte must be HDLC "flag". + * May be followed by additional flags. + * There must be terminating flag but it might not be byte aligned. + * + * ilen - Number of bytes in pin. + * + * Outputs: frame_buf - Frame contents including FCS. + * Bit stuffing is gone so it should be a whole number of bytes. + * + * Returns: Number of bytes in frame_buf, including 2 for FCS. + * This can never be larger than the max "data" size. + * 0 if any error. + * + * Errors: First byte is not not flag. + * Found seven '1' bits in a row. + * Result is not whole number of bytes after removing bit stuffing. + * Trailing flag not found. + * Most likely cause, for all of these, is defective sender software. + * + ***********************************************************************************/ + +static int my_unstuff (int chan, int subchan, int slice, unsigned char * restrict pin, int ilen, unsigned char * restrict frame_buf) +{ + unsigned char pat_det = 0; // Pattern detector. + unsigned char oacc = 0; // Accumulator for a byte out. + int olen = 0; // Number of good bits in oacc. + int frame_len = 0; // Number of bytes accumulated, including CRC. + + if (*pin != 0x7e) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d] error: Data section did not start with 0x7e.\n", chan, slice); + fx_hex_dump (pin, ilen); + return (0); + } + while (ilen > 0 && *pin == 0x7e) { + ilen--; + pin++; // Skip over leading flag byte(s). + } + + for (int i=0; i>= 1; // Shift the most recent eight bits thru the pattern detector. + pat_det |= dbit << 7; + + if (pat_det == 0xfe) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d]: Invalid AX.25 frame - Seven '1' bits in a row.\n", chan, slice); + fx_hex_dump (pin, ilen); + return 0; + } + + if (dbit) { + oacc >>= 1; + oacc |= 0x80; + } else { + if (pat_det == 0x7e) { // "flag" pattern - End of frame. + if (olen == 7) { + return (frame_len); // Whole number of bytes in result including CRC + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d]: Invalid AX.25 frame - Not a whole number of bytes.\n", chan, slice); + fx_hex_dump (pin, ilen); + return (0); + } + } else if ( (pat_det >> 2) == 0x1f ) { + continue; // Five '1' bits in a row, followed by '0'. Discard the '0'. + } + oacc >>= 1; + } + + olen++; + if (olen & 8) { + olen = 0; + frame_buf[frame_len++] = oacc; + } + } + } /* end of loop on all bits in block */ + + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d.%d]: Invalid AX.25 frame - Terminating flag not found.\n", chan, slice); + fx_hex_dump (pin, ilen); + + return (0); // Should never fall off the end. + +} // my_unstuff + +// end fx25_rec.c \ No newline at end of file diff --git a/src/fx25_send.c b/src/fx25_send.c new file mode 100644 index 00000000..7435be9f --- /dev/null +++ b/src/fx25_send.c @@ -0,0 +1,336 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2019 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "fx25.h" +#include "fcs_calc.h" +#include "textcolor.h" +#include "audio.h" +#include "gen_tone.h" + + +//#define FXTEST 1 // To build unit test application. + + +#ifndef FXTEST +static void send_bytes (int chan, unsigned char *b, int count); +static void send_bit (int chan, int b); +#endif +static int stuff_it (unsigned char *in, int ilen, unsigned char *out, int osize); + + +static int number_of_bits_sent[MAX_CHANS]; // Count number of bits sent by "fx25_send_frame" or "???" + + +#if FXTEST +static unsigned char preload[] = { + 'T'<<1, 'E'<<1, 'S'<<1, 'T'<<1, ' '<<1, ' '<<1, 0x60, + 'W'<<1, 'B'<<1, '2'<<1, 'O'<<1, 'S'<<1, 'Z'<<1, 0x63, + 0x03, 0xf0, + 'F', 'o', 'o', '?' , 'B', 'a', 'r', '?' , // '?' causes bit stuffing + 0, 0, 0 // Room for FCS + extra +}; + +int main () +{ + text_color_set(DW_COLOR_ERROR); + dw_printf("fxsend - FX.25 unit test.\n"); + dw_printf("This generates 11 files named fx01.dat, fx02.dat, ..., fx0b.dat\n"); + dw_printf("Run fxrec as second part of test.\n"); + + fx25_init (3); + for (int i = 100 + CTAG_MIN; i <= 100 + CTAG_MAX; i++) { + fx25_send_frame (0, preload, (int)sizeof(preload)-3, i); + } + exit(EXIT_SUCCESS); +} // end main +#endif + + +/*------------------------------------------------------------- + * + * Name: fx25_send_frame + * + * Purpose: Convert HDLC frames to a stream of bits. + * + * Inputs: chan - Audio channel number, 0 = first. + * + * fbuf - Frame buffer address. + * + * flen - Frame length, before bit-stuffing, not including the FCS. + * + * fx_mode - Normally, this would be 16, 32, or 64 for the desired number + * of check bytes. The shortest format, adequate for the + * required data length will be picked automatically. + * 0x01 thru 0x0b may also be specified for a specific format + * but this is expected to be mostly for testing, not normal + * operation. + * + * Outputs: Bits are shipped out by calling tone_gen_put_bit(). + * + * Returns: Number of bits sent including "flags" and the + * stuffing bits. + * The required time can be calculated by dividing this + * number by the transmit rate of bits/sec. + * -1 is returned for failure. + * + * Description: Generate an AX.25 frame in the usual way then wrap + * it inside of the FX.25 correlation tag and check bytes. + * + * Assumptions: It is assumed that the tone_gen module has been + * properly initialized so that bits sent with + * tone_gen_put_bit() are processed correctly. + * + * Errors: If something goes wrong, return -1 and the caller should + * fallback to sending normal AX.25. + * + * This could happen if the frame is too large. + * + *--------------------------------------------------------------*/ + +int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode) +{ + if (fx25_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("------\n"); + dw_printf ("FX.25[%d] send frame: FX.25 mode = %d\n", chan, fx_mode); + fx_hex_dump (fbuf, flen); + } + + number_of_bits_sent[chan] = 0; + + // Append the FCS. + + int fcs = fcs_calc (fbuf, flen); + fbuf[flen++] = fcs & 0xff; + fbuf[flen++] = (fcs >> 8) & 0xff; + + // Add bit-stuffing. + + unsigned char data[FX25_MAX_DATA+1]; + const unsigned char fence = 0xaa; + data[FX25_MAX_DATA] = fence; + + int dlen = stuff_it(fbuf, flen, data, FX25_MAX_DATA); + + assert (data[FX25_MAX_DATA] == fence); + if (dlen < 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d]: Frame length of %d + overhead is too large to encode.\n", chan, flen); + return (-1); + } + + // Pick suitable correlation tag depending on + // user's preference, for number of check bytes, + // and the data size. + + int ctag_num = fx25_pick_mode (fx_mode, dlen); + + if (ctag_num < CTAG_MIN || ctag_num > CTAG_MAX) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FX.25[%d]: Could not find suitable format for requested %d and data length %d.\n", chan, fx_mode, dlen); + return (-1); + } + + uint64_t ctag_value = fx25_get_ctag_value (ctag_num); + + // Zero out part of data which won't be transmitted. + // It should all be filled by extra HDLC "flag" patterns. + + int k_data_radio = fx25_get_k_data_radio (ctag_num); + int k_data_rs = fx25_get_k_data_rs (ctag_num); + int shorten_by = FX25_MAX_DATA - k_data_radio; + if (shorten_by > 0) { + memset (data + k_data_radio, 0, shorten_by); + } + + // Compute the check bytes. + + unsigned char check[FX25_MAX_CHECK+1]; + check[FX25_MAX_CHECK] = fence; + struct rs *rs = fx25_get_rs (ctag_num); + + assert (k_data_rs + NROOTS == NN); + + ENCODE_RS(rs, data, check); + assert (check[FX25_MAX_CHECK] == fence); + + if (fx25_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("FX.25[%d]: transmit %d data bytes, ctag number 0x%02x\n", chan, k_data_radio, ctag_num); + fx_hex_dump (data, k_data_radio); + dw_printf ("FX.25[%d]: transmit %d check bytes:\n", chan, NROOTS); + fx_hex_dump (check, NROOTS); + dw_printf ("------\n"); + } + +#if FXTEST + // Standalone text application. + + unsigned char flags[16] = { 0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e ,0x7e }; + char fname[32]; + snprintf (fname, sizeof(fname), "fx%02x.dat", ctag_num); + FILE *fp = fopen(fname, "wb"); + fwrite (flags, sizeof(flags), 1, fp); + //fwrite ((unsigned char *)(&ctag_value), sizeof(ctag_value), 1, fp); // No - assumes little endian. + for (int k = 0; k < 8; k++) { + unsigned char b = (ctag_value >> (k * 8)) & 0xff; // Should be portable to big endian too. + fwrite (&b, 1, 1, fp); + } +#if 1 + for (int j = 8; j < 16; j++) { // Introduce errors. + data[j] = ~ data[j]; + } +#endif + fwrite (data, k_data_radio, 1, fp); + fwrite (check, NROOTS, 1, fp); + fwrite (flags, sizeof(flags), 1, fp); + fflush(fp); + fclose (fp); +#else + // Normal usage. Send bits to modulator. + +// Temp hack for testing. Corrupt first 8 bytes. +// for (int j = 0; j < 16; j++) { +// data[j] = ~ data[j]; +// } + + for (int k = 0; k < 8; k++) { + unsigned char b = (ctag_value >> (k * 8)) & 0xff; + send_bytes (chan, &b, 1); + } + send_bytes (chan, data, k_data_radio); + send_bytes (chan, check, NROOTS); +#endif + + return (number_of_bits_sent[chan]); +} + + +#ifndef FXTEST + +static void send_bytes (int chan, unsigned char *b, int count) +{ + for (int j = 0; j < count; j++) { + unsigned char x = b[j]; + for (int k = 0; k < 8; k++) { + send_bit (chan, x & 0x01); + x >>= 1; + } + } +} + +/* + * NRZI encoding. + * data 1 bit -> no change. + * data 0 bit -> invert signal. + */ +static void send_bit (int chan, int b) +{ + static int output[MAX_CHANS]; + + if (b == 0) { + output[chan] = ! output[chan]; + } + tone_gen_put_bit (chan, output[chan]); + number_of_bits_sent[chan]++; +} +#endif // FXTEST + + +/*------------------------------------------------------------- + * + * Name: stuff_it + * + * Purpose: Perform HDLC bit-stuffing and add "flag" octets in + * preparation for the RS encoding. + * + * Inputs: in - Frame, including FCS, in. + * + * ilen - Number of bytes in. + * + * osize - Size of out area. + * + * Outputs: out - Location to receive result. + * + * Returns: Number of bytes needed in output area including one trailing flag. + * -1 if it won't fit. + * + * Description: Convert to stream of bits including: + * start flag + * bit stuffed data, including FCS + * end flag + * Fill remainder with flag octets which might not be on byte boundaries. + * + *--------------------------------------------------------------*/ + +#define put_bit(value) { \ + if (olen >= osize) return(-1); \ + if (value) out[olen>>3] |= 1 << (olen & 0x7); \ + olen++; \ + } + +static int stuff_it (unsigned char *in, int ilen, unsigned char *out, int osize) +{ + const unsigned char flag = 0x7e; + int ret = -1; + memset (out, 0, osize); + out[0] = flag; + int olen = 8; // Number of bits in output. + osize *= 8; // Now in bits rather than bytes. + int ones = 0; + + for (int i = 0; i < ilen; i++) { + for (unsigned char imask = 1; imask != 0; imask <<= 1) { + int v = in[i] & imask; + put_bit(v); + if (v) { + ones++; + if (ones == 5) { + put_bit(0); + ones = 0; + } + } + else { + ones = 0; + } + } + } + for (unsigned char imask = 1; imask != 0; imask <<= 1) { + put_bit(flag & imask); + } + ret = (olen + 7) / 8; // Includes any partial byte. + + unsigned char imask = 1; + while (olen < osize) { + put_bit( flag & imask); + imask = (imask << 1) | (imask >> 7); // Rotate. + } + + return (ret); + +} // end stuff_it + +// end fx25_send.c \ No newline at end of file diff --git a/gen_packets.c b/src/gen_packets.c similarity index 72% rename from gen_packets.c rename to src/gen_packets.c index c2390107..57b2741c 100644 --- a/gen_packets.c +++ b/src/gen_packets.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2013, 2014, 2015, 2016, 2019, 2021, 2023 John Langner, WB2OSZ // // 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 @@ -56,7 +56,15 @@ * gen_packets -n 100 -o z2.wav * atest z2.wav * - * + * Variable speed. e.g. 95% to 105% of normal speed. + * Required parameter is max % below and above normal. + * Optionally specify step other than 0.1%. + * Used to test how tolerant TNCs are to senders not + * not using exactly the right baud rate. + * + * gen_packets -v 5 + * gen_packets -v 5,0.5 + * *------------------------------------------------------------------*/ @@ -67,6 +75,7 @@ #include #include #include +#include #include "audio.h" #include "ax25_pad.h" @@ -75,6 +84,8 @@ #include "textcolor.h" #include "morse.h" #include "dtmf.h" +#include "fx25.h" +#include "il2p.h" /* Own random number generator so we can get */ @@ -84,7 +95,8 @@ static int seed = 1; static int my_rand (void) { - seed = ((seed * 1103515245) + 12345) & MY_RAND_MAX; + // Perform the calculation as unsigned to avoid signed overflow error. + seed = (int)(((unsigned)seed * 1103515245) + 12345) & MY_RAND_MAX; return (seed); } @@ -97,6 +109,7 @@ static float g_noise_level = 0; static int g_morse_wpm = 0; /* Send morse code at this speed. */ + static struct audio_s modem; @@ -105,25 +118,68 @@ static void send_packet (char *str) packet_t pp; unsigned char fbuf[AX25_MAX_PACKET_LEN+2]; int flen; - int c; + int c = 0; // channel number. if (g_morse_wpm > 0) { - // TODO: Why not use the destination field instead of command line option? + // Why not use the destination field instead of command line option? + // For one thing, this is not in TNC-2 monitor format. + + morse_send (c, str, g_morse_wpm, 100, 100); + } + else if (modem.achan[0].modem_type == MODEM_EAS) { + +// Generate EAS SAME signal FOR RESEARCH AND TESTING ONLY!!! +// There could be legal consequences for sending unauhorized SAME +// over the radio so don't do it! - morse_send (0, str, g_morse_wpm, 100, 100); + // I'm expecting to see TNC 2 monitor format. + // The source and destination are ignored. + // The optional destination SSID is the number of times to repeat. + // The user defined data type indicator can optionally be used + // for compatibility with how it is received and presented to client apps. + // Examples: + // X>X-3:{DEZCZC-WXR-RWT-033019-033017-033015-033013-033011-025011-025017-033007-033005-033003-033001-025009-025027-033009+0015-1691525-KGYX/NWS- + // X>X:NNNN + + pp = ax25_from_text (str, 1); + if (pp == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\"%s\" is not valid TNC2 monitoring format.\n", str); + return; + } + unsigned char *pinfo; + int info_len = ax25_get_info (pp, &pinfo); + if (info_len >= 3 && strncmp((char*)pinfo, "{DE", 3) == 0) { + pinfo += 3; + info_len -= 3; + } + + int repeat = ax25_get_ssid (pp, AX25_DESTINATION); + if (repeat == 0) repeat = 1; + + eas_send (c, pinfo, repeat, 500, 500); + ax25_delete (pp); } else { pp = ax25_from_text (str, 1); + if (pp == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\"%s\" is not valid TNC2 monitoring format.\n", str); + return; + } flen = ax25_pack (pp, fbuf); + (void)flen; + + // If stereo, put same thing in each channel. + for (c=0; c MAX_BAUD) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Use a more reasonable bit rate in range of %d - %d.\n", MIN_BAUD, MAX_BAUD); - exit (EXIT_FAILURE); - } /* We have similar logic in direwolf.c, config.c, gen_packets.c, and atest.c, */ /* that need to be kept in sync. Maybe it could be a common function someday. */ - if (modem.achan[0].baud < 600) { + if (modem.achan[0].baud == 100) { // What was this for? + modem.achan[0].modem_type = MODEM_AFSK; + modem.achan[0].mark_freq = 1615; + modem.achan[0].space_freq = 1785; + } + else if (modem.achan[0].baud == 0xEA5EA5) { + modem.achan[0].baud = 521; // Fine tuned later. 520.83333 + // Proper fix is to make this float. + modem.achan[0].modem_type = MODEM_EAS; + modem.achan[0].mark_freq = 2083.3333; // Ideally these should be floating point. + modem.achan[0].space_freq = 1562.5000 ; + } + else if (modem.achan[0].baud < 600) { modem.achan[0].modem_type = MODEM_AFSK; modem.achan[0].mark_freq = 1600; // Typical for HF SSB modem.achan[0].space_freq = 1800; @@ -298,13 +380,26 @@ int main(int argc, char **argv) text_color_set(DW_COLOR_INFO); dw_printf ("Using scrambled baseband signal rather than AFSK.\n"); } + if (modem.achan[0].baud != 100 && (modem.achan[0].baud < MIN_BAUD || modem.achan[0].baud > MAX_BAUD)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Use a more reasonable bit rate in range of %d - %d.\n", MIN_BAUD, MAX_BAUD); + exit (EXIT_FAILURE); + } break; case 'g': /* -g for g3ruh scrambling */ - modem.achan[0].modem_type = MODEM_SCRAMBLE; - text_color_set(DW_COLOR_INFO); - dw_printf ("Using scrambled baseband signal rather than AFSK.\n"); + g_opt = 1; + break; + + case 'j': /* -j V.26 compatible with earlier direwolf. */ + + j_opt = 1; + break; + + case 'J': /* -J V.26 compatible with MFJ-2400. */ + + J_opt = 1; break; case 'm': /* -m for Mark freq */ @@ -334,9 +429,13 @@ int main(int argc, char **argv) case 'n': /* -n number of packets with increasing noise. */ packet_count = atoi(optarg); - g_add_noise = 1; + break; + + case 'N': /* -N number of packets. Don't add noise. */ + packet_count = atoi(optarg); + g_add_noise = 0; break; case 'a': /* -a for amplitude */ @@ -392,7 +491,7 @@ int main(int argc, char **argv) case '2': /* -2 for 2 channels of sound */ modem.adev[0].num_channels = 2; - modem.achan[1].valid = 1; + modem.chan_medium[1] = MEDIUM_RADIO; text_color_set(DW_COLOR_INFO); dw_printf("2 channels of sound rather than 1.\n"); break; @@ -421,9 +520,29 @@ int main(int argc, char **argv) case 'X': - experiment = 1; + X_opt = atoi(optarg); + break; + + case 'I': // IL2P, normal polarity + + I_opt = atoi(optarg); + break; + + case 'i': // IL2P, inverted polarity + + i_opt = atoi(optarg); break; + case 'v': // Variable speed data + an - this percentage + // optional comma and increment. + + variable_speed_max_error = fabs(atof(optarg)); + char *q = strchr(optarg, ','); + if (q != NULL) { + variable_speed_increment = fabs(atof(q+1)); + } + break; + case '?': /* Unknown option message was already printed. */ @@ -434,11 +553,83 @@ int main(int argc, char **argv) /* Should not be here. */ text_color_set(DW_COLOR_ERROR); - dw_printf("?? getopt returned character code 0%o ??\n", c); + dw_printf("?? getopt returned character code 0%o ??\n", (unsigned)c); usage (argv); } } +// These must be processed after -B option. + + if (g_opt) { /* -g for g3ruh scrambling */ + + modem.achan[0].modem_type = MODEM_SCRAMBLE; + text_color_set(DW_COLOR_INFO); + dw_printf ("Using G3RUH mode regardless of bit rate.\n"); + } + + if (j_opt) { /* -j V.26 compatible with earlier direwolf. */ + + modem.achan[0].v26_alternative = V26_A; + modem.achan[0].modem_type = MODEM_QPSK; + modem.achan[0].mark_freq = 0; + modem.achan[0].space_freq = 0; + modem.achan[0].baud = 2400; + } + + if (J_opt) { /* -J V.26 compatible with MFJ-2400. */ + + modem.achan[0].v26_alternative = V26_B; + modem.achan[0].modem_type = MODEM_QPSK; + modem.achan[0].mark_freq = 0; + modem.achan[0].space_freq = 0; + modem.achan[0].baud = 2400; + } + + if (modem.achan[0].modem_type == MODEM_QPSK && + modem.achan[0].v26_alternative == V26_UNSPECIFIED) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR: Either -j or -J must be specified when using 2400 bps QPSK.\n"); + usage (argv); + exit (1); + } + + if (X_opt > 0) { + if (I_opt != -1 || i_opt != -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't mix -X with -I or -i.\n"); + exit (EXIT_FAILURE); + } + modem.achan[0].fx25_strength = X_opt; + modem.achan[0].layer2_xmit = LAYER2_FX25; + } + + if (I_opt != -1 && i_opt != -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't use both -I and -i at the same time.\n"); + exit (EXIT_FAILURE); + } + + if (I_opt >= 0) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Using IL2P normal polarity.\n"); + modem.achan[0].layer2_xmit = LAYER2_IL2P; + modem.achan[0].il2p_max_fec = (I_opt > 0); + modem.achan[0].il2p_invert_polarity = 0; // normal + } + + if (i_opt >= 0) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Using IL2P inverted polarity.\n"); + modem.achan[0].layer2_xmit = LAYER2_IL2P; + modem.achan[0].il2p_max_fec = (i_opt > 0); + modem.achan[0].il2p_invert_polarity = 1; // invert for transmit + if (modem.achan[0].baud == 1200) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Using -i with 1200 bps is a bad idea. Use -I instead.\n"); + } + } + /* * Open the output file. @@ -446,7 +637,7 @@ int main(int argc, char **argv) if (strlen(output_file) == 0) { text_color_set(DW_COLOR_ERROR); - dw_printf ("ERROR: The -o ouput file option must be specified.\n"); + dw_printf ("ERROR: The -o output file option must be specified.\n"); usage (argv); exit (1); } @@ -461,99 +652,21 @@ int main(int argc, char **argv) } - if (experiment) { - modem.achan[0].modem_type = MODEM_QPSK; - modem.achan[0].baud = 2400; // really bps not baud. - amplitude = 100; - } - gen_tone_init (&modem, amplitude/2, 1); morse_init (&modem, amplitude/2); dtmf_init (&modem, amplitude/2); + // We don't have -d or -q options here. + // Just use the default of minimal information. + + fx25_init (1); + il2p_init (0); // There are no "-d" options so far but it could be handy here. assert (modem.adev[0].bits_per_sample == 8 || modem.adev[0].bits_per_sample == 16); assert (modem.adev[0].num_channels == 1 || modem.adev[0].num_channels == 2); assert (modem.adev[0].samples_per_sec >= MIN_SAMPLES_PER_SEC && modem.adev[0].samples_per_sec <= MAX_SAMPLES_PER_SEC); - if (experiment) { - int chan = 0; - int n; - - // 6 cycles of 1800 Hz. - for (n=0; n<8; n++) { - tone_gen_put_bit (chan, 0); - } - - // Shift 90 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 1); - - // Shift 90 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 1); - - // Shift 90 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 1); - - // Shift 90 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 1); - - // Shift 180 - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - - // Shift 270 - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 0); - - // Shift 0 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 0); - - // Shift 0 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 0); - - - // HDLC flag - six 1 in a row. - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 0); - - tone_gen_put_bit (chan, 0); // reverse even/odd position - - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 1); - tone_gen_put_bit (chan, 0); - - tone_gen_put_bit (chan, 0); - - // Shift 0 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 0); - - // Shift 0 - tone_gen_put_bit (chan, 0); - tone_gen_put_bit (chan, 0); - - audio_file_close (); - return (EXIT_SUCCESS); - } - /* * Get user packets(s) from file or stdin if specified. * "-n" option is ignored in this case. @@ -608,9 +721,35 @@ int main(int argc, char **argv) */ text_color_set(DW_COLOR_INFO); dw_printf ("built in message...\n"); - - if (packet_count > 0) { +// +// Generate packets with variable speed. +// This overrides any other number of packets or adding noise. +// + + + if (variable_speed_max_error != 0) { + + int normal_speed = modem.achan[0].baud; + + text_color_set(DW_COLOR_INFO); + dw_printf ("Variable speed.\n"); + + for (double speed_error = - variable_speed_max_error; + speed_error <= variable_speed_max_error + 0.001; + speed_error += variable_speed_increment) { + + // Baud is int so we get some roundoff. Make it real? + modem.achan[0].baud = (int)round(normal_speed * (1. + speed_error / 100.)); + gen_tone_init (&modem, amplitude/2, 1); + + char stemp[256]; + snprintf (stemp, sizeof(stemp), "WB2OSZ-15>TEST:, speed %+0.1f%% The quick brown fox jumps over the lazy dog!", speed_error); + send_packet (stemp); + } + } + + else if (packet_count > 0) { /* * Generate packets with increasing noise level. @@ -619,7 +758,7 @@ int main(int argc, char **argv) */ for (i = 1; i <= packet_count; i++) { - char stemp[80]; + char stemp[88]; if (modem.achan[0].baud < 600) { /* e.g. 300 bps AFSK - About 2/3 should be decoded properly. */ @@ -652,14 +791,23 @@ int main(int argc, char **argv) } else { + // This should send a total of 6. + // Note that sticking in the user defined type {DE is optional. + + if (modem.achan[0].modem_type == MODEM_EAS) { + send_packet ("X>X-3:{DEZCZC-WXR-RWT-033019-033017-033015-033013-033011-025011-025017-033007-033005-033003-033001-025009-025027-033009+0015-1691525-KGYX/NWS-"); + send_packet ("X>X-2:{DENNNN"); + send_packet ("X>X:NNNN"); + } + else { /* * Builtin default 4 packets. */ - - send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 1 of 4"); - send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 2 of 4"); - send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 3 of 4"); - send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 4 of 4"); + send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 1 of 4"); + send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 2 of 4"); + send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 3 of 4"); + send_packet ("WB2OSZ-15>TEST:,The quick brown fox jumps over the lazy dog! 4 of 4"); + } } audio_file_close(); @@ -677,8 +825,13 @@ static void usage (char **argv) dw_printf ("Options:\n"); dw_printf (" -a Signal amplitude in range of 0 - 200%%. Default 50.\n"); dw_printf (" -b Bits / second for data. Default is %d.\n", DEFAULT_BAUD); - dw_printf (" -B Bits / second for data. Proper modem selected for 300, 1200, 2400, 4800, 9600.\n"); + dw_printf (" -B Bits / second for data. Proper modem selected for 300, 1200, 2400, 4800, 9600, EAS.\n"); dw_printf (" -g Scrambled baseband rather than AFSK.\n"); + dw_printf (" -j 2400 bps QPSK compatible with direwolf <= 1.5.\n"); + dw_printf (" -J 2400 bps QPSK compatible with MFJ-2400.\n"); + dw_printf (" -X n 1 to enable FX.25 transmit. 16, 32, 64 for specific number of check bytes.\n"); + dw_printf (" -I n Enable IL2P transmit. n=1 is recommended. 0 uses weaker FEC.\n"); + dw_printf (" -i n Enable IL2P transmit, inverted polarity. n=1 is recommended. 0 uses weaker FEC.\n"); dw_printf (" -m Mark frequency. Default is %d.\n", DEFAULT_MARK_FREQ); dw_printf (" -s Space frequency. Default is %d.\n", DEFAULT_SPACE_FREQ); dw_printf (" -r Audio sample Rate. Default is %d.\n", DEFAULT_SAMPLES_PER_SEC); @@ -686,6 +839,7 @@ static void usage (char **argv) dw_printf (" -o Send output to .wav file.\n"); dw_printf (" -8 8 bit audio rather than 16.\n"); dw_printf (" -2 2 channels (stereo) audio rather than one channel.\n"); + dw_printf (" -v max[,incr] Variable speed with specified maximum error and increment.\n"); // dw_printf (" -z Number of leading zero bits before frame.\n"); // dw_printf (" Default is 12 which is .01 seconds at 1200 bits/sec.\n"); @@ -694,6 +848,7 @@ static void usage (char **argv) dw_printf ("the default built-in message. The format should correspond to\n"); dw_printf ("the standard packet monitoring representation such as,\n\n"); dw_printf (" WB2OSZ-1>APDW12,WIDE2-2:!4237.14NS07120.83W#\n"); + dw_printf ("User defined content can't be used with -n option.\n"); dw_printf ("\n"); dw_printf ("Example: gen_packets -o x.wav \n"); dw_printf ("\n"); @@ -968,4 +1123,4 @@ static int audio_file_close (void) void dcd_change (int chan, int subchan, int slice, int state) { -} \ No newline at end of file +} diff --git a/gen_tone.c b/src/gen_tone.c similarity index 63% rename from gen_tone.c rename to src/gen_tone.c index 0d7c255b..6a816556 100644 --- a/gen_tone.c +++ b/src/gen_tone.c @@ -1,10 +1,7 @@ -//#define DEBUG 1 -//#define DEBUG2 1 - // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2014, 2015, 2016, 2019, 2023 John Langner, WB2OSZ // // 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 @@ -73,6 +70,7 @@ static int ticks_per_sample[MAX_CHANS]; /* Same for both channels of same soundc static int ticks_per_bit[MAX_CHANS]; static int f1_change_per_sample[MAX_CHANS]; static int f2_change_per_sample[MAX_CHANS]; +static float samples_per_symbol[MAX_CHANS]; static short sine_table[256]; @@ -106,59 +104,9 @@ static int bit_count[MAX_CHANS]; // Counter incremented for each bit transmitted static int save_bit[MAX_CHANS]; -/* - * The K9NG/G3RUH output originally took a very simple and lazy approach. - * We simply generated a square wave with + or - the desired amplitude. - * This has a couple undesirable properties. - * - * - Transmitting a square wave would splatter into adjacent - * channels of the transmitter doesn't limit the bandwidth. - * - * - The usual sample rate of 44100 is not a multiple of the - * baud rate so jitter would be added to the zero crossings. - * - * Starting in version 1.2, we try to overcome these issues by using - * a higher sample rate, low pass filtering, and down sampling. - * - * What sort of low pass filter would be appropriate? Intuitively, - * we would expect a cutoff frequency somewhere between baud/2 and baud. - * The current values were found with a small amount of trial and - * error for best results. Future improvement is certainly possible. - */ - -/* - * For low pass filtering of 9600 baud data. - */ - -/* Add sample to buffer and shift the rest down. */ -// TODO: Can we have one copy of these in dsp.h? - -static inline void push_sample (float val, float *buff, int size) -{ - memmove(buff+1,buff,(size-1)*sizeof(float)); - buff[0] = val; -} +static int prev_dat[MAX_CHANS]; // Previous data bit. Used for G3RUH style. -/* FIR filter kernel. */ - -static inline float convolve (const float *data, const float *filter, int filter_size) -{ - float sum = 0; - int j; - - for (j=0; jachan[chan].valid) { + if (audio_config_p->chan_medium[chan] == MEDIUM_RADIO) { int a = ACHAN2ADEV(chan); @@ -251,8 +199,11 @@ int gen_tone_init (struct audio_s *audio_config_p, int amp, int gen_packets) ticks_per_bit[chan] = (int) ((TICKS_PER_CYCLE / ((double)audio_config_p->achan[chan].baud * 0.5)) + 0.5); f1_change_per_sample[chan] = (int) (((double)audio_config_p->achan[chan].mark_freq * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); f2_change_per_sample[chan] = f1_change_per_sample[chan]; // Not used. + samples_per_symbol[chan] = 2. * (float)audio_config_p->adev[a].samples_per_sec / (float)audio_config_p->achan[chan].baud; tone_phase[chan] = PHASE_SHIFT_45; // Just to mimic first attempt. + // ??? Why? We are only concerned with the difference + // from one symbol to the next. break; case MODEM_8PSK: @@ -264,11 +215,33 @@ int gen_tone_init (struct audio_s *audio_config_p, int amp, int gen_packets) ticks_per_bit[chan] = (int) ((TICKS_PER_CYCLE / ((double)audio_config_p->achan[chan].baud / 3.)) + 0.5); f1_change_per_sample[chan] = (int) (((double)audio_config_p->achan[chan].mark_freq * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); f2_change_per_sample[chan] = f1_change_per_sample[chan]; // Not used. + samples_per_symbol[chan] = 3. * (float)audio_config_p->adev[a].samples_per_sec / (float)audio_config_p->achan[chan].baud; + break; + + case MODEM_BASEBAND: + case MODEM_SCRAMBLE: + case MODEM_AIS: + + // Tone is half baud. + ticks_per_bit[chan] = (int) ((TICKS_PER_CYCLE / (double)audio_config_p->achan[chan].baud ) + 0.5); + f1_change_per_sample[chan] = (int) (((double)audio_config_p->achan[chan].baud * 0.5 * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); + samples_per_symbol[chan] = (float)audio_config_p->adev[a].samples_per_sec / (float)audio_config_p->achan[chan].baud; + break; + + case MODEM_EAS: // EAS. + + // TODO: Proper fix would be to use float for baud, mark, space. + + ticks_per_bit[chan] = (int) ((TICKS_PER_CYCLE / 520.833333333333 ) + 0.5); + samples_per_symbol[chan] = (int)((audio_config_p->adev[a].samples_per_sec / 520.83333) + 0.5); + f1_change_per_sample[chan] = (int) ((2083.33333333333 * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); + f2_change_per_sample[chan] = (int) ((1562.5000000 * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); break; - default: + default: // AFSK ticks_per_bit[chan] = (int) ((TICKS_PER_CYCLE / (double)audio_config_p->achan[chan].baud ) + 0.5); + samples_per_symbol[chan] = (float)audio_config_p->adev[a].samples_per_sec / (float)audio_config_p->achan[chan].baud; f1_change_per_sample[chan] = (int) (((double)audio_config_p->achan[chan].mark_freq * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); f2_change_per_sample[chan] = (int) (((double)audio_config_p->achan[chan].space_freq * TICKS_PER_CYCLE / (double)audio_config_p->adev[a].samples_per_sec ) + 0.5); break; @@ -298,75 +271,6 @@ int gen_tone_init (struct audio_s *audio_config_p, int amp, int gen_packets) sine_table[j] = s; } - -/* - * Low pass filter for 9600 baud. - */ - - for (chan = 0; chan < MAX_CHANS; chan++) { - - if (audio_config_p->achan[chan].valid && - (audio_config_p->achan[chan].modem_type == MODEM_SCRAMBLE - || audio_config_p->achan[chan].modem_type == MODEM_BASEBAND)) { - - int a = ACHAN2ADEV(chan); - int samples_per_sec; /* Might be scaled up! */ - int baud; - - /* These numbers were by trial and error. Need more investigation here. */ - - float filter_len_bits = 88 * 9600.0 / (44100.0 * 2.0); - /* Filter length in number of data bits. */ - /* Currently 9.58 */ - - float lpf_baud = 0.8; /* Lowpass cutoff freq as fraction of baud rate */ - - float fc; /* Cutoff frequency as fraction of sampling frequency. */ - -/* - * Normally, we want to generate the same thing whether sending over the air - * or putting it into a file for other testing. - * (There is an important exception. gen_packets can introduce random noise.) - * In this case, we want more aggressive low pass filtering so it looks more like - * what we see coming out of a receiver. - * Specifically, single bits of the same state have considerably reduced amplitude - * below several same values in a row. - */ - - if (gen_packets) { - filter_len_bits = 4; - lpf_baud = 0.55; /* Lowpass cutoff freq as fraction of baud rate */ - } - - samples_per_sec = audio_config_p->adev[a].samples_per_sec * UPSAMPLE; - baud = audio_config_p->achan[chan].baud; - - ticks_per_sample[chan] = (int) ((TICKS_PER_CYCLE / (double)samples_per_sec ) + 0.5); - ticks_per_bit[chan] = (int) ((TICKS_PER_CYCLE / (double)baud ) + 0.5); - - lp_filter_size[chan] = (int) (( filter_len_bits * (float)samples_per_sec / baud) + 0.5); - - if (lp_filter_size[chan] < 10) { - text_color_set(DW_COLOR_DEBUG); - dw_printf ("gen_tone_init: unexpected, chan %d, lp_filter_size %d < 10\n", chan, lp_filter_size[chan]); - lp_filter_size[chan] = 10; - } - else if (lp_filter_size[chan] > MAX_FILTER_SIZE) { - text_color_set(DW_COLOR_DEBUG); - dw_printf ("gen_tone_init: unexpected, chan %d, lp_filter_size %d > %d\n", chan, lp_filter_size[chan], MAX_FILTER_SIZE); - lp_filter_size[chan] = MAX_FILTER_SIZE; - } - - fc = (float)baud * lpf_baud / (float)samples_per_sec; - - //text_color_set(DW_COLOR_DEBUG); - //dw_printf ("gen_tone_init: chan %d, call gen_lowpass(fc=%.2f, , size=%d, )\n", chan, fc, lp_filter_size[chan]); - - gen_lowpass (fc, lp_filter[chan], lp_filter_size[chan], BP_WINDOW_HAMMING); - - } - } - return (0); } /* end gen_tone_init */ @@ -389,19 +293,85 @@ int gen_tone_init (struct audio_s *audio_config_p, int amp, int gen_packets) * * Version 1.4: Attempt to implement 2400 and 4800 bps PSK modes. * + * Version 1.6: For G3RUH, rather than generating square wave and low + * pass filtering, generate the waveform directly. + * This avoids overshoot, ringing, and adding more jitter. + * Alternating bits come out has sine wave of baud/2 Hz. + * + * Version 1.6: MFJ-2400 compatibility for V.26. + * *--------------------------------------------------------------------*/ +// Interpolate between two values. +// My original approximation simply jumped between phases, producing a discontinuity, +// and increasing bandwidth. +// According to multiple sources, we should transition more gently. +// Below see see a rough approximation of: +// * A step function, immediately going to new value. +// * Linear interpoation. +// * Raised cosine. Square root of cosine is also mentioned. +// +// new - / -- +// | / / +// | / | +// | / / +// old ------- / -- +// step linear raised cosine +// +// Inputs are the old (previous value), new value, and a blending control +// 0 -> take old value +// 1 -> take new value. +// inbetween some sort of weighted average. + +static inline float interpol8 (float oldv, float newv, float bc) +{ + // Step function. + //return (newv); // 78 on 11/7 + + assert (bc >= 0); + assert (bc <= 1.1); + + if (bc < 0) return (oldv); + if (bc > 1) return (newv); + + // Linear interpolation, just for comparison. + //return (bc * newv + (1.0f - bc) * oldv); // 39 on 11/7 + + float rc = 0.5f * (cosf(bc * M_PI - M_PI) + 1.0f); + float rrc = bc >= 0.5f + ? 0.5f * (sqrtf(fabsf(cosf(bc * M_PI - M_PI))) + 1.0f) + : 0.5f * (-sqrtf(fabsf(cosf(bc * M_PI - M_PI))) + 1.0f); + + (void)rrc; + return (rc * newv + (1.0f - bc) * oldv); // 49 on 11/7 + //return (rrc * newv + (1.0f - bc) * oldv); // 55 on 11/7 +} + static const int gray2phase_v26[4] = {0, 1, 3, 2}; static const int gray2phase_v27[8] = {1, 0, 2, 3, 6, 7, 5, 4}; +// #define PSKIQ 1 // not ready for prime time yet. +#if PSKIQ +static int xmit_octant[MAX_CHANS]; // absolute phase in 45 degree units. +static int xmit_prev_octant[MAX_CHANS]; // from previous symbol. + +// For PSK, we generate the final signal by combining fixed frequency cosine and +// sine by the following weights. +static const float ci[8] = { 1, .7071, 0, -.7071, -1, -.7071, 0, .7071 }; +static const float sq[8] = { 0, .7071, 1, .7071, 0, -.7071, -1, -.7071 }; +#endif void tone_gen_put_bit (int chan, int dat) { int a = ACHAN2ADEV(chan); /* device for channel. */ assert (save_audio_config_p != NULL); - assert (save_audio_config_p->achan[chan].valid); + if (save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid channel %d for tone generation.\n", chan); + return; + } if (dat < 0) { /* Hack to test receive PLL recovery. */ @@ -409,6 +379,8 @@ void tone_gen_put_bit (int chan, int dat) dat = 0; } +// TODO: change to switch instead of if if if + if (save_audio_config_p->achan[chan].modem_type == MODEM_QPSK) { int dibit; @@ -424,12 +396,28 @@ void tone_gen_put_bit (int chan, int dat) // All zero bits should give us steady 1800 Hz. // All one bits should flip phase by 180 degrees each time. + // For V.26B, add another 45 degrees. + // This seems to work a little better. dibit = (save_bit[chan] << 1) | dat; - symbol = gray2phase_v26[dibit]; + symbol = gray2phase_v26[dibit]; // 0 .. 3 for QPSK. +#if PSKIQ + // One phase shift unit is 45 degrees. + // Remember what it was last time and calculate new. + // values 0 .. 7. + xmit_prev_octant[chan] = xmit_octant[chan]; + xmit_octant[chan] += symbol * 2; + if (save_audio_config_p->achan[chan].v26_alternative == V26_B) { + xmit_octant[chan] += 1; + } + xmit_octant[chan] &= 0x7; +#else tone_phase[chan] += symbol * PHASE_SHIFT_90; - + if (save_audio_config_p->achan[chan].v26_alternative == V26_B) { + tone_phase[chan] += PHASE_SHIFT_45; + } +#endif bit_count[chan]++; } @@ -458,18 +446,22 @@ void tone_gen_put_bit (int chan, int dat) bit_count[chan] = 0; } - if (save_audio_config_p->achan[chan].modem_type == MODEM_SCRAMBLE) { + // Would be logical to have MODEM_BASEBAND for IL2P rather than checking here. But... + // That would mean putting in at least 3 places and testing all rather than just one. + if (save_audio_config_p->achan[chan].modem_type == MODEM_SCRAMBLE && + save_audio_config_p->achan[chan].layer2_xmit != LAYER2_IL2P) { int x; x = (dat ^ (lfsr[chan] >> 16) ^ (lfsr[chan] >> 11)) & 1; lfsr[chan] = (lfsr[chan] << 1) | (x & 1); dat = x; } - +#if PSKIQ + int blend = 1; +#endif do { /* until enough audio samples for this symbol. */ int sam; - float fsam; switch (save_audio_config_p->achan[chan].modem_type) { @@ -479,43 +471,94 @@ void tone_gen_put_bit (int chan, int dat) text_color_set(DW_COLOR_DEBUG); dw_printf ("tone_gen_put_bit %d AFSK\n", __LINE__); #endif - tone_phase[chan] += dat ? f2_change_per_sample[chan] : f1_change_per_sample[chan]; + + // v1.7 reversed. + // Previously a data '1' selected the second (usually higher) tone. + // It never really mattered before because we were using NRZI. + // With the addition of IL2P, we need to be more careful. + // A data '1' should be the mark tone. + + tone_phase[chan] += dat ? f1_change_per_sample[chan] : f2_change_per_sample[chan]; + sam = sine_table[(tone_phase[chan] >> 24) & 0xff]; + gen_tone_put_sample (chan, a, sam); + break; + + case MODEM_EAS: + + tone_phase[chan] += dat ? f1_change_per_sample[chan] : f2_change_per_sample[chan]; sam = sine_table[(tone_phase[chan] >> 24) & 0xff]; gen_tone_put_sample (chan, a, sam); break; case MODEM_QPSK: - case MODEM_8PSK: #if DEBUG2 text_color_set(DW_COLOR_DEBUG); dw_printf ("tone_gen_put_bit %d PSK\n", __LINE__); #endif tone_phase[chan] += f1_change_per_sample[chan]; +#if PSKIQ +#if 1 // blend JWL + // remove loop invariant + float old_i = ci[xmit_prev_octant[chan]]; + float old_q = sq[xmit_prev_octant[chan]]; + + float new_i = ci[xmit_octant[chan]]; + float new_q = sq[xmit_octant[chan]]; + + float b = blend / samples_per_symbol[chan]; // roughly 0 to 1 + blend++; + // b = (b - 0.5) * 20 + 0.5; + // if (b < 0) b = 0; + // if (b > 1) b = 1; + // b = b > 0.5; + //b = 1; // 78 decoded with this. + // only 39 without. + + + //float blended_i = new_i * b + old_i * (1.0f - b); + //float blended_q = new_q * b + old_q * (1.0f - b); + + float blended_i = interpol8 (old_i, new_i, b); + float blended_q = interpol8 (old_q, new_q, b); + + sam = blended_i * sine_table[((tone_phase[chan] - PHASE_SHIFT_90) >> 24) & 0xff] + + blended_q * sine_table[(tone_phase[chan] >> 24) & 0xff]; +#else // jump + sam = ci[xmit_octant[chan]] * sine_table[((tone_phase[chan] - PHASE_SHIFT_90) >> 24) & 0xff] + + sq[xmit_octant[chan]] * sine_table[(tone_phase[chan] >> 24) & 0xff]; +#endif +#else sam = sine_table[(tone_phase[chan] >> 24) & 0xff]; +#endif gen_tone_put_sample (chan, a, sam); break; - case MODEM_BASEBAND: - case MODEM_SCRAMBLE: - + case MODEM_8PSK: #if DEBUG2 text_color_set(DW_COLOR_DEBUG); - dw_printf ("tone_gen_put_bit %d SCR\n", __LINE__); + dw_printf ("tone_gen_put_bit %d PSK\n", __LINE__); #endif - fsam = dat ? amp16bit : (-amp16bit); - - /* version 1.2 - added a low pass filter instead of square wave out. */ - - push_sample (fsam, raw[chan], lp_filter_size[chan]); + tone_phase[chan] += f1_change_per_sample[chan]; + sam = sine_table[(tone_phase[chan] >> 24) & 0xff]; + gen_tone_put_sample (chan, a, sam); + break; - resample[chan]++; - if (resample[chan] >= UPSAMPLE) { + case MODEM_BASEBAND: + case MODEM_SCRAMBLE: + case MODEM_AIS: - sam = (int) convolve (raw[chan], lp_filter[chan], lp_filter_size[chan]); - resample[chan] = 0; - gen_tone_put_sample (chan, a, sam); + if (dat != prev_dat[chan]) { + tone_phase[chan] += f1_change_per_sample[chan]; + } + else { + if (tone_phase[chan] & 0x80000000) + tone_phase[chan] = 0xc0000000; // 270 degrees. + else + tone_phase[chan] = 0x40000000; // 90 degrees. } + sam = sine_table[(tone_phase[chan] >> 24) & 0xff]; + gen_tone_put_sample (chan, a, sam); break; default: @@ -532,7 +575,10 @@ void tone_gen_put_bit (int chan, int dat) } while (bit_len_acc[chan] < ticks_per_bit[chan]); bit_len_acc[chan] -= ticks_per_bit[chan]; -} + + prev_dat[chan] = dat; // Only needed for G3RUH baseband/scrambled. + +} /* end tone_gen_put_bit */ void gen_tone_put_sample (int chan, int a, int sam) { @@ -547,10 +593,21 @@ void gen_tone_put_sample (int chan, int a, int sam) { assert (save_audio_config_p->adev[a].bits_per_sample == 16 || save_audio_config_p->adev[a].bits_per_sample == 8); - // TODO: Should print message telling user to reduce output level. + // Bad news if we are clipping and distorting the signal. + // We are using the full range. + // Too late to change now because everyone would need to recalibrate their + // transmit audio level. - if (sam < -32767) sam = -32767; - else if (sam > 32767) sam = 32767; + if (sam < -32767) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Warning: Audio sample %d clipped to -32767.\n", sam); + sam = -32767; + } + else if (sam > 32767) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Warning: Audio sample %d clipped to +32767.\n", sam); + sam = 32767; + } if (save_audio_config_p->adev[a].num_channels == 1) { @@ -601,6 +658,20 @@ void gen_tone_put_sample (int chan, int a, int sam) { } } +void gen_tone_put_quiet_ms (int chan, int time_ms) { + + int a = ACHAN2ADEV(chan); /* device for channel. */ + int sam = 0; + + int nsamples = (int) ((time_ms * (float)save_audio_config_p->adev[a].samples_per_sec / 1000.) + 0.5); + + for (int j=0; j 30 OVERLAYED CAR + sym_car, // > 30 OVERLAID CAR sym_default, // ? 31 INFO Kiosk (Blue box with ?) - sym_default, // @ 32 HURICANE/Trop-Storm + sym_default, // @ 32 HURRICANE/Trop-Storm sym_default, // A 33 overlayBOX DTMF & RFID & XO sym_default, // B 34 Blwng Snow (& future codes) sym_coast_guard, // C 35 Coast Guard @@ -445,7 +445,7 @@ static const symbol_type_t grm_alternate_symtab[SYMTAB_SIZE] = { sym_default, // G 39 Snow Shwr (& future ovrlys) sym_default, // H 40 Haze (& Overlay Hazards) sym_default, // I 41 Rain Shower - sym_default, // J 42 Lightening (& future ovrlys) + sym_default, // J 42 Lightning (& future ovrlys) sym_rbcn, // K 43 Kenwood HT (W) sym_light, // L 44 Lighthouse sym_default, // M 45 MARS (A=Army,N=Navy,F=AF) @@ -488,12 +488,12 @@ static const symbol_type_t grm_alternate_symtab[SYMTAB_SIZE] = { sym_restrooms, // r 82 Restrooms sym_default, // s 83 OVERLAY SHIP/boat (top view) sym_default, // t 84 Tornado - sym_car, // u 85 OVERLAYED TRUCK - sym_car, // v 86 OVERLAYED Van + sym_car, // u 85 OVERLAID TRUCK + sym_car, // v 86 OVERLAID Van sym_default, // w 87 Flooding sym_wreck, // x 88 Wreck or Obstruction ->X<- sym_default, // y 89 Skywarn - sym_default, // z 90 OVERLAYED Shelter + sym_default, // z 90 OVERLAID Shelter sym_default, // { 91 Fog (& future ovrly codes) sym_default, // | 92 TNC Stream Switch sym_default, // } 93 diff --git a/hdlc_rec.c b/src/hdlc_rec.c similarity index 57% rename from hdlc_rec.c rename to src/hdlc_rec.c index 15b72c2a..d87a1b50 100644 --- a/hdlc_rec.c +++ b/src/hdlc_rec.c @@ -32,7 +32,9 @@ #include #include #include +#include // uint64_t +//#include "tune.h" #include "demod.h" #include "hdlc_rec.h" #include "hdlc_rec2.h" @@ -43,11 +45,12 @@ #include "multi_modem.h" #include "demod_9600.h" /* for descramble() */ #include "ptt.h" +#include "fx25.h" +#include "il2p.h" //#define TEST 1 /* Define for unit testing. */ - //#define DEBUG3 1 /* monitor the data detect signal. */ @@ -100,12 +103,15 @@ struct hdlc_state_s { int frame_len; /* Number of octets in frame_buf. */ /* Should be in range of 0 .. MAX_FRAME_LEN. */ - int data_detect; /* True when HDLC data is detected. */ - /* This will not be triggered by voice or other */ - /* noise or even tones. */ - rrbb_t rrbb; /* Handle for bit array for raw received bits. */ - + + uint64_t eas_acc; /* Accumulate most recent 64 bits received for EAS. */ + + int eas_gathering; /* Decoding in progress. */ + + int eas_plus_found; /* "+" seen, indicating end of geographical area list. */ + + int eas_fields_after_plus; /* Number of "-" characters after the "+". */ }; static struct hdlc_state_s hdlc_state[MAX_CHANS][MAX_SUBCHANS][MAX_SLICERS]; @@ -127,6 +133,9 @@ static int composite_dcd[MAX_CHANS][MAX_SUBCHANS+1]; static int was_init = 0; +static struct audio_s *g_audio_p; + + void hdlc_rec_init (struct audio_s *pa) { int ch, sub, slice; @@ -136,13 +145,14 @@ void hdlc_rec_init (struct audio_s *pa) //dw_printf ("hdlc_rec_init (%p) \n", pa); assert (pa != NULL); - + g_audio_p = pa; + memset (composite_dcd, 0, sizeof(composite_dcd)); for (ch = 0; ch < MAX_CHANS; ch++) { - if (pa->achan[ch].valid) { + if (pa->chan_medium[ch] == MEDIUM_RADIO) { num_subchan[ch] = pa->achan[ch].num_subchan; @@ -168,6 +178,227 @@ void hdlc_rec_init (struct audio_s *pa) was_init = 1; } +/* Own copy of random number generator so we can get */ +/* same predictable results on different operating systems. */ +/* TODO: Consolidate multiple copies somewhere. */ + +#define MY_RAND_MAX 0x7fffffff +static int seed = 1; + +static int my_rand (void) { + // Perform the calculation as unsigned to avoid signed overflow error. + seed = (int)(((unsigned)seed * 1103515245) + 12345) & MY_RAND_MAX; + return (seed); +} + + +/*********************************************************************************** + * + * Name: eas_rec_bit + * + * Purpose: Extract EAS trasmissions from a stream of bits. + * + * Inputs: chan - Channel number. + * + * subchan - This allows multiple demodulators per channel. + * + * slice - Allows multiple slicers per demodulator (subchannel). + * + * raw - One bit from the demodulator. + * should be 0 or 1. + * + * future_use - Not implemented yet. PSK already provides it. + * + * + * Description: This is called once for each received bit. + * For each valid transmission, process_rec_frame() + * is called for further processing. + * + ***********************************************************************************/ + +#define PREAMBLE 0xababababababababULL +#define PREAMBLE_ZCZC 0x435a435aababababULL +#define PREAMBLE_NNNN 0x4e4e4e4eababababULL +#define EAS_MAX_LEN 268 // Not including preamble. Up to 31 geographic areas. + + +static void eas_rec_bit (int chan, int subchan, int slice, int raw, int future_use) +{ + struct hdlc_state_s *H; + +/* + * Different state information for each channel / subchannel / slice. + */ + H = &hdlc_state[chan][subchan][slice]; + + //dw_printf ("slice %d = %d\n", slice, raw); + +// Accumulate most recent 64 bits. + + H->eas_acc >>= 1; + if (raw) { + H->eas_acc |= 0x8000000000000000ULL; + } + + int done = 0; + + if (H->eas_acc == PREAMBLE_ZCZC) { + //dw_printf ("ZCZC\n"); + H->olen = 0; + H->eas_gathering = 1; + H->eas_plus_found = 0; + H->eas_fields_after_plus = 0; + strlcpy ((char*)(H->frame_buf), "ZCZC", sizeof(H->frame_buf)); + H->frame_len = 4; + } + else if (H->eas_acc == PREAMBLE_NNNN) { + //dw_printf ("NNNN\n"); + H->olen = 0; + H->eas_gathering = 1; + strlcpy ((char*)(H->frame_buf), "NNNN", sizeof(H->frame_buf)); + H->frame_len = 4; + done = 1; + } + else if (H->eas_gathering) { + H->olen++; + if (H->olen == 8) { + H->olen = 0; + char ch = H->eas_acc >> 56; + H->frame_buf[H->frame_len++] = ch; + H->frame_buf[H->frame_len] = '\0'; + //dw_printf ("frame_buf = %s\n", H->frame_buf); + + // What characters are acceptable? + // Only ASCII is allowed. i.e. the MSB must be 0. + // The examples show only digits but the geographical area can + // contain anything in range of '!' to DEL or CR or LF. + // There are no restrictions listed for the originator and + // examples contain a slash. + // It's not clear if a space can occur in other places. + + if ( ! (( ch >= ' ' && ch <= 0x7f) || ch == '\r' || ch == '\n')) { +//#define DEBUG_E 1 +#ifdef DEBUG_E + dw_printf ("reject %d invalid character = %s\n", slice, H->frame_buf); +#endif + H->eas_gathering = 0; + return; + } + if (H->frame_len > EAS_MAX_LEN) { // FIXME: look for other places with max length +#ifdef DEBUG_E + dw_printf ("reject %d too long = %s\n", slice, H->frame_buf); +#endif + H->eas_gathering = 0; + return; + } + if (ch == '+') { + H->eas_plus_found = 1; + H->eas_fields_after_plus = 0; + } + if (H->eas_plus_found && ch == '-') { + H->eas_fields_after_plus++; + if (H->eas_fields_after_plus == 3) { + done = 1; // normal case + } + } + } + } + + if (done) { +#ifdef DEBUG_E + dw_printf ("frame_buf %d = %s\n", slice, H->frame_buf); +#endif + alevel_t alevel = demod_get_audio_level (chan, subchan); + multi_modem_process_rec_frame (chan, subchan, slice, H->frame_buf, H->frame_len, alevel, 0, 0); + H->eas_gathering = 0; + } + +} // end eas_rec_bit + + +/* + +EAS has no error detection. +Maybe that doesn't matter because we would normally be dealing with a reasonable +VHF FM or TV signal. +Let's see what happens when we intentionally introduce errors. +When some match and others don't, the multislice voting should give preference +to those matching others. + + $ src/atest -P+ -B EAS -e 3e-3 ../../ref-doc/EAS/same.wav + Demodulator profile set to "+" + 96000 samples per second. 16 bits per sample. 1 audio channels. + 2079360 audio bytes in file. Duration = 10.8 seconds. + Fix Bits level = 0 + Channel 0: 521 baud, AFSK 2083 & 1563 Hz, D+, 96000 sample rate / 3. + +case 1: Slice 6 is different than others (EQS vs. EAS) so we want one of the others that match. + Slice 3 has an unexpected character (in 0120u7) so it is a mismatch. + At this point we are not doing validity checking other than all printable characters. + + We are left with 0 & 4 which don't match (012057 vs. 012077). + So I guess we don't have any two that match so it is a toss up. + + reject 7 invalid character = ZCZC-EAS-RWT-0120▒ + reject 5 invalid character = ZCZC-ECW-RWT-012057-012081-012101-012103-012115+003 + frame_buf 6 = ZCZC-EQS-RWT-012057-012081-012101-012103-012115+0030-2780415-WTSP/TV- + frame_buf 4 = ZCZC-EAS-RWT-012077-012081-012101-012103-012115+0030-2780415-WTSP/TV- + frame_buf 3 = ZCZC-EAS-RWT-0120u7-012281-012101-012103-092115+0038-2780415-VTSP/TV- + frame_buf 0 = ZCZC-EAS-RWT-012057-412081-012101-012103-012115+0030-2780415-WTSP/TV- + + DECODED[1] 0:01.313 EAS audio level = 194(106/108) |__||_|__ + [0.0] EAS>APDW16:{DEZCZC-EAS-RWT-012057-412081-012101-012103-012115+0030-2780415-WTSP/TV- + +Case 2: We have two that match so pick either one. + + reject 5 invalid character = ZCZC-EAS-RW▒ + reject 7 invalid character = ZCZC-EAS-RWT-0 + reject 3 invalid character = ZCZC-EAS-RWT-012057-012080-012101-012103-01211 + reject 0 invalid character = ZCZC-EAS-RWT-012057-012081-012101-012103-012115+0030-2780415-W▒ + frame_buf 6 = ZCZC-EAS-RWT-012057-012081-012!01-012103-012115+0030-2780415-WTSP/TV- + frame_buf 1 = ZCZC-EAS-RWT-012057-012081-012101-012103-012115+0030-2780415-WTSP/TV- + + DECODED[2] 0:03.617 EAS audio level = 194(106/108) _|____|__ + [0.1] EAS>APDW16:{DEZCZC-EAS-RWT-012057-012081-012101-012103-012115+0030-2780415-WTSP/TV- + +Case 3: Slice 6 is a mismatch (EAs vs. EAS). + Slice 7 has RST rather than RWT. + 2 & 4 don't match either (012141 vs. 012101). + We have another case where no two match so there is no clear winner. + + + reject 5 invalid character = ZCZC-EAS-RWT-012057-012081-012101-012103-012115+▒ + frame_buf 7 = ZCZC-EAS-RST-012057-012081-012101-012103-012115+0030-2780415-WTSP/TV- + frame_buf 6 = ZCZC-EAs-RWT-012057-012081-012101-012103-012115+0030-2780415-WTSP/TV- + frame_buf 4 = ZCZC-EAS-RWT-112057-012081-012101-012103-012115+0030-2780415-WTSP/TV- + frame_buf 2 = ZCZC-EAS-RWT-012057-012081-012141-012103-012115+0030-2780415-WTSP/TV- + + DECODED[3] 0:05.920 EAS audio level = 194(106/108) __|_|_||_ + [0.2] EAS>APDW16:{DEZCZC-EAS-RWT-012057-012081-012141-012103-012115+0030-2780415-WTSP/TV- + +Conclusions: + + (1) The existing algorithm gives a higher preference to those frames matching others. + We didn't see any cases here where that would be to our advantage. + + (2) A partial solution would be more validity checking. (i.e. non-digit where + digit is expected.) But wait... We might want to keep it for consideration: + + (3) If I got REALLY ambitious, some day, we could compare all of them one column + at a time and take the most popular (and valid for that column) character and + use all of the most popular characters. Better yet, at the bit level. + +Of course this is probably all overkill because we would normally expect to have pretty +decent signals. The designers didn't even bother to add any sort of checksum for error checking. + +The random errors injected are also not realistic. Actual noise would probably wipe out the +same bit(s) for all of the slices. + +The protocol specification suggests comparing all 3 transmissions and taking the best 2 out of 3. +I think that would best be left to an external application and we just concentrate on being +a good modem here and providing a result when it is received. + +*/ /*********************************************************************************** @@ -188,6 +419,7 @@ void hdlc_rec_init (struct audio_s *pa) * is_scrambled - Is the data scrambled? * * descram_state - Current descrambler state. (not used - remove) + * Not so fast - plans to add new parameter. PSK already provides it. * * * Description: This is called once for each received bit. @@ -196,8 +428,6 @@ void hdlc_rec_init (struct audio_s *pa) * ***********************************************************************************/ -// TODO: int not_used_remove - void hdlc_rec_bit (int chan, int subchan, int slice, int raw, int is_scrambled, int not_used_remove) { @@ -212,11 +442,34 @@ void hdlc_rec_bit (int chan, int subchan, int slice, int raw, int is_scrambled, assert (slice >= 0 && slice < MAX_SLICERS); +// -e option can be used to artificially introduce the desired +// Bit Error Rate (BER) for testing. + + if (g_audio_p->recv_ber != 0) { + double r = (double)my_rand() / (double)MY_RAND_MAX; // calculate as double to preserve all 31 bits. + if (g_audio_p->recv_ber > r) { + +// FIXME +//text_color_set(DW_COLOR_DEBUG); +//dw_printf ("hdlc_rec_bit randomly clobber bit, ber = %.6f\n", g_audio_p->recv_ber); + + raw = ! raw; + } + } + +// EAS does not use HDLC. + + if (g_audio_p->achan[chan].modem_type == MODEM_EAS) { + eas_rec_bit (chan, subchan, slice, raw, not_used_remove); + return; + } + /* * Different state information for each channel / subchannel / slice. */ H = &hdlc_state[chan][subchan][slice]; + /* * Using NRZI encoding, * A '0' bit is represented by an inversion since previous bit. @@ -230,13 +483,23 @@ void hdlc_rec_bit (int chan, int subchan, int slice, int raw, int is_scrambled, dbit = (descram == H->prev_descram); H->prev_descram = descram; - H->prev_raw = raw; } + H->prev_raw = raw; + } else { dbit = (raw == H->prev_raw); + H->prev_raw = raw; } +// After BER insertion, NRZI, and any descrambling, feed into FX.25 decoder as well. +// Don't waste time on this if AIS. EAS does not get this far. + + if (g_audio_p->achan[chan].modem_type != MODEM_AIS) { + fx25_rec_bit (chan, subchan, slice, dbit); + il2p_rec_bit (chan, subchan, slice, raw); // Note: skip NRZI. + } + /* * Octets are sent LSB first. * Shift the most recent 8 bits thru the pattern detector. @@ -251,94 +514,6 @@ void hdlc_rec_bit (int chan, int subchan, int slice, int raw, int is_scrambled, H->flag4_det |= 0x80000000; } - -/* - * "Data Carrier detect" function based on data patterns rather than - * audio signal strength. - * - * Idle time, at beginning of transmission should be filled - * with the special "flag" characters. - * - * Idle time of all zero bits (alternating tones at maximum rate) - * has also been observed rarely. It is easy to understand the reasoning. - * The tones alternate at the maximum rate, making it symmetrical and providing - * the most opportunity for the PLL to lock on to the edges. - * It also violates the published protocol spec. - * - * Recognize zero(s) followed by a single flag even though it violates the spec. - * - * It has been reported that the TinyTrak4 does this. - * https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/messages/1207 - */ - -/* - * Originally, this looked for 4 flags in a row or 3 zeros and a flag. - * Is that too fussy? - * Here are the numbers of start of DCD for our favorite Track 2 test. - * - * 7e7e7e7e 504 7e000000 32 - * 7e7e7e-- 513 7e0000-- 33 - * 7e7e---- 555 7e00---- 42 - * 7e------ 2088 - * - * I don't think we want to look for a single flag because that would - * make DCD too sensitive to noise and it would interfere with waiting for a - * clear channel to transmit. Even a two byte match causes a lot of flickering - * when listening to live signals. Let's try 3 and see how that works out. - */ - - - //if (H->flag4_det == 0x7e7e7e7e) { - if ((H->flag4_det & 0xffffff00) == 0x7e7e7e00) { - //if ((H->flag4_det & 0xffff0000) == 0x7e7e0000) { - - if ( ! H->data_detect) { - H->data_detect = 1; - dcd_change (chan, subchan, slice, 1); - } - } - //else if (H->flag4_det == 0x7e000000) { - else if ((H->flag4_det & 0xffffff00) == 0x7e000000) { - //else if ((H->flag4_det & 0xffff0000) == 0x7e000000) { - - if ( ! H->data_detect) { - H->data_detect = 1; - dcd_change (chan, subchan, slice, 1); - } - } - - -/* - * Loss of signal should result in lack of transitions. - * (all '1' bits) for at least a little while. - * - * When this was written, I was only concerned about 1200 baud. - * For 9600, added later, there is a (de)scrambling function. - * So if there is no change in the signal, we would get pseudo random bits here. - * Maybe we need to put in another check earlier so DCD is not held on too long - * after loss of signal for 9600. - * No, that would not be a good idea. part of a valid frame, when scrambled, - * could have seven or more "1" bits in a row. - * Needs more study. - */ - - - if (H->pat_det == 0xff) { - - if ( H->data_detect ) { - H->data_detect = 0; - dcd_change (chan, subchan, slice, 0); - } - } - - -/* - * End of data carrier detect. - * - * The rest is concerned with framing. - */ - - rrbb_append_bit (H->rrbb, raw); if (H->pat_det == 0x7e) { @@ -391,7 +566,7 @@ void hdlc_rec_bit (int chan, int subchan, int slice, int raw, int is_scrambled, if (actual_fcs == expected_fcs) { alevel_t alevel = demod_get_audio_level (chan, subchan); - multi_modem_process_rec_frame (chan, subchan, slice, H->frame_buf, H->frame_len - 2, alevel, RETRY_NONE); /* len-2 to remove FCS. */ + multi_modem_process_rec_frame (chan, subchan, slice, H->frame_buf, H->frame_len - 2, alevel, RETRY_NONE, 0); /* len-2 to remove FCS. */ } else { @@ -520,47 +695,10 @@ void hdlc_rec_bit (int chan, int subchan, int slice, int raw, int is_scrambled, } } - - -/*------------------------------------------------------------------- - * - * Name: hdlc_rec_gathering - * - * Purpose: Report whether bits are currently being gathered into a frame. - * This is used to influence the PLL inertia. - * The idea is that the PLL should be a little more agreeable to - * synchronize with the incoming data stream when not in a frame - * and resist changing a little more when capturing a frame. - * - * Inputs: chan - * subchan - * slice - * - * Returns: True if we are currently gathering bits. - * In this case we want the PLL to have more inertia. - * - * Discussion: This simply returns the data carrier detect state. - * A couple other variations were tried but turned out to - * be slightly worse. - * - *--------------------------------------------------------------------*/ - -int hdlc_rec_gathering (int chan, int subchan, int slice) -{ - assert (chan >= 0 && chan < MAX_CHANS); - assert (subchan >= 0 && subchan < MAX_SUBCHANS); - assert (slice >= 0 && slice < MAX_SLICERS); - - // Counts from Track 1 & Track 2 - // data_detect 992 988 - // olen>=0 992 985 - // OR-ed 992 985 - - return ( hdlc_state[chan][subchan][slice].data_detect ); - -} /* end hdlc_rec_gathering */ - - +// TODO: Data Carrier Detect (DCD) is now based on DPLL lock +// rather than data patterns found here. +// It would make sense to move the next 2 functions to demod.c +// because this is done at the modem level, rather than HDLC decoder. /*------------------------------------------------------------------- * @@ -623,7 +761,7 @@ void dcd_change (int chan, int subchan, int slice, int state) * * Name: hdlc_rec_data_detect_any * - * Purpose: Determine if the radio channel is curently busy + * Purpose: Determine if the radio channel is currently busy * with packet data. * This version doesn't care about voice or other sounds. * This is used by the transmit logic to transmit only @@ -638,7 +776,7 @@ void dcd_change (int chan, int subchan, int slice, int state) * Description: We have two different versions here. * * hdlc_rec_data_detect_any sees if ANY of the decoders - * for this channel are receving a signal. This is + * for this channel are receiving a signal. This is * used to determine whether the channel is clear and * we can transmit. This would apply to the 300 baud * HF SSB case where we have multiple decoders running diff --git a/hdlc_rec.h b/src/hdlc_rec.h similarity index 100% rename from hdlc_rec.h rename to src/hdlc_rec.h diff --git a/hdlc_rec2.c b/src/hdlc_rec2.c similarity index 89% rename from hdlc_rec2.c rename to src/hdlc_rec2.c index a4764f50..b817018f 100644 --- a/hdlc_rec2.c +++ b/src/hdlc_rec2.c @@ -17,6 +17,7 @@ // + /******************************************************************************** * * File: hdlc_rec2.c @@ -98,13 +99,12 @@ #include "textcolor.h" #include "ax25_pad.h" #include "rrbb.h" -#include "rdq.h" #include "multi_modem.h" #include "dtime_now.h" #include "demod_9600.h" /* for descramble() */ #include "audio.h" /* for struct audio_s */ //#include "ax25_pad.h" /* for AX25_MAX_ADDR_LEN */ - +#include "ais.h" //#define DEBUG 1 //#define DEBUGx 1 @@ -123,6 +123,7 @@ static struct audio_s *save_audio_config_p; #define MAX_FRAME_LEN ((AX25_MAX_PACKET_LEN) + 2) + /* * This is the current state of the HDLC decoder. * @@ -132,7 +133,11 @@ static struct audio_s *save_audio_config_p; * Should have a reset function instead of initializations here. */ -struct hdlc_state_s { +// TODO: Clean up. This is a remnant of splitting hdlc_rec.c into 2 parts. +// This is not the same as hdlc_state_s in hdlc_rec.c +// "2" was added to reduce confusion. Can be trimmed down. + +struct hdlc_state2_s { int prev_raw; /* Keep track of previous bit so */ /* we can look for transitions. */ @@ -268,7 +273,7 @@ void hdlc_rec2_block (rrbb_t block) } /* - * Not successful with frame in orginal form. + * Not successful with frame in original form. * See if we can "fix" it. */ if (try_to_fix_quick_now (block, chan, subchan, slice, alevel)) { @@ -582,7 +587,7 @@ inline static char is_sep_bit_modified(int bit_idx, retry_conf_t retry_conf) { static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t alevel, retry_conf_t retry_conf, int passall) { - struct hdlc_state_s H; + struct hdlc_state2_s H2; int blen; /* Block length in bits. */ int i; int raw; /* From demodulator. Should be 0 or 1. */ @@ -594,10 +599,10 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t int retry_conf_retry = retry_conf.retry; - H.is_scrambled = rrbb_get_is_scrambled (block); - H.prev_descram = rrbb_get_prev_descram (block); - H.lfsr = rrbb_get_descram_state (block); - H.prev_raw = rrbb_get_bit (block, 0); /* Actually last bit of the */ + H2.is_scrambled = rrbb_get_is_scrambled (block); + H2.prev_descram = rrbb_get_prev_descram (block); + H2.lfsr = rrbb_get_descram_state (block); + H2.prev_raw = rrbb_get_bit (block, 0); /* Actually last bit of the */ /* opening flag so we can derive the */ /* first data bit. */ @@ -608,13 +613,13 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t if ((retry_conf.mode == RETRY_MODE_CONTIGUOUS && is_contig_bit_modified(0, retry_conf)) || (retry_conf.mode == RETRY_MODE_SEPARATED && is_sep_bit_modified(0, retry_conf))) { - H.prev_raw = ! H.prev_raw; + H2.prev_raw = ! H2.prev_raw; } - H.pat_det = 0; - H.oacc = 0; - H.olen = 0; - H.frame_len = 0; + H2.pat_det = 0; + H2.oacc = 0; + H2.olen = 0; + H2.frame_len = 0; blen = rrbb_get_len(block); @@ -646,49 +651,49 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t * Octets are sent LSB first. * Shift the most recent 8 bits thru the pattern detector. */ - H.pat_det >>= 1; + H2.pat_det >>= 1; /* * Using NRZI encoding, * A '0' bit is represented by an inversion since previous bit. * A '1' bit is represented by no change. - * Note: this code can be factorized with the raw != H.prev_raw code at the cost of processing time + * Note: this code can be factorized with the raw != H2.prev_raw code at the cost of processing time */ int dbit ; - if (H.is_scrambled) { + if (H2.is_scrambled) { int descram; - descram = descramble(raw, &(H.lfsr)); + descram = descramble(raw, &(H2.lfsr)); - dbit = (descram == H.prev_descram); - H.prev_descram = descram; - H.prev_raw = raw; + dbit = (descram == H2.prev_descram); + H2.prev_descram = descram; + H2.prev_raw = raw; } else { - dbit = (raw == H.prev_raw); - H.prev_raw = raw; + dbit = (raw == H2.prev_raw); + H2.prev_raw = raw; } if (dbit) { - H.pat_det |= 0x80; + H2.pat_det |= 0x80; /* Valid data will never have 7 one bits in a row: exit. */ - if (H.pat_det == 0xfe) { + if (H2.pat_det == 0xfe) { #if DEBUGx text_color_set(DW_COLOR_DEBUG); dw_printf ("try_decode: found abort, i=%d\n", i); #endif return 0; } - H.oacc >>= 1; - H.oacc |= 0x80; + H2.oacc >>= 1; + H2.oacc |= 0x80; } else { /* The special pattern 01111110 indicates beginning and ending of a frame: exit. */ - if (H.pat_det == 0x7e) { + if (H2.pat_det == 0x7e) { #if DEBUGx text_color_set(DW_COLOR_DEBUG); dw_printf ("try_decode: found flag, i=%d\n", i); @@ -703,10 +708,10 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t * "bit stuffing." */ - } else if ( (H.pat_det >> 2) == 0x1f ) { + } else if ( (H2.pat_det >> 2) == 0x1f ) { continue; } - H.oacc >>= 1; + H2.oacc >>= 1; } /* @@ -714,14 +719,14 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t * into the frame buffer. */ - H.olen++; + H2.olen++; - if (H.olen & 8) { - H.olen = 0; + if (H2.olen & 8) { + H2.olen = 0; - if (H.frame_len < MAX_FRAME_LEN) { - H.frame_buf[H.frame_len] = H.oacc; - H.frame_len++; + if (H2.frame_len < MAX_FRAME_LEN) { + H2.frame_buf[H2.frame_len] = H2.oacc; + H2.frame_len++; } } @@ -732,10 +737,10 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t #if DEBUGx text_color_set(DW_COLOR_DEBUG); - dw_printf ("try_decode: olen=%d, frame_len=%d\n", H.olen, H.frame_len); + dw_printf ("try_decode: olen=%d, frame_len=%d\n", H2.olen, H2.frame_len); #endif - if (H.olen == 0 && H.frame_len >= MIN_FRAME_LEN) { + if (H2.olen == 0 && H2.frame_len >= MIN_FRAME_LEN) { unsigned short actual_fcs, expected_fcs; @@ -743,9 +748,9 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t if (retry_conf.type == RETRY_TYPE_NONE) { int j; text_color_set(DW_COLOR_DEBUG); - dw_printf ("NEW WAY: frame len = %d\n", H.frame_len); - for (j=0; jachan[chan].sanity_test)) { + if (actual_fcs == expected_fcs && save_audio_config_p->achan[chan].modem_type == MODEM_AIS) { + + // Sanity check for AIS. + if (ais_check_length((H2.frame_buf[0] >> 2) & 0x3f, H2.frame_len - 2) == 0) { + multi_modem_process_rec_frame (chan, subchan, slice, H2.frame_buf, H2.frame_len - 2, alevel, retry_conf.retry, 0); /* len-2 to remove FCS. */ + return 1; /* success */ + } + else { + return 0; /* did not pass sanity check */ + } + } + else if (actual_fcs == expected_fcs && + sanity_check (H2.frame_buf, H2.frame_len - 2, retry_conf.retry, save_audio_config_p->achan[chan].sanity_test)) { // TODO: Shouldn't be necessary to pass chan, subchan, alevel into // try_decode because we can obtain them from block. @@ -773,7 +789,7 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t assert (rrbb_get_chan(block) == chan); assert (rrbb_get_subchan(block) == subchan); - multi_modem_process_rec_frame (chan, subchan, slice, H.frame_buf, H.frame_len - 2, alevel, retry_conf.retry); /* len-2 to remove FCS. */ + multi_modem_process_rec_frame (chan, subchan, slice, H2.frame_buf, H2.frame_len - 2, alevel, retry_conf.retry, 0); /* len-2 to remove FCS. */ return 1; /* success */ } else if (passall) { @@ -782,7 +798,7 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t //text_color_set(DW_COLOR_ERROR); //dw_printf ("ATTEMPTING PASSALL PROCESSING\n"); - multi_modem_process_rec_frame (chan, subchan, slice, H.frame_buf, H.frame_len - 2, alevel, RETRY_MAX); /* len-2 to remove FCS. */ + multi_modem_process_rec_frame (chan, subchan, slice, H2.frame_buf, H2.frame_len - 2, alevel, RETRY_MAX, 0); /* len-2 to remove FCS. */ return 1; /* success */ } else { @@ -807,25 +823,25 @@ static int try_decode (rrbb_t block, int chan, int subchan, int slice, alevel_t text_color_set(DW_COLOR_ERROR); if (crc_failed) dw_printf ("CRC failed\n"); - if (H.olen != 0) - dw_printf ("Bad olen: %d \n", H.olen); - else if (H.frame_len < MIN_FRAME_LEN) { + if (H2.olen != 0) + dw_printf ("Bad olen: %d \n", H2.olen); + else if (H2.frame_len < MIN_FRAME_LEN) { dw_printf ("Frame too small\n"); goto end; } - dw_printf ("FAILURE with frame: frame len = %d\n", H.frame_len); + dw_printf ("FAILURE with frame: frame len = %d\n", H2.frame_len); dw_printf ("\n"); - for (j=0; j>1); + for (j=0; j>1); } dw_printf ("\nORIG\n"); - for (j=0; j. +// + +#include "direwolf.h" + +#include + +#include "hdlc_send.h" +#include "audio.h" +#include "gen_tone.h" +#include "textcolor.h" +#include "fcs_calc.h" +#include "ax25_pad.h" +#include "fx25.h" +#include "il2p.h" + +static void send_byte_msb_first (int chan, int x, int polarity); + +static void send_control_nrzi (int, int); +static void send_data_nrzi (int, int); +static void send_bit_nrzi (int, int); + + + +static int number_of_bits_sent[MAX_CHANS]; // Count number of bits sent by "hdlc_send_frame" or "hdlc_send_flags" + + + +/*------------------------------------------------------------- + * + * Name: layer2_send_frame + * + * Purpose: Convert frames to a stream of bits. + * Originally this was for AX.25 only, hence the file name. + * Over time, FX.25 and IL2P were shoehorned in. + * + * Inputs: chan - Audio channel number, 0 = first. + * + * pp - Packet object. + * + * bad_fcs - Append an invalid FCS for testing purposes. + * Applies only to regular AX.25. + * + * Outputs: Bits are shipped out by calling tone_gen_put_bit(). + * + * Returns: Number of bits sent including "flags" and the + * stuffing bits. + * The required time can be calculated by dividing this + * number by the transmit rate of bits/sec. + * + * Description: For AX.25, send: + * start flag + * bit stuffed data + * calculated FCS + * end flag + * NRZI encoding for all but the "flags." + * + * + * Assumptions: It is assumed that the tone_gen module has been + * properly initialized so that bits sent with + * tone_gen_put_bit() are processed correctly. + * + *--------------------------------------------------------------*/ + +static int ax25_only_hdlc_send_frame (int chan, unsigned char *fbuf, int flen, int bad_fcs); + + +int layer2_send_frame (int chan, packet_t pp, int bad_fcs, struct audio_s *audio_config_p) +{ + if (audio_config_p->achan[chan].layer2_xmit == LAYER2_IL2P) { + + int n = il2p_send_frame (chan, pp, audio_config_p->achan[chan].il2p_max_fec, + audio_config_p->achan[chan].il2p_invert_polarity); + if (n > 0) { + return (n); + } + text_color_set(DW_COLOR_ERROR); + dw_printf ("Unable to send IL2p frame. Falling back to regular AX.25.\n"); + // Not sure if we should fall back to AX.25 or not here. + } + else if (audio_config_p->achan[chan].layer2_xmit == LAYER2_FX25) { + unsigned char fbuf[AX25_MAX_PACKET_LEN+2]; + int flen = ax25_pack (pp, fbuf); + int n = fx25_send_frame (chan, fbuf, flen, audio_config_p->achan[chan].fx25_strength); + if (n > 0) { + return (n); + } + text_color_set(DW_COLOR_ERROR); + dw_printf ("Unable to send FX.25. Falling back to regular AX.25.\n"); + // Definitely need to fall back to AX.25 here because + // the FX.25 frame length is so limited. + } + + unsigned char fbuf[AX25_MAX_PACKET_LEN+2]; + int flen = ax25_pack (pp, fbuf); + return (ax25_only_hdlc_send_frame (chan, fbuf, flen, bad_fcs)); +} + + + +static int ax25_only_hdlc_send_frame (int chan, unsigned char *fbuf, int flen, int bad_fcs) +{ + int j, fcs; + + + number_of_bits_sent[chan] = 0; + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("hdlc_send_frame ( chan = %d, fbuf = %p, flen = %d, bad_fcs = %d)\n", chan, fbuf, flen, bad_fcs); + fflush (stdout); +#endif + + send_control_nrzi (chan, 0x7e); /* Start frame */ + + for (j=0; j> 8) & 0xff); + } + else { + send_data_nrzi (chan, fcs & 0xff); + send_data_nrzi (chan, (fcs >> 8) & 0xff); + } + + send_control_nrzi (chan, 0x7e); /* End frame */ + + return (number_of_bits_sent[chan]); +} + + +/*------------------------------------------------------------- + * + * Name: layer2_preamble_postamble + * + * Purpose: Send filler pattern before and after the frame. + * For HDLC it is 01111110, for IL2P 01010101. + * + * Inputs: chan - Audio channel number, 0 = first. + * + * nbytes - Number of bytes to send. + * + * finish - True for end of transmission. + * This causes the last audio buffer to be flushed. + * + * audio_config_p - Configuration for audio and modems. + * + * Outputs: Bits are shipped out by calling tone_gen_put_bit(). + * + * Returns: Number of bits sent. + * There is no bit-stuffing so we would expect this to + * be 8 * nbytes. + * The required time can be calculated by dividing this + * number by the transmit rate of bits/sec. + * + * Assumptions: It is assumed that the tone_gen module has been + * properly initialized so that bits sent with + * tone_gen_put_bit() are processed correctly. + * + *--------------------------------------------------------------*/ + +int layer2_preamble_postamble (int chan, int nbytes, int finish, struct audio_s *audio_config_p) +{ + int j; + + number_of_bits_sent[chan] = 0; + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("hdlc_send_flags ( chan = %d, nflags = %d, finish = %d )\n", chan, nflags, finish); + fflush (stdout); +#endif + + // When the transmitter is on but not sending data, it should be sending + // a stream of a filler pattern. + // For AX.25, it is the 01111110 "flag" pattern with NRZI and no bit stuffing. + // For IL2P, it is 01010101 without NRZI. + + for (j=0; jachan[chan].layer2_xmit == LAYER2_IL2P) { + send_byte_msb_first (chan, IL2P_PREAMBLE, audio_config_p->achan[chan].il2p_invert_polarity); + } + else { + send_control_nrzi (chan, 0x7e); + } + } + +/* Push out the final partial buffer! */ + + if (finish) { + audio_flush(ACHAN2ADEV(chan)); + } + + return (number_of_bits_sent[chan]); +} + + + +// The next one is only for IL2P. No NRZI. +// MSB first, opposite of AX.25. + +static void send_byte_msb_first (int chan, int x, int polarity) +{ + int i; + + for (i=0; i<8; i++) { + int dbit = (x & 0x80) != 0; + tone_gen_put_bit (chan, (dbit ^ polarity) & 1); + x <<= 1; + number_of_bits_sent[chan]++; + } +} + + +// The following are only for HDLC. +// All bits are sent NRZI. +// Data (non flags) use bit stuffing. + + +static int stuff[MAX_CHANS]; // Count number of "1" bits to keep track of when we + // need to break up a long run by "bit stuffing." + // Needs to be array because we could be transmitting + // on multiple channels at the same time. + +static void send_control_nrzi (int chan, int x) +{ + int i; + + for (i=0; i<8; i++) { + send_bit_nrzi (chan, x & 1); + x >>= 1; + } + + stuff[chan] = 0; +} + +static void send_data_nrzi (int chan, int x) +{ + int i; + + for (i=0; i<8; i++) { + send_bit_nrzi (chan, x & 1); + if (x & 1) { + stuff[chan]++; + if (stuff[chan] == 5) { + send_bit_nrzi (chan, 0); + stuff[chan] = 0; + } + } else { + stuff[chan] = 0; + } + x >>= 1; + } +} + +/* + * NRZI encoding. + * data 1 bit -> no change. + * data 0 bit -> invert signal. + */ + +static void send_bit_nrzi (int chan, int b) +{ + static int output[MAX_CHANS]; + + if (b == 0) { + output[chan] = ! output[chan]; + } + + tone_gen_put_bit (chan, output[chan]); + + number_of_bits_sent[chan]++; +} + + +// The rest of this is for EAS SAME. +// This is sort of a logical place because it serializes a frame, but not in HDLC. +// We have a parallel where SAME deserialization is in hdlc_rec. +// Maybe both should be pulled out and moved to a same.c. + + +/*------------------------------------------------------------------- + * + * Name: eas_send + * + * Purpose: Serialize EAS SAME for transmission. + * + * Inputs: chan - Radio channel number. + * str - Character string to send. + * repeat - Number of times to repeat with 1 sec quiet between. + * txdelay - Delay (ms) from PTT to first preamble bit. + * txtail - Delay (ms) from last data bit to PTT off. + * + * + * Returns: Total number of milliseconds to activate PTT. + * This includes delays before the first character + * and after the last to avoid chopping off part of it. + * + * Description: xmit_thread calls this instead of the usual hdlc_send + * when we have a special packet that means send EAS SAME + * code. + * + *--------------------------------------------------------------------*/ + +static inline void eas_put_byte (int chan, unsigned char b) +{ + for (int n=0; n<8; n++) { + tone_gen_put_bit (chan, (b & 1)); + b >>= 1; + } +} + +int eas_send (int chan, unsigned char *str, int repeat, int txdelay, int txtail) +{ + int bytes_sent = 0; + const int gap = 1000; + int gaps_sent = 0; + + gen_tone_put_quiet_ms (chan, txdelay); + + for (int r=0; rAPRS:?", strlen("W1ABC>APRS:?"); + send_msg_to_server ("W1ABC>APRS:?", strlen("W1ABC>APRS:?")); } #endif return 0; @@ -319,7 +328,7 @@ static int stats_uplink_packets; /* Number of packets passed along to the IGate /* server after filtering. */ static int stats_uplink_bytes; /* Total number of bytes sent to IGate server */ - /* including login, packets, and hearbeats. */ + /* including login, packets, and heartbeats. */ static int stats_downlink_bytes; /* Total number of bytes from IGate server including */ /* packets, heartbeats, other messages. */ @@ -463,7 +472,7 @@ void igate_init (struct audio_s *p_audio_config, struct igate_config_s *p_igate_ return; } #else - e = pthread_create (&connect_listen_tid, NULL, connnect_thread, (void *)NULL); + e = pthread_create (&connect_listen_tid, NULL, connnect_thread, NULL); if (e != 0) { text_color_set(DW_COLOR_ERROR); perror("Internal error: Could not create IGate connection thread"); @@ -846,6 +855,9 @@ static void * connnect_thread (void *arg) * Purpose: Send a packet to the IGate server * * Inputs: chan - Radio channel it was received on. + * This is required for the RF>IS filtering. + * Beaconing (sendto=ig, chan=-1) and a client app sending + * to ICHANNEL should bypass the filtering. * * recv_pp - Pointer to packet object. * *** CALLER IS RESPONSIBLE FOR DELETING IT! ** @@ -893,7 +905,12 @@ void igate_send_rec_packet (int chan, packet_t recv_pp) * In that case, the payload will have TCPIP in the path and it will be dropped. */ - if (save_digi_config_p->filter_str[chan][MAX_CHANS] != NULL) { +// Apply RF>IS filtering only if it same from a radio channel. +// Beacon will be channel -1. +// Client app to ICHANNEL is outside of radio channel range. + + if (chan >= 0 && chan < MAX_CHANS && // in radio channel range + save_digi_config_p->filter_str[chan][MAX_CHANS] != NULL) { if (pfilter(chan, MAX_CHANS, save_digi_config_p->filter_str[chan][MAX_CHANS], recv_pp, 1) != 1) { @@ -1212,7 +1229,7 @@ static void send_packet_to_server (packet_t pp, int chan) * Name: send_msg_to_server * * Purpose: Send something to the IGate server. - * This one function should be used for login, hearbeats, + * This one function should be used for login, heartbeats, * and packets. * * Inputs: imsg - Message. We will add CR/LF here. @@ -1263,7 +1280,7 @@ static void send_msg_to_server (const char *imsg, int imsg_len) #if __WIN32__ - err = send (igate_sock, stemp, stemp_len, 0); + err = SOCK_SEND (igate_sock, stemp, stemp_len); if (err == SOCKET_ERROR) { text_color_set(DW_COLOR_ERROR); @@ -1274,7 +1291,7 @@ static void send_msg_to_server (const char *imsg, int imsg_len) WSACleanup(); } #else - err = write (igate_sock, stemp, stemp_len); + err = SOCK_SEND (igate_sock, stemp, stemp_len); if (err <= 0) { text_color_set(DW_COLOR_ERROR); @@ -1316,11 +1333,7 @@ static int get1ch (void) // TODO: might read complete packets and unpack from own buffer // rather than using a system call for each byte. -#if __WIN32__ - n = recv (igate_sock, (char*)(&ch), 1, 0); -#else - n = read (igate_sock, &ch, 1); -#endif + n = SOCK_RECV (igate_sock, (char*)(&ch), 1); if (n == 1) { #if DEBUG9 @@ -1468,7 +1481,7 @@ static void * igate_recv_thread (void *arg) * * Future: might have ability to configure multiple transmit * channels, each with own client side filtering and via path. - * Loop here over all configured channels. + * If so, loop here over all configured channels. */ text_color_set(DW_COLOR_REC); dw_printf ("\n[ig>tx] "); // formerly just [ig] @@ -1503,6 +1516,64 @@ static void * igate_recv_thread (void *arg) if (to_chan >= 0) { maybe_xmit_packet_from_igate ((char*)message, to_chan); } + + +/* + * New in 1.7: If ICHANNEL was specified, send packet to client app as specified channel. + */ + if (save_audio_config_p->igate_vchannel >= 0) { + + int ichan = save_audio_config_p->igate_vchannel; + + // My original poorly thoughtout idea was to parse it into a packet object, + // using the non-strict option, and send to the client app. + // + // A lot of things can go wrong with that approach. + + // (1) Up to 8 digipeaters are allowed in radio format. + // There is a potential of finding a larger number here. + // + // (2) The via path can have names that are not valid in the radio format. + // e.g. qAC, T2HAKATA, N5JXS-F1. + // Non-strict parsing would force uppercase, truncate names too long, + // and drop unacceptable SSIDs. + // + // (3) The source address could be invalid for the RF address format. + // e.g. WHO-IS>APJIW4,TCPIP*,qAC,AE5PL-JF::ZL1JSH-9 :Charles Beadfield/New Zealand{583 + // That is essential information that we absolutely need to preserve. + // + // I think the only correct solution is to apply a third party header + // wrapper so the original contents are preserved. This will be a little + // more work for the application developer. Search for ":}" and use only + // the part after that. At this point, I don't see any value in encoding + // information in the source/destination so I will just use "X>X:}" as a prefix + + char stemp[AX25_MAX_INFO_LEN]; + strlcpy (stemp, "X>X:}", sizeof(stemp)); + strlcat (stemp, (char*)message, sizeof(stemp)); + + packet_t pp3 = ax25_from_text(stemp, 0); // 0 means not strict + if (pp3 != NULL) { + + alevel_t alevel; + memset (&alevel, 0, sizeof(alevel)); + alevel.mark = -2; // FIXME: Do we want some other special case? + alevel.space = -2; + + int subchan = -2; // FIXME: -1 is special case for APRStt. + // See what happens with -2 and follow up on this. + // Do we need something else here? + int slice = 0; + fec_type_t fec_type = fec_type_none; + char spectrum[] = "APRS-IS"; + dlq_rec_frame (ichan, subchan, slice, pp3, alevel, fec_type, RETRY_NONE, spectrum); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ICHANNEL %d: Could not parse message from APRS-IS server.\n", ichan); + dw_printf ("%s\n", message); + } + } // end ICHANNEL option } } /* while (1) */ @@ -1537,6 +1608,15 @@ static void * igate_recv_thread (void *arg) * Duplicate removal will drop the original if there is no * corresponding digipeated version. * + * + * This was an idea that came up in one of the discussion forums. + * I rushed in without thinking about it very much. + * + * In retrospect, I don't think this was such a good idea. + * It would be of value only if there is no other IGate nearby + * that would report on the original transmission. + * I wonder if anyone would notice if this silently disappeared. + * *--------------------------------------------------------------------*/ static void satgate_delay_packet (packet_t pp, int chan) @@ -1647,7 +1727,14 @@ static void * satgate_delay_thread (void *arg) * K1RI-2>APWW10,WIDE1-1,WIDE2-1,qAS,K1RI:/221700h/9AmAT3PQ3S,WIDE1-1,WIDE2-1,qAR,W1TG-1:`c)@qh\>/"50}TinyTrak4 Mobile * - * Notice how the final address in the header might not + * This is interesting because the source is not a valid AX.25 address. + * Non-RF stations can have 2 alphanumeric characters for SSID. + * In this example, the WHO-IS server is responding to a message. + * + * WHO-IS>APJIW4,TCPIP*,qAC,AE5PL-JF::ZL1JSH-9 :Charles Beadfield/New Zealand{583 + * + * + * Notice how the final digipeater address, in the header, might not * be a valid AX.25 address. We see a 9 character address * (with no ssid) and an ssid of two letters. * We don't care because we end up discarding them before @@ -1662,29 +1749,61 @@ static void * satgate_delay_thread (void *arg) * *--------------------------------------------------------------------*/ -static void maybe_xmit_packet_from_igate (char *message, int to_chan) + +// It is unforunate that the : data type indicator (DTI) was overloaded with +// so many different meanings. Simply looking at the DTI is not adequate for +// determining whether a packet is a message. +// We need to exclude the other special cases of telemetry metadata, +// bulletins, and weather bulletins. + +static int is_message_message (char *infop) { - packet_t pp3; - char payload[AX25_MAX_PACKET_LEN]; /* what is max len? */ - char src[AX25_MAX_ADDR_LEN]; /* Source address. */ + if (*infop != ':') return (0); + if (strlen(infop) < 11) return (0); // too short for : addressee : + if (strlen(infop) >= 16) { + if (strncmp(infop+10, ":PARM.", 6) == 0) return (0); + if (strncmp(infop+10, ":UNIT.", 6) == 0) return (0); + if (strncmp(infop+10, ":EQNS.", 6) == 0) return (0); + if (strncmp(infop+10, ":BITS.", 6) == 0) return (0); + } + if (strlen(infop) >= 4) { + if (strncmp(infop+1, "BLN", 3) == 0) return (0); + if (strncmp(infop+1, "NWS", 3) == 0) return (0); + if (strncmp(infop+1, "SKY", 3) == 0) return (0); + if (strncmp(infop+1, "CWA", 3) == 0) return (0); + if (strncmp(infop+1, "BOM", 3) == 0) return (0); + } + return (1); // message, including ack, rej +} - char *pinfo = NULL; - int info_len; + +static void maybe_xmit_packet_from_igate (char *message, int to_chan) +{ int n; assert (to_chan >= 0 && to_chan < MAX_CHANS); - /* - * Try to parse it into a packet object. - * This will contain "q constructs" and we might see an address - * with two alphnumeric characters in the SSID so we must use - * the non-strict parsing. + * Try to parse it into a packet object; we need this for the packet filtering. * - * Bug: Up to 8 digipeaters are allowed in radio format. - * There is a potential of finding a larger number here. + * We use the non-strict option because there the via path can have: + * - station names longer than 6. + * - alphanumeric SSID. + * - lower case for "q constructs. + * We don't care about any of those because the via path will be discarded anyhow. + * + * The other issue, that I did not think of originally, is that the "source" + * address might not conform to AX.25 restrictions when it originally came + * from a non-RF source. For example an APRS "message" might be sent to the + * "WHO-IS" server, and the reply message would have that for the source address. + * + * Originally, I used the source address from the packet object but that was + * missing the alphanumeric SSID. This needs to be done differently. + * + * Potential Bug: Up to 8 digipeaters are allowed in radio format. + * Is there a possibility of finding a larger number here? */ - pp3 = ax25_from_text(message, 0); + packet_t pp3 = ax25_from_text(message, 0); if (pp3 == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Tx IGate: Could not parse message from server.\n"); @@ -1692,7 +1811,18 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) return; } - ax25_get_addr_with_ssid (pp3, AX25_SOURCE, src); +// Issue 408: The source address might not be valid AX.25 because it +// came from a non-RF station. e.g. some server responding to a message. +// We need to take source address from original rather than extracting it +// from the packet object. + + char src[AX25_MAX_ADDR_LEN]; /* Source address. */ + memset (src, 0, sizeof(src)); + memcpy (src, message, sizeof(src)-1); + char *gt = strchr(src, '>'); + if (gt != NULL) { + *gt = '\0'; + } /* * Drop if path contains: @@ -1704,8 +1834,8 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) ax25_get_addr_with_ssid (pp3, n + AX25_REPEATER_1, via); - if (strcmp(via, "qAX") == 0 || - strcmp(via, "TCPXX") == 0 || + if (strcmp(via, "qAX") == 0 || // qAX deprecated. http://www.aprs-is.net/q.aspx + strcmp(via, "TCPXX") == 0 || // TCPXX deprecated. strcmp(via, "RFONLY") == 0 || strcmp(via, "NOGATE") == 0) { @@ -1734,14 +1864,26 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) * If we recently transmitted a 'message' from some station, * send the position of the message sender when it comes along later. * + * Some refer to this as a "courtesy posit report" but I don't + * think that is an official term. + * * If we have a position report, look up the sender and see if we should * bypass the normal filtering. + * + * Reference: https://www.aprs-is.net/IGating.aspx + * + * "Passing all message packets also includes passing the sending station's position + * along with the message. When APRS-IS was small, we did this using historical position + * packets. This has become problematic as it introduces historical data on to RF. + * The IGate should note the station(s) it has gated messages to RF for and pass + * the next position packet seen for that station(s) to RF." */ // TODO: Not quite this simple. Should have a function to check for position. // $ raw gps could be a position. @ could be weather data depending on symbol. - info_len = ax25_get_info (pp3, (unsigned char **)(&pinfo)); + char *pinfo = NULL; + int info_len = ax25_get_info (pp3, (unsigned char **)(&pinfo)); int msp_special_case = 0; @@ -1771,12 +1913,6 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) // Previously there was a debug message here about the packet being dropped by filtering. // This is now handled better by the "-df" command line option for filtering details. - // TODO: clean up - remove these lines. - //if (s_debug >= 1) { - // text_color_set(DW_COLOR_INFO); - // dw_printf ("Packet from IGate to channel %d was rejected by filter: %s\n", to_chan, save_digi_config_p->filter_str[MAX_CHANS][to_chan]); - //} - ax25_delete (pp3); return; } @@ -1785,13 +1921,15 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) /* - * Remove the VIA path. + * We want to discard the via path, as received from the APRS-IS, then + * replace it with TCPIP and our own call, marked as used. + * * * For example, we might get something like this from the server. - * K1USN-1>APWW10,TCPIP*,qAC,N5JXS-F1:T#479,100,048,002,500,000,10000000<0x0d><0x0a> + * K1USN-1>APWW10,TCPIP*,qAC,N5JXS-F1:T#479,100,048,002,500,000,10000000 * - * We want to reduce it to this before wrapping it as third party traffic. - * K1USN-1>APWW10:T#479,100,048,002,500,000,10000000<0x0d><0x0a> + * We want to transform it to this before wrapping it as third party traffic. + * K1USN-1>APWW10,TCPIP,mycall*:T#479,100,048,002,500,000,10000000 */ /* @@ -1819,36 +1957,23 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) * * What is the ",I" construct? * Do we care here? - * Is is something new and improved that we should be using in the other direction? + * Is it something new and improved that we should be using in the other direction? */ - while (ax25_get_num_repeaters(pp3) > 0) { - ax25_remove_addr (pp3, AX25_REPEATER_1); - } + char payload[AX25_MAX_PACKET_LEN]; + char dest[AX25_MAX_ADDR_LEN]; /* Destination field. */ + ax25_get_addr_with_ssid (pp3, AX25_DESTINATION, dest); + snprintf (payload, sizeof(payload), "%s>%s,TCPIP,%s*:%s", + src, dest, save_audio_config_p->achan[to_chan].mycall, pinfo); -/* - * Replace the VIA path with TCPIP and my call. - * Mark my call as having been used. - */ - ax25_set_addr (pp3, AX25_REPEATER_1, "TCPIP"); - ax25_set_h (pp3, AX25_REPEATER_1); - ax25_set_addr (pp3, AX25_REPEATER_2, save_audio_config_p->achan[to_chan].mycall); - ax25_set_h (pp3, AX25_REPEATER_2); -/* - * Convert to text representation. - */ - memset (payload, 0, sizeof(payload)); - - ax25_format_addrs (pp3, payload); - info_len = ax25_get_info (pp3, (unsigned char **)(&pinfo)); - (void)(info_len); - strlcat (payload, pinfo, sizeof(payload)); #if DEBUGx text_color_set(DW_COLOR_DEBUG); - dw_printf ("Tx IGate: payload=%s\n", payload); + dw_printf ("Tx IGate: DEBUG payload=%s\n", payload); #endif + + /* * Encapsulate for sending over radio if no reason to drop it. @@ -1864,20 +1989,14 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) * -> Raise the rate limiting value. */ if (ig_to_tx_allow (pp3, to_chan)) { - char radio [500]; - packet_t pradio; - + char radio [2400]; snprintf (radio, sizeof(radio), "%s>%s%d%d%s:}%s", save_audio_config_p->achan[to_chan].mycall, APP_TOCALL, MAJOR_VERSION, MINOR_VERSION, save_igate_config_p->tx_via, payload); - pradio = ax25_from_text (radio, 1); - - /* Oops. Didn't have a check for NULL here. */ - /* Could this be the cause of rare and elusive crashes in 1.2? */ - + packet_t pradio = ax25_from_text (radio, 1); if (pradio != NULL) { #if ITEST @@ -1890,10 +2009,7 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) #endif stats_rf_xmit_packets++; // Any type of packet. - // TEMP TEST: metadata temporarily allowed during testing. - - if (*pinfo == ':' && ! is_telem_metadata(pinfo)) { - // temp test // if (*pinfo == ':') { + if (is_message_message(pinfo)) { // We transmitted a "message." Telemetry metadata is excluded. // Remember to pass along address of the sender later. @@ -1959,9 +2075,37 @@ static void maybe_xmit_packet_from_igate (char *message, int to_chan) * There is a 1 / 65536 chance of getting a false positive match * which is good enough for this application. * + * + * Original thinking: + * + * Occasionally someone will get on one of the discussion groups and say: + * I don't think my IGate is working. I look at packets, from local stations, + * on aprs.fi or findu.com, and they are always through some other IGate station, + * never mine. + * Then someone has to explain, this is not a valid strategy for analyzing + * everything going thru the network. The APRS-IS servers drop duplicate + * packets (ignoring the via path) within a 30 second period. If some + * other IGate gets the same thing there a millisecond faster than you, + * the one you send is discarded. + * In this scenario, it would make sense to perform additional duplicate + * suppression before forwarding RF packets to the Server. + * I don't recall if I saw some specific recommendation to do this or if + * it just seemed like the obvious thing to do to avoid sending useless + * stuff that would just be discarded anyhow. It seems others came to the + * same conclusion. http://www.tapr.org/pipermail/aprssig/2016-July/045907.html + * + * Version 1.5: Rethink strategy. + * + * Issue 85, https://github.com/wb2osz/direwolf/issues/85 , + * got me thinking about this some more. Sending more information will + * allow the APRS-IS servers to perform future additional network analysis. + * To make a long story short, the RF>IS direction duplicate checking + * is now disabled. The code is still there in case I change my mind + * and want to add a configuration option to allow it. The dedupe + * time is set to 0 which means don't do the checking. + * *--------------------------------------------------------------------*/ -#define RX2IG_DEDUPE_TIME 60 /* Do not send duplicate within 60 seconds. */ #define RX2IG_HISTORY_MAX 30 /* Remember the last 30 sent to IGate server. */ static int rx2ig_insert_next; @@ -1982,6 +2126,12 @@ static void rx_to_ig_init (void) static void rx_to_ig_remember (packet_t pp) { +// No need to save the information if we are not doing duplicate checking. + + if (save_igate_config_p->rx2ig_dedupe_time == 0) { + return; + } + rx2ig_time_stamp[rx2ig_insert_next] = time(NULL); rx2ig_checksum[rx2ig_insert_next] = ax25_dedupe_crc(pp); @@ -2031,8 +2181,21 @@ static int rx_to_ig_allow (packet_t pp) dw_printf ("rx_to_ig_allow? %d \"%s>%s:%s\"\n", crc, src, dest, pinfo); } + +// Do we have duplicate checking at all in the RF>IS direction? + + if (save_igate_config_p->rx2ig_dedupe_time == 0) { + if (s_debug >= 2) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("rx_to_ig_allow? YES, no dedupe checking\n"); + } + return 1; + } + +// Yes, check for duplicates within certain time. + for (j=0; j= now - RX2IG_DEDUPE_TIME) { + if (rx2ig_checksum[j] == crc && rx2ig_time_stamp[j] >= now - save_igate_config_p->rx2ig_dedupe_time) { if (s_debug >= 2) { text_color_set(DW_COLOR_DEBUG); // could be multiple entries and this might not be the most recent. @@ -2159,7 +2322,7 @@ static int rx_to_ig_allow (packet_t pp) * * Future: * Should the digipeater function avoid transmitting something if it - * was recently transmitted by the IGate funtion? + * was recently transmitted by the IGate function? * This code is pretty much the same as dedupe.c. Maybe it could all * be combined into one. Need to ponder this some more. * @@ -2304,6 +2467,8 @@ void ig_to_tx_remember (packet_t pp, int chan, int bydigi) } } + + static int ig_to_tx_allow (packet_t pp, int chan) { unsigned short crc = ax25_dedupe_crc(pp); @@ -2336,7 +2501,7 @@ static int ig_to_tx_allow (packet_t pp, int chan) /* We have a duplicate within some time period. */ - if (*pinfo == ':' && ! is_telem_metadata((char*)pinfo)) { + if (is_message_message((char*)pinfo)) { /* I think I want to avoid the duplicate suppression for "messages." */ /* Suppose we transmit a message from station X and it doesn't get an ack back. */ @@ -2385,7 +2550,7 @@ static int ig_to_tx_allow (packet_t pp, int chan) /* the normal limit for them. */ increase_limit = 1; - if (*pinfo == ':' && ! is_telem_metadata((char*)pinfo)) { + if (is_message_message((char*)pinfo)) { increase_limit = 3; } diff --git a/igate.h b/src/igate.h similarity index 86% rename from igate.h rename to src/igate.h index b98e5a9a..8203ac70 100644 --- a/igate.h +++ b/src/igate.h @@ -39,6 +39,9 @@ struct igate_config_s { char t2_passcode[8]; /* Max. 5 digits. Could be "-1". */ char *t2_filter; /* Optional filter for IS -> RF direction. */ + /* This is the "server side" filter. */ + /* A better name would be subscription or something */ + /* like that because we can only ask for more. */ /* * For transmitting. @@ -69,6 +72,11 @@ struct igate_config_s { /* We allow additional flexibility of 0 to disable feature */ /* or a small number to allow more. */ +/* + * Receiver to IS data options. + */ + int rx2ig_dedupe_time; /* seconds. 0 to disable. */ + /* * Special SATgate mode to delay packets heard directly. */ @@ -82,6 +90,10 @@ struct igate_config_s { #define IGATE_TX_LIMIT_5_DEFAULT 20 #define IGATE_TX_LIMIT_5_MAX 80 +#define IGATE_RX2IG_DEDUPE_TIME 0 /* Issue 85. 0 means disable dupe checking in RF>IS direction. */ + /* See comments in rx_to_ig_remember & rx_to_ig_allow. */ + /* Currently there is no configuration setting to change this. */ + #define DEFAULT_SATGATE_DELAY 10 #define MIN_SATGATE_DELAY 5 #define MAX_SATGATE_DELAY 30 diff --git a/src/il2p.h b/src/il2p.h new file mode 100644 index 00000000..d18e2d05 --- /dev/null +++ b/src/il2p.h @@ -0,0 +1,145 @@ + + +#ifndef IL2P_H +#define IL2P_H 1 + + +#define IL2P_PREAMBLE 0x55 + +#define IL2P_SYNC_WORD 0xF15E48 + +#define IL2P_SYNC_WORD_SIZE 3 +#define IL2P_HEADER_SIZE 13 // Does not include 2 parity. +#define IL2P_HEADER_PARITY 2 + +#define IL2P_MAX_PAYLOAD_SIZE 1023 +#define IL2P_MAX_PAYLOAD_BLOCKS 5 +#define IL2P_MAX_PARITY_SYMBOLS 16 // For payload only. +#define IL2P_MAX_ENCODED_PAYLOAD_SIZE (IL2P_MAX_PAYLOAD_SIZE + IL2P_MAX_PAYLOAD_BLOCKS * IL2P_MAX_PARITY_SYMBOLS) + +#define IL2P_MAX_PACKET_SIZE (IL2P_SYNC_WORD_SIZE + IL2P_HEADER_SIZE + IL2P_HEADER_PARITY + IL2P_MAX_ENCODED_PAYLOAD_SIZE) + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_init.c +// +/////////////////////////////////////////////////////////////////////////////// + + +// Init must be called at start of application. + +extern void il2p_init (int debug); + +#include "fx25.h" // For Reed Solomon stuff. e.g. struct rs + // Maybe rearrange someday because RS now used another place. + +extern struct rs *il2p_find_rs(int nparity); // Internal later? + +extern void il2p_encode_rs (unsigned char *tx_data, int data_size, int num_parity, unsigned char *parity_out); + +extern int il2p_decode_rs (unsigned char *rec_block, int data_size, int num_parity, unsigned char *out); + +extern int il2p_get_debug(void); +extern void il2p_set_debug(int debug); + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_rec.c +// +/////////////////////////////////////////////////////////////////////////////// + +// Receives a bit stream from demodulator. + +extern void il2p_rec_bit (int chan, int subchan, int slice, int dbit); + + + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_send.c +// +/////////////////////////////////////////////////////////////////////////////// + +#include "ax25_pad.h" // For packet object. + +// Send bit stream to modulator. + +int il2p_send_frame (int chan, packet_t pp, int max_fec, int polarity); + + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_codec.c +// +/////////////////////////////////////////////////////////////////////////////// + +#include "ax25_pad.h" + +extern int il2p_encode_frame (packet_t pp, int max_fec, unsigned char *iout); + +packet_t il2p_decode_frame (unsigned char *irec); + +packet_t il2p_decode_header_payload (unsigned char* uhdr, unsigned char *epayload, int *symbols_corrected); + + + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_header.c +// +/////////////////////////////////////////////////////////////////////////////// + + +extern int il2p_type_1_header (packet_t pp, int max_fec, unsigned char *hdr); + +extern packet_t il2p_decode_header_type_1 (unsigned char *hdr, int num_sym_changed); + + +extern int il2p_type_0_header (packet_t pp, int max_fec, unsigned char *hdr); + +extern int il2p_clarify_header(unsigned char *rec_hdr, unsigned char *corrected_descrambled_hdr); + + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_scramble.c +// +/////////////////////////////////////////////////////////////////////////////// + +extern void il2p_scramble_block (unsigned char *in, unsigned char *out, int len); + +extern void il2p_descramble_block (unsigned char *in, unsigned char *out, int len); + + +/////////////////////////////////////////////////////////////////////////////// +// +// il2p_payload.c +// +/////////////////////////////////////////////////////////////////////////////// + + +typedef struct { + int payload_byte_count; // Total size, 0 thru 1023 + int payload_block_count; + int small_block_size; + int large_block_size; + int large_block_count; + int small_block_count; + int parity_symbols_per_block; // 2, 4, 6, 8, 16 +} il2p_payload_properties_t; + +extern int il2p_payload_compute (il2p_payload_properties_t *p, int payload_size, int max_fec); + +extern int il2p_encode_payload (unsigned char *payload, int payload_size, int max_fec, unsigned char *enc); + +extern int il2p_decode_payload (unsigned char *received, int payload_size, int max_fec, unsigned char *payload_out, int *symbols_corrected); + +extern int il2p_get_header_attributes (unsigned char *hdr, int *hdr_type, int *max_fec); + +#endif diff --git a/src/il2p_codec.c b/src/il2p_codec.c new file mode 100644 index 00000000..a20aed65 --- /dev/null +++ b/src/il2p_codec.c @@ -0,0 +1,263 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "il2p.h" +#include "textcolor.h" +#include "demod.h" + + +/*------------------------------------------------------------- + * + * File: il2p_codec.c + * + * Purpose: Convert IL2P encoded format from and to direwolf internal packet format. + * + *--------------------------------------------------------------*/ + + +/*------------------------------------------------------------- + * + * Name: il2p_encode_frame + * + * Purpose: Convert AX.25 frame to IL2P encoding. + * + * Inputs: chan - Audio channel number, 0 = first. + * + * pp - Packet object pointer. + * + * max_fec - 1 to send maximum FEC size rather than automatic. + * + * Outputs: iout - Encoded result, excluding the 3 byte sync word. + * Caller should provide IL2P_MAX_PACKET_SIZE bytes. + * + * Returns: Number of bytes for transmission. + * -1 is returned for failure. + * + * Description: Encode into IL2P format. + * + * Errors: If something goes wrong, return -1. + * + * Most likely reason is that the frame is too large. + * IL2P has a max payload size of 1023 bytes. + * For a type 1 header, this is the maximum AX.25 Information part size. + * For a type 0 header, this is the entire AX.25 frame. + * + *--------------------------------------------------------------*/ + +int il2p_encode_frame (packet_t pp, int max_fec, unsigned char *iout) +{ + +// Can a type 1 header be used? + + unsigned char hdr[IL2P_HEADER_SIZE + IL2P_HEADER_PARITY]; + int e; + int out_len = 0; + + e = il2p_type_1_header (pp, max_fec, hdr); + if (e >= 0) { + il2p_scramble_block (hdr, iout, IL2P_HEADER_SIZE); + il2p_encode_rs (iout, IL2P_HEADER_SIZE, IL2P_HEADER_PARITY, iout+IL2P_HEADER_SIZE); + out_len = IL2P_HEADER_SIZE + IL2P_HEADER_PARITY; + + if (e == 0) { + // Success. No info part. + return (out_len); + } + + // Payload is AX.25 info part. + unsigned char *pinfo; + int info_len; + info_len = ax25_get_info (pp, &pinfo); + + int k = il2p_encode_payload (pinfo, info_len, max_fec, iout+out_len); + if (k > 0) { + out_len += k; + // Success. Info part was <= 1023 bytes. + return (out_len); + } + + // Something went wrong with the payload encoding. + return (-1); + } + else if (e == -1) { + +// Could not use type 1 header for some reason. +// e.g. More than 2 addresses, extended (mod 128) sequence numbers, etc. + + e = il2p_type_0_header (pp, max_fec, hdr); + if (e > 0) { + + il2p_scramble_block (hdr, iout, IL2P_HEADER_SIZE); + il2p_encode_rs (iout, IL2P_HEADER_SIZE, IL2P_HEADER_PARITY, iout+IL2P_HEADER_SIZE); + out_len = IL2P_HEADER_SIZE + IL2P_HEADER_PARITY; + + // Payload is entire AX.25 frame. + + unsigned char *frame_data_ptr = ax25_get_frame_data_ptr (pp); + int frame_len = ax25_get_frame_len (pp); + int k = il2p_encode_payload (frame_data_ptr, frame_len, max_fec, iout+out_len); + if (k > 0) { + out_len += k; + // Success. Entire AX.25 frame <= 1023 bytes. + return (out_len); + } + // Something went wrong with the payload encoding. + return (-1); + } + else if (e == 0) { + // Impossible condition. Type 0 header must have payload. + return (-1); + } + else { + // AX.25 frame is too large. + return (-1); + } + } + + // AX.25 Information part is too large. + return (-1); +} + + + +/*------------------------------------------------------------- + * + * Name: il2p_decode_frame + * + * Purpose: Convert IL2P encoding to AX.25 frame. + * This is only used during testing, with a whole encoded frame. + * During reception, the header would have FEC and descrambling + * applied first so we would know how much to collect for the payload. + * + * Inputs: irec - Received IL2P frame excluding the 3 byte sync word. + * + * Future Out: Number of symbols corrected. + * + * Returns: Packet pointer or NULL for error. + * + *--------------------------------------------------------------*/ + +packet_t il2p_decode_frame (unsigned char *irec) +{ + unsigned char uhdr[IL2P_HEADER_SIZE]; // After FEC and descrambling. + int e = il2p_clarify_header (irec, uhdr); + + // TODO?: for symmetry we might want to clarify the payload before combining. + + return (il2p_decode_header_payload(uhdr, irec + IL2P_HEADER_SIZE + IL2P_HEADER_PARITY, &e)); +} + + +/*------------------------------------------------------------- + * + * Name: il2p_decode_header_payload + * + * Purpose: Convert IL2P encoding to AX.25 frame + * + * Inputs: uhdr - Received header after FEC and descrambling. + * epayload - Encoded payload. + * + * In/Out: symbols_corrected - Symbols (bytes) corrected in the header. + * Should be 0 or 1 because it has 2 parity symbols. + * Here we add number of corrections for the payload. + * + * Returns: Packet pointer or NULL for error. + * + *--------------------------------------------------------------*/ + +packet_t il2p_decode_header_payload (unsigned char* uhdr, unsigned char *epayload, int *symbols_corrected) +{ + int hdr_type; + int max_fec; + int payload_len = il2p_get_header_attributes (uhdr, &hdr_type, &max_fec); + + packet_t pp = NULL; + + if (hdr_type == 1) { + +// Header type 1. Any payload is the AX.25 Information part. + + pp = il2p_decode_header_type_1 (uhdr, *symbols_corrected); + if (pp == NULL) { + // Failed for some reason. + return (NULL); + } + + if (payload_len > 0) { + // This is the AX.25 Information part. + + unsigned char extracted[IL2P_MAX_PAYLOAD_SIZE]; + int e = il2p_decode_payload (epayload, payload_len, max_fec, extracted, symbols_corrected); + + // It would be possible to have a good header but too many errors in the payload. + + if (e <= 0) { + ax25_delete (pp); + pp = NULL; + return (pp); + } + + if (e != payload_len) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P Internal Error: %s(): hdr_type=%d, max_fec=%d, payload_len=%d, e=%d.\n", __func__, hdr_type, max_fec, payload_len, e); + } + + ax25_set_info (pp, extracted, payload_len); + } + return (pp); + } + else { + +// Header type 0. The payload is the entire AX.25 frame. + + unsigned char extracted[IL2P_MAX_PAYLOAD_SIZE]; + int e = il2p_decode_payload (epayload, payload_len, max_fec, extracted, symbols_corrected); + + if (e <= 0) { // Payload was not received correctly. + return (NULL); + } + + if (e != payload_len) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P Internal Error: %s(): hdr_type=%d, e=%d, payload_len=%d\n", __func__, hdr_type, e, payload_len); + return (NULL); + } + + alevel_t alevel; + memset (&alevel, 0, sizeof(alevel)); + //alevel = demod_get_audio_level (chan, subchan); // What TODO? We don't know channel here. + // I think alevel gets filled in somewhere later making + // this redundant. + + pp = ax25_from_frame (extracted, payload_len, alevel); + return (pp); + } + +} // end il2p_decode_header_payload + +// end il2p_codec.c + + diff --git a/src/il2p_header.c b/src/il2p_header.c new file mode 100644 index 00000000..94fd25ba --- /dev/null +++ b/src/il2p_header.c @@ -0,0 +1,679 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "textcolor.h" +#include "ax25_pad.h" +#include "ax25_pad2.h" +#include "il2p.h" + + + +/*-------------------------------------------------------------------------------- + * + * File: il2p_header.c + * + * Purpose: Functions to deal with the IL2P header. + * + * Reference: http://tarpn.net/t/il2p/il2p-specification0-4.pdf + * + *--------------------------------------------------------------------------------*/ + + + +// Convert ASCII to/from DEC SIXBIT as defined here: +// https://en.wikipedia.org/wiki/Six-bit_character_code#DEC_six-bit_code + +static inline int ascii_to_sixbit (int a) +{ + if (a >= ' ' && a <= '_') return (a - ' '); + return (31); // '?' for any invalid. +} + +static inline int sixbit_to_ascii (int s) +{ + return (s + ' '); +} + +// Functions for setting the various header fields. +// It is assumed that it was zeroed first so only the '1' bits are set. + +static void set_field (unsigned char *hdr, int bit_num, int lsb_index, int width, int value) +{ + while (width > 0 && value != 0) { + assert (lsb_index >= 0 && lsb_index <= 11); + if (value & 1) { + hdr[lsb_index] |= 1 << bit_num; + } + value >>= 1; + lsb_index--; + width--; + } + assert (value == 0); +} + +#define SET_UI(hdr,val) set_field(hdr, 6, 0, 1, val) + +#define SET_PID(hdr,val) set_field(hdr, 6, 4, 4, val) + +#define SET_CONTROL(hdr,val) set_field(hdr, 6, 11, 7, val) + + +#define SET_FEC_LEVEL(hdr,val) set_field(hdr, 7, 0, 1, val) + +#define SET_HDR_TYPE(hdr,val) set_field(hdr, 7, 1, 1, val) + +#define SET_PAYLOAD_BYTE_COUNT(hdr,val) set_field(hdr, 7, 11, 10, val) + + +// Extracting the fields. + +static int get_field (unsigned char *hdr, int bit_num, int lsb_index, int width) +{ + int result = 0; + lsb_index -= width - 1; + while (width > 0) { + result <<= 1; + assert (lsb_index >= 0 && lsb_index <= 11); + if (hdr[lsb_index] & (1 << bit_num)) { + result |= 1; + } + lsb_index++; + width--; + } + return (result); +} + +#define GET_UI(hdr) get_field(hdr, 6, 0, 1) + +#define GET_PID(hdr) get_field(hdr, 6, 4, 4) + +#define GET_CONTROL(hdr) get_field(hdr, 6, 11, 7) + + +#define GET_FEC_LEVEL(hdr) get_field(hdr, 7, 0, 1) + +#define GET_HDR_TYPE(hdr) get_field(hdr, 7, 1, 1) + +#define GET_PAYLOAD_BYTE_COUNT(hdr) get_field(hdr, 7, 11, 10) + + + +// AX.25 'I' and 'UI' frames have a protocol ID which determines how the +// information part should be interpreted. +// Here we squeeze the most common cases down to 4 bits. +// Return -1 if translation is not possible. Fall back to type 0 header in this case. + +static int encode_pid (packet_t pp) +{ + int pid = ax25_get_pid(pp); + + if ((pid & 0x30) == 0x20) return (0x2); // AX.25 Layer 3 + if ((pid & 0x30) == 0x10) return (0x2); // AX.25 Layer 3 + if (pid == 0x01) return (0x3); // ISO 8208 / CCIT X.25 PLP + if (pid == 0x06) return (0x4); // Compressed TCP/IP + if (pid == 0x07) return (0x5); // Uncompressed TCP/IP + if (pid == 0x08) return (0x6); // Segmentation fragmen + if (pid == 0xcc) return (0xb); // ARPA Internet Protocol + if (pid == 0xcd) return (0xc); // ARPA Address Resolution + if (pid == 0xce) return (0xd); // FlexNet + if (pid == 0xcf) return (0xe); // TheNET + if (pid == 0xf0) return (0xf); // No L3 + return (-1); +} + +// Convert IL2P 4 bit PID to AX.25 8 bit PID. + + +static int decode_pid (int pid) +{ + static const unsigned char axpid[16] = { + 0xf0, // Should not happen. 0 is for 'S' frames. + 0xf0, // Should not happen. 1 is for 'U' frames (but not UI). + 0x20, // AX.25 Layer 3 + 0x01, // ISO 8208 / CCIT X.25 PLP + 0x06, // Compressed TCP/IP + 0x07, // Uncompressed TCP/IP + 0x08, // Segmentation fragment + 0xf0, // Future + 0xf0, // Future + 0xf0, // Future + 0xf0, // Future + 0xcc, // ARPA Internet Protocol + 0xcd, // ARPA Address Resolution + 0xce, // FlexNet + 0xcf, // TheNET + 0xf0 }; // No L3 + + assert (pid >= 0 && pid <= 15); + return (axpid[pid]); +} + + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_type_1_header + * + * Purpose: Attempt to create type 1 header from packet object. + * + * Inputs: pp - Packet object. + * + * max_fec - 1 to use maximum FEC symbols , 0 for automatic. + * + * Outputs: hdr - IL2P header with no scrambling or parity symbols. + * Must be large enough to hold IL2P_HEADER_SIZE unsigned bytes. + * + * Returns: Number of bytes for information part or -1 for failure. + * In case of failure, fall back to type 0 transparent encapsulation. + * + * Description: Type 1 Headers do not support AX.25 repeater callsign addressing, + * Modulo-128 extended mode window sequence numbers, nor any callsign + * characters that cannot translate to DEC SIXBIT. + * If these cases are encountered during IL2P packet encoding, + * the encoder switches to Type 0 Transparent Encapsulation. + * SABME can't be handled by type 1. + * + *--------------------------------------------------------------------------------*/ + +int il2p_type_1_header (packet_t pp, int max_fec, unsigned char *hdr) +{ + memset (hdr, 0, IL2P_HEADER_SIZE); + + if (ax25_get_num_addr(pp) != 2) { + // Only two addresses are allowed for type 1 header. + return (-1); + } + + // Check does not apply for 'U' frames but put in one place rather than two. + + if (ax25_get_modulo(pp) == 128) return(-1); + +// Destination and source addresses go into low bits 0-5 for bytes 0-11. + + char dst_addr[AX25_MAX_ADDR_LEN]; + char src_addr[AX25_MAX_ADDR_LEN]; + + ax25_get_addr_no_ssid (pp, AX25_DESTINATION, dst_addr); + int dst_ssid = ax25_get_ssid (pp, AX25_DESTINATION); + + ax25_get_addr_no_ssid (pp, AX25_SOURCE, src_addr); + int src_ssid = ax25_get_ssid (pp, AX25_SOURCE); + + unsigned char *a = (unsigned char *)dst_addr; + for (int i = 0; *a != '\0'; i++, a++) { + if (*a < ' ' || *a > '_') { + // Shouldn't happen but follow the rule. + return (-1); + } + hdr[i] = ascii_to_sixbit(*a); + } + + a = (unsigned char *)src_addr; + for (int i = 6; *a != '\0'; i++, a++) { + if (*a < ' ' || *a > '_') { + // Shouldn't happen but follow the rule. + return (-1); + } + hdr[i] = ascii_to_sixbit(*a); + } + + // Byte 12 has DEST SSID in upper nybble and SRC SSID in lower nybble and + hdr[12] = (dst_ssid << 4) | src_ssid; + + ax25_frame_type_t frame_type; + cmdres_t cr; // command or response. + char description[64]; + int pf; // Poll/Final. + int nr, ns; // Sequence numbers. + + frame_type = ax25_frame_type (pp, &cr, description, &pf, &nr, &ns); + + //dw_printf ("%s(): %s-%d>%s-%d: %s\n", __func__, src_addr, src_ssid, dst_addr, dst_ssid, description); + + switch (frame_type) { + + case frame_type_S_RR: // Receive Ready - System Ready To Receive + case frame_type_S_RNR: // Receive Not Ready - TNC Buffer Full + case frame_type_S_REJ: // Reject Frame - Out of Sequence or Duplicate + case frame_type_S_SREJ: // Selective Reject - Request single frame repeat + + // S frames (RR, RNR, REJ, SREJ), mod 8, have control N(R) P/F S S 0 1 + // These are mapped into P/F N(R) C S S + // Bit 6 is not mentioned in documentation but it is used for P/F for the other frame types. + // C is copied from the C bit in the destination addr. + // C from source is not used here. Reception assumes it is the opposite. + // PID is set to 0, meaning none, for S frames. + + SET_UI(hdr, 0); + SET_PID(hdr, 0); + SET_CONTROL(hdr, (pf<<6) | (nr<<3) | (((cr == cr_cmd) | (cr == cr_11))<<2)); + + // This gets OR'ed into the above. + switch (frame_type) { + case frame_type_S_RR: SET_CONTROL(hdr, 0); break; + case frame_type_S_RNR: SET_CONTROL(hdr, 1); break; + case frame_type_S_REJ: SET_CONTROL(hdr, 2); break; + case frame_type_S_SREJ: SET_CONTROL(hdr, 3); break; + default: break; + } + + break; + + case frame_type_U_SABM: // Set Async Balanced Mode + case frame_type_U_DISC: // Disconnect + case frame_type_U_DM: // Disconnect Mode + case frame_type_U_UA: // Unnumbered Acknowledge + case frame_type_U_FRMR: // Frame Reject + case frame_type_U_UI: // Unnumbered Information + case frame_type_U_XID: // Exchange Identification + case frame_type_U_TEST: // Test + + // The encoding allows only 3 bits for frame type and SABME got left out. + // Control format: P/F opcode[3] C n/a n/a + // The grayed out n/a bits are observed as 00 in the example. + // The header UI field must also be set for UI frames. + // PID is set to 1 for all U frames other than UI. + + if (frame_type == frame_type_U_UI) { + SET_UI(hdr, 1); // I guess this is how we distinguish 'I' and 'UI' + // on the receiving end. + int pid = encode_pid(pp); + if (pid < 0) return (-1); + SET_PID(hdr, pid); + } + else { + SET_PID(hdr, 1); // 1 for 'U' other than 'UI'. + } + + // Each of the destination and source addresses has a "C" bit. + // They should normally have the opposite setting. + // IL2P has only a single bit to represent 4 possbilities. + // + // dst src il2p meaning + // --- --- ---- ------- + // 0 0 0 Not valid (earlier protocol version) + // 1 0 1 Command (v2) + // 0 1 0 Response (v2) + // 1 1 1 Not valid (earlier protocol version) + // + // APRS does not mention how to set these bits and all 4 combinations + // are seen in the wild. Apparently these are ignored on receive and no + // one cares. Here we copy from the C bit in the destination address. + // It should be noted that the case of both C bits being the same can't + // be represented so the il2p encode/decode bit not produce exactly the + // same bits. We see this in the second example in the protocol spec. + // The original UI frame has both C bits of 0 so it is received as a response. + + SET_CONTROL(hdr, (pf<<6) | (((cr == cr_cmd) | (cr == cr_11))<<2)); + + // This gets OR'ed into the above. + switch (frame_type) { + case frame_type_U_SABM: SET_CONTROL(hdr, 0<<3); break; + case frame_type_U_DISC: SET_CONTROL(hdr, 1<<3); break; + case frame_type_U_DM: SET_CONTROL(hdr, 2<<3); break; + case frame_type_U_UA: SET_CONTROL(hdr, 3<<3); break; + case frame_type_U_FRMR: SET_CONTROL(hdr, 4<<3); break; + case frame_type_U_UI: SET_CONTROL(hdr, 5<<3); break; + case frame_type_U_XID: SET_CONTROL(hdr, 6<<3); break; + case frame_type_U_TEST: SET_CONTROL(hdr, 7<<3); break; + default: break; + } + break; + + case frame_type_I: // Information + + // I frames (mod 8 only) + // encoded control: P/F N(R) N(S) + + SET_UI(hdr, 0); + + int pid2 = encode_pid(pp); + if (pid2 < 0) return (-1); + SET_PID(hdr, pid2); + + SET_CONTROL(hdr, (pf<<6) | (nr<<3) | ns); + break; + + case frame_type_U_SABME: // Set Async Balanced Mode, Extended + case frame_type_U: // other Unnumbered, not used by AX.25. + case frame_not_AX25: // Could not get control byte from frame. + default: + + // Fall back to the header type 0 for these. + return (-1); + } + +// Common for all header type 1. + + // Bit 7 has [FEC Level:1], [HDR Type:1], [Payload byte Count:10] + + SET_FEC_LEVEL(hdr, max_fec); + SET_HDR_TYPE(hdr, 1); + + unsigned char *pinfo; + int info_len; + + info_len = ax25_get_info (pp, &pinfo); + if (info_len < 0 || info_len > IL2P_MAX_PAYLOAD_SIZE) { + return (-2); + } + + SET_PAYLOAD_BYTE_COUNT(hdr, info_len); + return (info_len); +} + + +// This should create a packet from the IL2P header. +// The information part will not be filled in. + +static void trim (char *stuff) +{ + char *p = stuff + strlen(stuff) - 1; + while (strlen(stuff) > 0 && (*p == ' ')) { + *p = '\0'; + p--; + } +} + + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_decode_header_type_1 + * + * Purpose: Attempt to convert type 1 header to a packet object. + * + * Inputs: hdr - IL2P header with no scrambling or parity symbols. + * + * num_sym_changed - Number of symbols changed by FEC in the header. + * Should be 0 or 1. + * + * Returns: Packet Object or NULL for failure. + * + * Description: A later step will process the payload for the information part. + * + *--------------------------------------------------------------------------------*/ + +packet_t il2p_decode_header_type_1 (unsigned char *hdr, int num_sym_changed) +{ + + if (GET_HDR_TYPE(hdr) != 1 ) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P Internal error. Should not be here: %s, when header type is 0.\n", __func__); + return (NULL); + } + +// First get the addresses including SSID. + + char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; + int num_addr = 2; + memset (addrs, 0, 2*AX25_MAX_ADDR_LEN); + +// The IL2P header uses 2 parity symbols which means a single corrupted symbol (byte) +// can always be corrected. +// However, I have seen cases, where the error rate is very high, where the RS decoder +// thinks it found a valid code block by changing one symbol but it was the wrong one. +// The result is trash. This shows up as address fields like 'R&G4"A' and 'TEW\ !'. +// I added a sanity check here to catch characters other than upper case letters and digits. +// The frame should be rejected in this case. The question is whether to discard it +// silently or print a message so the user can see that something strange is happening? +// My current thinking is that it should be silently ignored if the header has been +// modified (correctee or more likely, made worse in this cases). +// If no changes were made, something weird is happening. We should mention it for +// troubleshooting rather than sweeping it under the rug. + +// The same thing has been observed with the payload, under very high error conditions, +// and max_fec==0. Here I don't see a good solution. AX.25 information can contain +// "binary" data so I'm not sure what sort of sanity check could be added. +// This was not observed with max_fec==1. If we make that the default, same as Nino TNC, +// it would be extremely extremely unlikely unless someone explicitly selects weaker FEC. + +// TODO: We could do something similar for header type 0. +// The address fields should be all binary zero values. +// Someone overly ambitious might check the addresses found in the first payload block. + + for (int i = 0; i <= 5; i++) { + addrs[AX25_DESTINATION][i] = sixbit_to_ascii(hdr[i] & 0x3f); + } + trim (addrs[AX25_DESTINATION]); + for (int i = 0; i < strlen(addrs[AX25_DESTINATION]); i++) { + if (! isupper(addrs[AX25_DESTINATION][i]) && ! isdigit(addrs[AX25_DESTINATION][i])) { + if (num_sym_changed == 0) { + // This can pop up sporadically when receiving random noise. + // Would be better to show only when debug is enabled but variable not available here. + // TODO: For now we will just suppress it. + //text_color_set(DW_COLOR_ERROR); + //dw_printf ("IL2P: Invalid character '%c' in destination address '%s'\n", addrs[AX25_DESTINATION][i], addrs[AX25_DESTINATION]); + } + return (NULL); + } + } + snprintf (addrs[AX25_DESTINATION]+strlen(addrs[AX25_DESTINATION]), 4, "-%d", (hdr[12] >> 4) &0xf); + + for (int i = 0; i <= 5; i++) { + addrs[AX25_SOURCE][i] = sixbit_to_ascii(hdr[i+6] & 0x3f); + } + trim (addrs[AX25_SOURCE]); + for (int i = 0; i < strlen(addrs[AX25_SOURCE]); i++) { + if (! isupper(addrs[AX25_SOURCE][i]) && ! isdigit(addrs[AX25_SOURCE][i])) { + if (num_sym_changed == 0) { + // This can pop up sporadically when receiving random noise. + // Would be better to show only when debug is enabled but variable not available here. + // TODO: For now we will just suppress it. + //text_color_set(DW_COLOR_ERROR); + //dw_printf ("IL2P: Invalid character '%c' in source address '%s'\n", addrs[AX25_SOURCE][i], addrs[AX25_SOURCE]); + } + return (NULL); + } + } + snprintf (addrs[AX25_SOURCE]+strlen(addrs[AX25_SOURCE]), 4, "-%d", hdr[12] &0xf); + +// The PID field gives us the general type. +// 0 = 'S' frame. +// 1 = 'U' frame other than UI. +// others are either 'UI' or 'I' depending on the UI field. + + int pid = GET_PID(hdr); + int ui = GET_UI(hdr); + + if (pid == 0) { + +// 'S' frame. +// The control field contains: P/F N(R) C S S + + int control = GET_CONTROL(hdr); + cmdres_t cr = (control & 0x04) ? cr_cmd : cr_res; + ax25_frame_type_t ftype; + switch (control & 0x03) { + case 0: ftype = frame_type_S_RR; break; + case 1: ftype = frame_type_S_RNR; break; + case 2: ftype = frame_type_S_REJ; break; + default: ftype = frame_type_S_SREJ; break; + } + int modulo = 8; + int nr = (control >> 3) & 0x07; + int pf = (control >> 6) & 0x01; + unsigned char *pinfo = NULL; // Any info for SREJ will be added later. + int info_len = 0; + return (ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, pinfo, info_len)); + } + else if (pid == 1) { + +// 'U' frame other than 'UI'. +// The control field contains: P/F OPCODE{3) C x x + + int control = GET_CONTROL(hdr); + cmdres_t cr = (control & 0x04) ? cr_cmd : cr_res; + int axpid = 0; // unused for U other than UI. + ax25_frame_type_t ftype; + switch ((control >> 3) & 0x7) { + case 0: ftype = frame_type_U_SABM; break; + case 1: ftype = frame_type_U_DISC; break; + case 2: ftype = frame_type_U_DM; break; + case 3: ftype = frame_type_U_UA; break; + case 4: ftype = frame_type_U_FRMR; break; + case 5: ftype = frame_type_U_UI; axpid = 0xf0; break; // Should not happen with IL2P pid == 1. + case 6: ftype = frame_type_U_XID; break; + default: ftype = frame_type_U_TEST; break; + } + int pf = (control >> 6) & 0x01; + unsigned char *pinfo = NULL; // Any info for UI, XID, TEST will be added later. + int info_len = 0; + return (ax25_u_frame (addrs, num_addr, cr, ftype, pf, axpid, pinfo, info_len)); + } + else if (ui) { + +// 'UI' frame. +// The control field contains: P/F OPCODE{3) C x x + + int control = GET_CONTROL(hdr); + cmdres_t cr = (control & 0x04) ? cr_cmd : cr_res; + ax25_frame_type_t ftype = frame_type_U_UI; + int pf = (control >> 6) & 0x01; + int axpid = decode_pid(GET_PID(hdr)); + unsigned char *pinfo = NULL; // Any info for UI, XID, TEST will be added later. + int info_len = 0; + return (ax25_u_frame (addrs, num_addr, cr, ftype, pf, axpid, pinfo, info_len)); + } + else { + +// 'I' frame. +// The control field contains: P/F N(R) N(S) + + int control = GET_CONTROL(hdr); + cmdres_t cr = cr_cmd; // Always command. + int pf = (control >> 6) & 0x01; + int nr = (control >> 3) & 0x7; + int ns = control & 0x7; + int modulo = 8; + int axpid = decode_pid(GET_PID(hdr)); + unsigned char *pinfo = NULL; // Any info for UI, XID, TEST will be added later. + int info_len = 0; + return (ax25_i_frame (addrs, num_addr, cr, modulo, nr, ns, pf, axpid, pinfo, info_len)); + } + return (NULL); // unreachable but avoid warning. + +} // end + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_type_0_header + * + * Purpose: Attempt to create type 0 header from packet object. + * + * Inputs: pp - Packet object. + * + * max_fec - 1 to use maximum FEC symbols, 0 for automatic. + * + * Outputs: hdr - IL2P header with no scrambling or parity symbols. + * Must be large enough to hold IL2P_HEADER_SIZE unsigned bytes. + * + * Returns: Number of bytes for information part or -1 for failure. + * In case of failure, fall back to type 0 transparent encapsulation. + * + * Description: The type 0 header is used when it is not one of the restricted cases + * covered by the type 1 header. + * The AX.25 frame is put in the payload. + * This will cover: more than one address, mod 128 sequences, etc. + * + *--------------------------------------------------------------------------------*/ + +int il2p_type_0_header (packet_t pp, int max_fec, unsigned char *hdr) +{ + memset (hdr, 0, IL2P_HEADER_SIZE); + + // Bit 7 has [FEC Level:1], [HDR Type:1], [Payload byte Count:10] + + SET_FEC_LEVEL(hdr, max_fec); + SET_HDR_TYPE(hdr, 0); + + int frame_len = ax25_get_frame_len (pp); + + if (frame_len < 14 || frame_len > IL2P_MAX_PAYLOAD_SIZE) { + return (-2); + } + + SET_PAYLOAD_BYTE_COUNT(hdr, frame_len); + return (frame_len); +} + + +/*********************************************************************************** + * + * Name: il2p_get_header_attributes + * + * Purpose: Extract a few attributes from an IL2p header. + * + * Inputs: hdr - IL2P header structure. + * + * Outputs: hdr_type - 0 or 1. + * + * max_fec - 0 for automatic or 1 for fixed maximum size. + * + * Returns: Payload byte count. (actual payload size, not the larger encoded format) + * + ***********************************************************************************/ + + +int il2p_get_header_attributes (unsigned char *hdr, int *hdr_type, int *max_fec) +{ + *hdr_type = GET_HDR_TYPE(hdr); + *max_fec = GET_FEC_LEVEL(hdr); + return(GET_PAYLOAD_BYTE_COUNT(hdr)); +} + + +/*********************************************************************************** + * + * Name: il2p_clarify_header + * + * Purpose: Convert received header to usable form. + * This involves RS FEC then descrambling. + * + * Inputs: rec_hdr - Header as received over the radio. + * + * Outputs: corrected_descrambled_hdr - After RS FEC and unscrambling. + * + * Returns: Number of symbols that were corrected: + * 0 = No errors + * 1 = Single symbol corrected. + * <0 = Unable to obtain good header. + * + ***********************************************************************************/ + +int il2p_clarify_header(unsigned char *rec_hdr, unsigned char *corrected_descrambled_hdr) +{ + unsigned char corrected[IL2P_HEADER_SIZE+IL2P_HEADER_PARITY]; + + int e = il2p_decode_rs (rec_hdr, IL2P_HEADER_SIZE, IL2P_HEADER_PARITY, corrected); + + il2p_descramble_block (corrected, corrected_descrambled_hdr, IL2P_HEADER_SIZE); + + return (e); +} + +// end il2p_header.c \ No newline at end of file diff --git a/src/il2p_init.c b/src/il2p_init.c new file mode 100644 index 00000000..533a2213 --- /dev/null +++ b/src/il2p_init.c @@ -0,0 +1,226 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "textcolor.h" +#include "fx25.h" // For Reed Solomon stuff. +#include "il2p.h" + +// Interesting related stuff: +// https://www.kernel.org/doc/html/v4.15/core-api/librs.html +// https://berthub.eu/articles/posts/reed-solomon-for-programmers/ + + +#define MAX_NROOTS 16 + +#define NTAB 5 + +static struct { + int symsize; // Symbol size, bits (1-8). Always 8 for this application. + int genpoly; // Field generator polynomial coefficients. + int fcs; // First root of RS code generator polynomial, index form. + // FX.25 uses 1 but IL2P uses 0. + int prim; // Primitive element to generate polynomial roots. + int nroots; // RS code generator polynomial degree (number of roots). + // Same as number of check bytes added. + struct rs *rs; // Pointer to RS codec control block. Filled in at init time. +} Tab[NTAB] = { + {8, 0x11d, 0, 1, 2, NULL }, // 2 parity + {8, 0x11d, 0, 1, 4, NULL }, // 4 parity + {8, 0x11d, 0, 1, 6, NULL }, // 6 parity + {8, 0x11d, 0, 1, 8, NULL }, // 8 parity + {8, 0x11d, 0, 1, 16, NULL }, // 16 parity +}; + + + +static int g_il2p_debug = 0; + + +/*------------------------------------------------------------- + * + * Name: il2p_init + * + * Purpose: This must be called at application start up time. + * It sets up tables for the Reed-Solomon functions. + * + * Inputs: debug - Enable debug output. + * + *--------------------------------------------------------------*/ + +void il2p_init (int il2p_debug) +{ + g_il2p_debug = il2p_debug; + + for (int i = 0 ; i < NTAB ; i++) { + assert (Tab[i].nroots <= MAX_NROOTS); + Tab[i].rs = INIT_RS(Tab[i].symsize, Tab[i].genpoly, Tab[i].fcs, Tab[i].prim, Tab[i].nroots); + if (Tab[i].rs == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf("IL2P internal error: init_rs_char failed!\n"); + exit(EXIT_FAILURE); + } + } + +} // end il2p_init + + +int il2p_get_debug(void) +{ + return (g_il2p_debug); +} +void il2p_set_debug(int debug) +{ + g_il2p_debug = debug; +} + + +// Find RS codec control block for specified number of parity symbols. + +struct rs *il2p_find_rs(int nparity) +{ + for (int n = 0; n < NTAB; n++) { + if (Tab[n].nroots == nparity) { + return (Tab[n].rs); + } + } + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P INTERNAL ERROR: il2p_find_rs: control block not found for nparity = %d.\n", nparity); + return (Tab[0].rs); +} + + +/*------------------------------------------------------------- + * + * Name: void il2p_encode_rs + * + * Purpose: Add parity symbols to a block of data. + * + * Inputs: tx_data Header or other data to transmit. + * data_size Number of data bytes in above. + * num_parity Number of parity symbols to add. + * Maximum of IL2P_MAX_PARITY_SYMBOLS. + * + * Outputs: parity_out Specified number of parity symbols + * + * Restriction: data_size + num_parity <= 255 which is the RS block size. + * The caller must ensure this. + * + *--------------------------------------------------------------*/ + +void il2p_encode_rs (unsigned char *tx_data, int data_size, int num_parity, unsigned char *parity_out) +{ + assert (data_size >= 1); + assert (num_parity == 2 || num_parity == 4 || num_parity == 6 || num_parity == 8 || num_parity == 16); + assert (data_size + num_parity <= 255); + + unsigned char rs_block[FX25_BLOCK_SIZE]; + memset (rs_block, 0, sizeof(rs_block)); + memcpy (rs_block + sizeof(rs_block) - data_size - num_parity, tx_data, data_size); + ENCODE_RS (il2p_find_rs(num_parity), rs_block, parity_out); +} + +/*------------------------------------------------------------- + * + * Name: void il2p_decode_rs + * + * Purpose: Check and attempt to fix block with FEC. + * + * Inputs: rec_block Received block composed of data and parity. + * Total size is sum of following two parameters. + * data_size Number of data bytes in above. + * num_parity Number of parity symbols (bytes) in above. + * + * Outputs: out Original with possible corrections applied. + * data_size bytes. + * + * Returns: -1 for unrecoverable. + * >= 0 for success. Number of symbols corrected. + * + *--------------------------------------------------------------*/ + +int il2p_decode_rs (unsigned char *rec_block, int data_size, int num_parity, unsigned char *out) +{ + + // Use zero padding in front if data size is too small. + + int n = data_size + num_parity; // total size in. + + unsigned char rs_block[FX25_BLOCK_SIZE]; + + // We could probably do this more efficiently by skipping the + // processing of the bytes known to be zero. Good enough for now. + + memset (rs_block, 0, sizeof(rs_block) - n); + memcpy (rs_block + sizeof(rs_block) - n, rec_block, n); + + if (il2p_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("============================== il2p_decode_rs ==============================\n"); + dw_printf ("%d filler zeros, %d data, %d parity\n", (int)(sizeof(rs_block) - n), data_size, num_parity); + fx_hex_dump (rs_block, sizeof (rs_block)); + } + + int derrlocs[FX25_MAX_CHECK]; // Half would probably be OK. + + int derrors = DECODE_RS(il2p_find_rs(num_parity), rs_block, derrlocs, 0); + memcpy (out, rs_block + sizeof(rs_block) - n, data_size); + + if (il2p_get_debug() >= 3) { + if (derrors == 0) { + dw_printf ("No errors reported for RS block.\n"); + } + else if (derrors > 0) { + dw_printf ("%d errors fixed in positions:\n", derrors); + for (int j = 0; j < derrors; j++) { + dw_printf (" %3d (0x%02x)\n", derrlocs[j] , derrlocs[j]); + } + fx_hex_dump (rs_block, sizeof (rs_block)); + } + } + + // It is possible to have a situation where too many errors are + // present but the algorithm could get a good code block by "fixing" + // one of the padding bytes that should be 0. + + for (int i = 0; i < derrors; i++) { + if (derrlocs[i] < sizeof(rs_block) - n) { + if (il2p_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("RS DECODE ERROR! Padding position %d should be 0 but it was set to %02x.\n", derrlocs[i], rs_block[derrlocs[i]]); + } + derrors = -1; + break; + } + } + + if (il2p_get_debug() >= 3) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("============================== il2p_decode_rs returns %d ==============================\n", derrors); + } + return (derrors); +} + +// end il2p_init.c \ No newline at end of file diff --git a/src/il2p_payload.c b/src/il2p_payload.c new file mode 100644 index 00000000..d5fb4887 --- /dev/null +++ b/src/il2p_payload.c @@ -0,0 +1,298 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "textcolor.h" +#include "il2p.h" + + +/*-------------------------------------------------------------------------------- + * + * File: il2p_payload.c + * + * Purpose: Functions dealing with the payload. + * + *--------------------------------------------------------------------------------*/ + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_payload_compute + * + * Purpose: Compute number and sizes of data blocks based on total size. + * + * Inputs: payload_size 0 to 1023. (IL2P_MAX_PAYLOAD_SIZE) + * max_fec true for 16 parity symbols, false for automatic. + * + * Outputs: *p Payload block sizes and counts. + * Number of parity symbols per block. + * + * Returns: Number of bytes in the encoded format. + * Could be 0 for no payload blocks. + * -1 for error (i.e. invalid unencoded size: <0 or >1023) + * + *--------------------------------------------------------------------------------*/ + +int il2p_payload_compute (il2p_payload_properties_t *p, int payload_size, int max_fec) +{ + memset (p, 0, sizeof(il2p_payload_properties_t)); + + if (payload_size < 0 || payload_size > IL2P_MAX_PAYLOAD_SIZE) { + return (-1); + } + if (payload_size == 0) { + return (0); + } + + if (max_fec) { + p->payload_byte_count = payload_size; + p->payload_block_count = (p->payload_byte_count + 238) / 239; + p->small_block_size = p->payload_byte_count / p->payload_block_count; + p->large_block_size = p->small_block_size + 1; + p->large_block_count = p->payload_byte_count - (p->payload_block_count * p->small_block_size); + p->small_block_count = p->payload_block_count - p->large_block_count; + p->parity_symbols_per_block = 16; + } + else { + p->payload_byte_count = payload_size; + p->payload_block_count = (p->payload_byte_count + 246) / 247; + p->small_block_size = p->payload_byte_count / p->payload_block_count; + p->large_block_size = p->small_block_size + 1; + p->large_block_count = p->payload_byte_count - (p->payload_block_count * p->small_block_size); + p->small_block_count = p->payload_block_count - p->large_block_count; + //p->parity_symbols_per_block = (p->small_block_size / 32) + 2; // Looks like error in documentation + + // It would work if the number of parity symbols was based on large block size. + + if (p->small_block_size <= 61) p->parity_symbols_per_block = 2; + else if (p->small_block_size <= 123) p->parity_symbols_per_block = 4; + else if (p->small_block_size <= 185) p->parity_symbols_per_block = 6; + else if (p->small_block_size <= 247) p->parity_symbols_per_block = 8; + else { + // Should not happen. But just in case... + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P parity symbol per payload block error. small_block_size = %d\n", p->small_block_size); + return (-1); + } + } + + // Return the total size for the encoded format. + + return (p->small_block_count * (p->small_block_size + p->parity_symbols_per_block) + + p->large_block_count * (p->large_block_size + p->parity_symbols_per_block)); + +} // end il2p_payload_compute + + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_encode_payload + * + * Purpose: Split payload into multiple blocks such that each set + * of data and parity symbols fit into a 255 byte RS block. + * + * Inputs: *payload Array of bytes. + * payload_size 0 to 1023. (IL2P_MAX_PAYLOAD_SIZE) + * max_fec true for 16 parity symbols, false for automatic. + * + * Outputs: *enc Encoded payload for transmission. + * Up to IL2P_MAX_ENCODED_SIZE bytes. + * + * Returns: -1 for error (i.e. invalid size) + * 0 for no blocks. (i.e. size zero) + * Number of bytes generated. Maximum IL2P_MAX_ENCODED_SIZE. + * + * Note: I interpreted the protocol spec as saying the LFSR state is retained + * between data blocks. During interoperability testing, I found that + * was not the case. It is reset for each data block. + * + *--------------------------------------------------------------------------------*/ + + +int il2p_encode_payload (unsigned char *payload, int payload_size, int max_fec, unsigned char *enc) +{ + if (payload_size > IL2P_MAX_PAYLOAD_SIZE) return (-1); + if (payload_size == 0) return (0); + +// Determine number of blocks and sizes. + + il2p_payload_properties_t ipp; + int e; + e = il2p_payload_compute (&ipp, payload_size, max_fec); + if (e <= 0) { + return (e); + } + + unsigned char *pin = payload; + unsigned char *pout = enc; + int encoded_length = 0; + unsigned char scram[256]; + unsigned char parity[IL2P_MAX_PARITY_SYMBOLS]; + +// First the large blocks. + + for (int b = 0; b < ipp.large_block_count; b++) { + + il2p_scramble_block (pin, scram, ipp.large_block_size); + memcpy (pout, scram, ipp.large_block_size); + pin += ipp.large_block_size; + pout += ipp.large_block_size; + encoded_length += ipp.large_block_size; + il2p_encode_rs (scram, ipp.large_block_size, ipp.parity_symbols_per_block, parity); + memcpy (pout, parity, ipp.parity_symbols_per_block); + pout += ipp.parity_symbols_per_block; + encoded_length += ipp.parity_symbols_per_block; + } + +// Then the small blocks. + + for (int b = 0; b < ipp.small_block_count; b++) { + + il2p_scramble_block (pin, scram, ipp.small_block_size); + memcpy (pout, scram, ipp.small_block_size); + pin += ipp.small_block_size; + pout += ipp.small_block_size; + encoded_length += ipp.small_block_size; + il2p_encode_rs (scram, ipp.small_block_size, ipp.parity_symbols_per_block, parity); + memcpy (pout, parity, ipp.parity_symbols_per_block); + pout += ipp.parity_symbols_per_block; + encoded_length += ipp.parity_symbols_per_block; + } + + return (encoded_length); + +} // end il2p_encode_payload + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_decode_payload + * + * Purpose: Extract original data from encoded payload. + * + * Inputs: received Array of bytes. Size is unknown but in practice it + * must not exceed IL2P_MAX_ENCODED_SIZE. + * payload_size 0 to 1023. (IL2P_MAX_PAYLOAD_SIZE) + * Expected result size based on header. + * max_fec true for 16 parity symbols, false for automatic. + * + * Outputs: payload_out Recovered payload. + * + * In/Out: symbols_corrected Number of symbols corrected. + * + * + * Returns: Number of bytes extracted. Should be same as payload_size going in. + * -3 for unexpected internal inconsistency. + * -2 for unable to recover from signal corruption. + * -1 for invalid size. + * 0 for no blocks. (i.e. size zero) + * + * Description: Each block is scrambled separately but the LSFR state is carried + * from the first payload block to the next. + * + *--------------------------------------------------------------------------------*/ + +int il2p_decode_payload (unsigned char *received, int payload_size, int max_fec, unsigned char *payload_out, int *symbols_corrected) +{ +// Determine number of blocks and sizes. + + il2p_payload_properties_t ipp; + int e; + e = il2p_payload_compute (&ipp, payload_size, max_fec); + if (e <= 0) { + return (e); + } + + unsigned char *pin = received; + unsigned char *pout = payload_out; + int decoded_length = 0; + int failed = 0; + +// First the large blocks. + + for (int b = 0; b < ipp.large_block_count; b++) { + unsigned char corrected_block[255]; + int e = il2p_decode_rs (pin, ipp.large_block_size, ipp.parity_symbols_per_block, corrected_block); + + // dw_printf ("%s:%d: large block decode_rs returned status = %d\n", __FILE__, __LINE__, e); + + if (e < 0) failed = 1; + *symbols_corrected += e; + + il2p_descramble_block (corrected_block, pout, ipp.large_block_size); + + if (il2p_get_debug() >= 2) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Descrambled large payload block, %d bytes:\n", ipp.large_block_size); + fx_hex_dump(pout, ipp.large_block_size); + } + + pin += ipp.large_block_size + ipp.parity_symbols_per_block; + pout += ipp.large_block_size; + decoded_length += ipp.large_block_size; + } + +// Then the small blocks. + + for (int b = 0; b < ipp.small_block_count; b++) { + unsigned char corrected_block[255]; + int e = il2p_decode_rs (pin, ipp.small_block_size, ipp.parity_symbols_per_block, corrected_block); + + // dw_printf ("%s:%d: small block decode_rs returned status = %d\n", __FILE__, __LINE__, e); + + if (e < 0) failed = 1; + *symbols_corrected += e; + + il2p_descramble_block (corrected_block, pout, ipp.small_block_size); + + if (il2p_get_debug() >= 2) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Descrambled small payload block, %d bytes:\n", ipp.small_block_size); + fx_hex_dump(pout, ipp.small_block_size); + } + + pin += ipp.small_block_size + ipp.parity_symbols_per_block; + pout += ipp.small_block_size; + decoded_length += ipp.small_block_size; + } + + if (failed) { + //dw_printf ("%s:%d: failed = %0x\n", __FILE__, __LINE__, failed); + return (-2); + } + + if (decoded_length != payload_size) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P Internal error: decoded_length = %d, payload_size = %d\n", decoded_length, payload_size); + return (-3); + } + + return (decoded_length); + +} // end il2p_decode_payload + +// end il2p_payload.c + diff --git a/src/il2p_rec.c b/src/il2p_rec.c new file mode 100644 index 00000000..5ad457a0 --- /dev/null +++ b/src/il2p_rec.c @@ -0,0 +1,276 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + + +/******************************************************************************** + * + * File: il2p_rec.c + * + * Purpose: Extract IL2P frames from a stream of bits and process them. + * + * References: http://tarpn.net/t/il2p/il2p-specification0-4.pdf + * + *******************************************************************************/ + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "textcolor.h" +#include "il2p.h" +#include "multi_modem.h" +#include "demod.h" + + +struct il2p_context_s { + + enum { IL2P_SEARCHING=0, IL2P_HEADER, IL2P_PAYLOAD, IL2P_DECODE } state; + + unsigned int acc; // Accumulate most recent 24 bits for sync word matching. + // Lower 8 bits are also used for accumulating bytes for + // the header and payload. + + int bc; // Bit counter so we know when a complete byte has been accumulated. + + int polarity; // 1 if opposite of expected polarity. + + unsigned char shdr[IL2P_HEADER_SIZE+IL2P_HEADER_PARITY]; + // Scrambled header as received over the radio. Includes parity. + int hc; // Number if bytes placed in above. + + unsigned char uhdr[IL2P_HEADER_SIZE]; // Header after FEC and unscrambling. + + int eplen; // Encoded payload length. This is not the nuumber from + // from the header but rather the number of encoded bytes to gather. + + unsigned char spayload[IL2P_MAX_ENCODED_PAYLOAD_SIZE]; + // Scrambled and encoded payload as received over the radio. + int pc; // Number of bytes placed in above. + + int corrected; // Number of symbols corrected by RS FEC. +}; + +static struct il2p_context_s *il2p_context[MAX_CHANS][MAX_SUBCHANS][MAX_SLICERS]; + + + +/*********************************************************************************** + * + * Name: il2p_rec_bit + * + * Purpose: Extract FX.25 packets from a stream of bits. + * + * Inputs: chan - Channel number. + * + * subchan - This allows multiple demodulators per channel. + * + * slice - Allows multiple slicers per demodulator (subchannel). + * + * dbit - One bit from the received data stream. + * + * Description: This is called once for each received bit. + * For each valid packet, process_rec_frame() is called for further processing. + * It can gather multiple candidates from different parallel demodulators + * ("subchannels") and slicers, then decide which one is the best. + * + ***********************************************************************************/ + +void il2p_rec_bit (int chan, int subchan, int slice, int dbit) +{ + +// Allocate context blocks only as needed. + + struct il2p_context_s *F = il2p_context[chan][subchan][slice]; + if (F == NULL) { + assert (chan >= 0 && chan < MAX_CHANS); + assert (subchan >= 0 && subchan < MAX_SUBCHANS); + assert (slice >= 0 && slice < MAX_SLICERS); + F = il2p_context[chan][subchan][slice] = (struct il2p_context_s *)malloc(sizeof (struct il2p_context_s)); + assert (F != NULL); + memset (F, 0, sizeof(struct il2p_context_s)); + } + +// Accumulate most recent 24 bits received. Most recent is LSB. + + F->acc = ((F->acc << 1) | (dbit & 1)) & 0x00ffffff; + +// State machine to look for sync word then gather appropriate number of header and payload bytes. + + switch (F->state) { + + case IL2P_SEARCHING: // Searching for the sync word. + + if (__builtin_popcount (F->acc ^ IL2P_SYNC_WORD) <= 1) { // allow single bit mismatch + //text_color_set (DW_COLOR_INFO); + //dw_printf ("IL2P header has normal polarity\n"); + F->polarity = 0; + F->state = IL2P_HEADER; + F->bc = 0; + F->hc = 0; + } + else if (__builtin_popcount ((~F->acc & 0x00ffffff) ^ IL2P_SYNC_WORD) <= 1) { + text_color_set (DW_COLOR_INFO); + // FIXME - this pops up occasionally with random noise. Find better way to convey information. + // This also happens for each slicer - to noisy. + //dw_printf ("IL2P header has reverse polarity\n"); + F->polarity = 1; + F->state = IL2P_HEADER; + F->bc = 0; + F->hc = 0; + } + break; + + case IL2P_HEADER: // Gathering the header. + + F->bc++; + if (F->bc == 8) { // full byte has been collected. + F->bc = 0; + if ( ! F->polarity) { + F->shdr[F->hc++] = F->acc & 0xff; + } + else { + F->shdr[F->hc++] = (~ F->acc) & 0xff; + } + if (F->hc == IL2P_HEADER_SIZE+IL2P_HEADER_PARITY) { // Have all of header + + if (il2p_get_debug() >= 1) { + text_color_set (DW_COLOR_DEBUG); + dw_printf ("IL2P header as received [%d.%d.%d]:\n", chan, subchan, slice); + fx_hex_dump (F->shdr, IL2P_HEADER_SIZE+IL2P_HEADER_PARITY); + } + + // Fix any errors and descramble. + F->corrected = il2p_clarify_header(F->shdr, F->uhdr); + + if (F->corrected >= 0) { // Good header. + // How much payload is expected? + il2p_payload_properties_t plprop; + int hdr_type, max_fec; + int len = il2p_get_header_attributes (F->uhdr, &hdr_type, &max_fec); + + F->eplen = il2p_payload_compute (&plprop, len, max_fec); + + if (il2p_get_debug() >= 1) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("IL2P header after correcting %d symbols and unscrambling [%d.%d.%d]:\n", F->corrected, chan, subchan, slice); + fx_hex_dump (F->uhdr, IL2P_HEADER_SIZE); + dw_printf ("Header type %d, max fec = %d\n", hdr_type, max_fec); + dw_printf ("Need to collect %d encoded bytes for %d byte payload.\n", F->eplen, len); + dw_printf ("%d small blocks of %d and %d large blocks of %d. %d parity symbols per block\n", + plprop.small_block_count, plprop.small_block_size, + plprop.large_block_count, plprop.large_block_size, plprop.parity_symbols_per_block); + } + + if (F->eplen >= 1) { // Need to gather payload. + F->pc = 0; + F->state = IL2P_PAYLOAD; + } + else if (F->eplen == 0) { // No payload. + F->pc = 0; + F->state = IL2P_DECODE; + } + else { // Error. + + if (il2p_get_debug() >= 1) { + text_color_set (DW_COLOR_ERROR); + dw_printf ("IL2P header INVALID.\n"); + } + + F->state = IL2P_SEARCHING; + } + } // good header after FEC. + else { + F->state = IL2P_SEARCHING; // Header failed FEC check. + } + } // entire header has been collected. + } // full byte collected. + break; + + case IL2P_PAYLOAD: // Gathering the payload, if any. + + F->bc++; + if (F->bc == 8) { // full byte has been collected. + F->bc = 0; + if ( ! F->polarity) { + F->spayload[F->pc++] = F->acc & 0xff; + } + else { + F->spayload[F->pc++] = (~ F->acc) & 0xff; + } + if (F->pc == F->eplen) { + + // TODO?: for symmetry it seems like we should clarify the payload before combining. + + F->state = IL2P_DECODE; + } + } + break; + + case IL2P_DECODE: + // We get here after a good header and any payload has been collected. + // Processing is delayed by one bit but I think it makes the logic cleaner. + // During unit testing be sure to send an extra bit to flush it out at the end. + + // in uhdr[IL2P_HEADER_SIZE]; // Header after FEC and descrambling. + + // TODO?: for symmetry, we might decode the payload here and later build the frame. + + { + packet_t pp = il2p_decode_header_payload (F->uhdr, F->spayload, &(F->corrected)); + + if (il2p_get_debug() >= 1) { + if (pp != NULL) { + ax25_hex_dump (pp); + } + else { + // Most likely too many FEC errors. + text_color_set(DW_COLOR_ERROR); + dw_printf ("FAILED to construct frame in %s.\n", __func__); + } + } + + if (pp != NULL) { + alevel_t alevel = demod_get_audio_level (chan, subchan); + retry_t retries = F->corrected; + int is_fx25 = 1; // FIXME: distinguish fx.25 and IL2P. + // Currently this just means that a FEC mode was used. + + // TODO: Could we put last 3 arguments in packet object rather than passing around separately? + + multi_modem_process_rec_packet (chan, subchan, slice, pp, alevel, retries, is_fx25); + } + } // end block for local variables. + + if (il2p_get_debug() >= 1) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("-----\n"); + } + + F->state = IL2P_SEARCHING; + break; + + } // end of switch + +} // end il2p_rec_bit + + +// end il2p_rec.c diff --git a/src/il2p_scramble.c b/src/il2p_scramble.c new file mode 100644 index 00000000..6219cddb --- /dev/null +++ b/src/il2p_scramble.c @@ -0,0 +1,157 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +/*-------------------------------------------------------------------------------- + * + * File: il2p_scramble.c + * + * Purpose: Scramble / descramble data as specified in the IL2P protocol specification. + * + *--------------------------------------------------------------------------------*/ + + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "il2p.h" + + +// Scramble bits for il2p transmit. + +// Note that there is a delay of 5 until the first bit comes out. +// So we need to need to ignore the first 5 out and stick in +// an extra 5 filler bits to flush at the end. + +#define INIT_TX_LSFR 0x00f + +static inline int scramble_bit (int in, int *state) +{ + int out = ((*state >> 4) ^ *state) & 1; + *state = ( (((in ^ *state) & 1) << 9) | (*state ^ ((*state & 1) << 4)) ) >> 1; + return (out); +} + + +// Undo data scrambling for il2p receive. + +#define INIT_RX_LSFR 0x1f0 + +static inline int descramble_bit (int in, int *state) +{ + int out = (in ^ *state) & 1; + *state = ((*state >> 1) | ((in & 1) << 8)) ^ ((in & 1) << 3); + return (out); +} + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_scramble_block + * + * Purpose: Scramble a block before adding RS parity. + * + * Inputs: in Array of bytes. + * len Number of bytes both in and out. + * + * Outputs: out Array of bytes. + * + *--------------------------------------------------------------------------------*/ + +void il2p_scramble_block (unsigned char *in, unsigned char *out, int len) +{ + int tx_lfsr_state = INIT_TX_LSFR; + + memset (out, 0, len); + + int skipping = 1; // Discard the first 5 out. + int ob = 0; // Index to output byte. + int om = 0x80; // Output bit mask; + for (int ib = 0; ib < len; ib++) { + for (int im = 0x80; im != 0; im >>= 1) { + int s = scramble_bit((in[ib] & im) != 0, &tx_lfsr_state); + if (ib == 0 && im == 0x04) skipping = 0; + if ( ! skipping) { + if (s) { + out[ob] |= om; + } + om >>= 1; + if (om == 0) { + om = 0x80; + ob++; + } + } + } + } + // Flush it. + + // This is a relic from when I thought the state would need to + // be passed along for the next block. + // Preserve the LSFR state from before flushing. + // This might be needed as the initial state for later payload blocks. + int x = tx_lfsr_state; + for (int n = 0; n < 5; n++) { + int s = scramble_bit(0, &x); + if (s) { + out[ob] |= om; + } + om >>=1; + if (om == 0) { + om = 0x80; + ob++; + } + } + +} // end il2p_scramble_block + + + +/*-------------------------------------------------------------------------------- + * + * Function: il2p_descramble_block + * + * Purpose: Descramble a block after removing RS parity. + * + * Inputs: in Array of bytes. + * len Number of bytes both in and out. + * + * Outputs: out Array of bytes. + * + *--------------------------------------------------------------------------------*/ + +void il2p_descramble_block (unsigned char *in, unsigned char *out, int len) +{ + int rx_lfsr_state = INIT_RX_LSFR; + + memset (out, 0, len); + + for (int b = 0; b < len; b++) { + for (int m = 0x80; m != 0; m >>= 1) { + int d = descramble_bit((in[b] & m) != 0, &rx_lfsr_state); + if (d) { + out[b] |= m; + } + } + } +} + +// end il2p_scramble.c diff --git a/src/il2p_send.c b/src/il2p_send.c new file mode 100644 index 00000000..3c4554e0 --- /dev/null +++ b/src/il2p_send.c @@ -0,0 +1,147 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "il2p.h" +#include "textcolor.h" +#include "audio.h" +#include "gen_tone.h" + + +static int number_of_bits_sent[MAX_CHANS]; // Count number of bits sent by "il2p_send_frame" + +static void send_bytes (int chan, unsigned char *b, int count, int polarity); +static void send_bit (int chan, int b, int polarity); + + + +/*------------------------------------------------------------- + * + * Name: il2p_send_frame + * + * Purpose: Convert frames to a stream of bits in IL2P format. + * + * Inputs: chan - Audio channel number, 0 = first. + * + * pp - Pointer to packet object. + * + * max_fec - 1 to force 16 parity symbols for each payload block. + * 0 for automatic depending on block size. + * + * polarity - 0 for normal. 1 to invert signal. + * 2 special case for testing - introduce some errors to test FEC. + * + * Outputs: Bits are shipped out by calling tone_gen_put_bit(). + * + * Returns: Number of bits sent including + * - Preamble (01010101...) + * - 3 byte Sync Word. + * - 15 bytes for Header. + * - Optional payload. + * The required time can be calculated by dividing this + * number by the transmit rate of bits/sec. + * -1 is returned for failure. + * + * Description: Generate an IL2P encoded frame. + * + * Assumptions: It is assumed that the tone_gen module has been + * properly initialized so that bits sent with + * tone_gen_put_bit() are processed correctly. + * + * Errors: Return -1 for error. Probably frame too large. + * + * Note: Inconsistency here. ax25 version has just a byte array + * and length going in. Here we need the full packet object. + * + *--------------------------------------------------------------*/ + +int il2p_send_frame (int chan, packet_t pp, int max_fec, int polarity) +{ + unsigned char encoded[IL2P_MAX_PACKET_SIZE]; + + encoded[0] = ( IL2P_SYNC_WORD >> 16 ) & 0xff; + encoded[1] = ( IL2P_SYNC_WORD >> 8 ) & 0xff; + encoded[2] = ( IL2P_SYNC_WORD ) & 0xff; + + int elen = il2p_encode_frame (pp, max_fec, encoded + IL2P_SYNC_WORD_SIZE); + if (elen <= 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("IL2P: Unable to encode frame into IL2P.\n"); + return (-1); + } + + elen += IL2P_SYNC_WORD_SIZE; + + number_of_bits_sent[chan] = 0; + + if (il2p_get_debug() >= 1) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("IL2P frame, max_fec = %d, %d encoded bytes total\n", max_fec, elen); + fx_hex_dump (encoded, elen); + } + + // Clobber some bytes for testing. + if (polarity >= 2) { + for (int j = 10; j < elen; j+=100) { + encoded[j] = ~ encoded[j]; + } + } + + // Send bits to modulator. + + static unsigned char preamble = IL2P_PREAMBLE; + + send_bytes (chan, &preamble, 1, polarity); + send_bytes (chan, encoded, elen, polarity); + + return (number_of_bits_sent[chan]); +} + + +static void send_bytes (int chan, unsigned char *b, int count, int polarity) +{ + for (int j = 0; j < count; j++) { + unsigned int x = b[j]; + for (int k = 0; k < 8; k++) { + send_bit (chan, (x & 0x80) != 0, polarity); + x <<= 1; + } + } +} + +// NRZI would be applied for AX.25 but IL2P does not use it. +// However we do have an option to invert the signal. +// The direwolf receive implementation will automatically compensate +// for either polarity but other implementations might not. + +static void send_bit (int chan, int b, int polarity) +{ + tone_gen_put_bit (chan, (b ^ polarity) & 1); + number_of_bits_sent[chan]++; +} + + + +// end il2p_send.c \ No newline at end of file diff --git a/src/il2p_test.c b/src/il2p_test.c new file mode 100644 index 00000000..17995259 --- /dev/null +++ b/src/il2p_test.c @@ -0,0 +1,977 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2021 John Langner, WB2OSZ +// +// 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, see . +// + +#include "direwolf.h" + +#include +#include +#include +#include + +#include "textcolor.h" +#include "il2p.h" +#include "ax25_pad.h" +#include "ax25_pad2.h" +#include "multi_modem.h" + + +static void test_scramble(void); +static void test_rs(void); +static void test_payload(void); +static void test_example_headers(void); +static void all_frame_types(void); +static void test_serdes(void); +static void decode_bitstream(void); + +/*------------------------------------------------------------- + * + * Name: il2p_test.c + * + * Purpose: Unit tests for IL2P protocol functions. + * + * Errors: Die if anything goes wrong. + * + *--------------------------------------------------------------*/ + + +int main () +{ + int enable_color = 1; + text_color_init (enable_color); + + int enable_debug_out = 0; + il2p_init(enable_debug_out); + + text_color_set(DW_COLOR_INFO); + dw_printf ("Begin IL2P unit tests.\n"); + +// These start simple and later complex cases build upon earlier successes. + +// Test scramble and descramble. + + test_scramble(); + +// Test Reed Solomon error correction. + + test_rs(); + +// Test payload functions. + + test_payload(); + +// Try encoding the example headers in the protocol spec. + + test_example_headers(); + +// Convert all of the AX.25 frame types to IL2P and back again. + + all_frame_types(); + +// Use same serialize / deserialize functions used on the air. + + test_serdes (); + +// Decode bitstream from demodulator if data file is available. +// TODO: Very large info parts. Appropriate error if too long. +// TODO: More than 2 addresses. + + decode_bitstream(); + + text_color_set(DW_COLOR_REC); + dw_printf ("\n----------\n\n"); + dw_printf ("\nSUCCESS!\n"); + + return (EXIT_SUCCESS); +} + + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test scrambling and descrambling. +// +///////////////////////////////////////////////////////////////////////////////////////////// + +static void test_scramble(void) +{ + text_color_set(DW_COLOR_INFO); + dw_printf ("Test scrambling...\n"); + +// First an example from the protocol specification to make sure I'm compatible. + + static unsigned char scramin1[] = { 0x63, 0xf1, 0x40, 0x40, 0x40, 0x00, 0x6b, 0x2b, 0x54, 0x28, 0x25, 0x2a, 0x0f }; + static unsigned char scramout1[] = { 0x6a, 0xea, 0x9c, 0xc2, 0x01, 0x11, 0xfc, 0x14, 0x1f, 0xda, 0x6e, 0xf2, 0x53 }; + unsigned char scramout[sizeof(scramin1)]; + + il2p_scramble_block (scramin1, scramout, sizeof(scramin1)); + assert (memcmp(scramout, scramout, sizeof(scramout1)) == 0); + +} // end test_scramble. + + + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test Reed Solomon encode/decode examples found in the protocol spec. +// The data part is scrambled but that does not matter here because. +// We are only concerned abound adding the parity and verifying. +// +///////////////////////////////////////////////////////////////////////////////////////////// + + +static void test_rs() +{ + text_color_set(DW_COLOR_INFO); + dw_printf ("Test Reed Solomon functions...\n"); + + static unsigned char example_s[] = { 0x26, 0x57, 0x4d, 0x57, 0xf1, 0x96, 0xcc, 0x85, 0x42, 0xe7, 0x24, 0xf7, 0x2e, + 0x8a, 0x97 }; + unsigned char parity_out[2]; + il2p_encode_rs (example_s, 13, 2, parity_out); + //dw_printf ("DEBUG RS encode %02x %02x\n", parity_out[0], parity_out[1]); + assert (memcmp(parity_out, example_s + 13, 2) == 0); + + + static unsigned char example_u[] = { 0x6a, 0xea, 0x9c, 0xc2, 0x01, 0x11, 0xfc, 0x14, 0x1f, 0xda, 0x6e, 0xf2, 0x53, + 0x91, 0xbd }; + il2p_encode_rs (example_u, 13, 2, parity_out); + //dw_printf ("DEBUG RS encode %02x %02x\n", parity_out[0], parity_out[1]); + assert (memcmp(parity_out, example_u + 13, 2) == 0); + + // See if we can go the other way. + + unsigned char received[15]; + unsigned char corrected[15]; + int e; + + e = il2p_decode_rs (example_s, 13, 2, corrected); + assert (e == 0); + assert (memcmp(example_s, corrected, 13) == 0); + + memcpy (received, example_s, 15); + received[0] = '?'; + e = il2p_decode_rs (received, 13, 2, corrected); + assert (e == 1); + assert (memcmp(example_s, corrected, 13) == 0); + + e = il2p_decode_rs (example_u, 13, 2, corrected); + assert (e == 0); + assert (memcmp(example_u, corrected, 13) == 0); + + memcpy (received, example_u, 15); + received[12] = '?'; + e = il2p_decode_rs (received, 13, 2, corrected); + assert (e == 1); + assert (memcmp(example_u, corrected, 13) == 0); + + received[1] = '?'; + received[2] = '?'; + e = il2p_decode_rs (received, 13, 2, corrected); + assert (e == -1); +} + + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test payload functions. +// +///////////////////////////////////////////////////////////////////////////////////////////// + +static void test_payload(void) +{ + text_color_set(DW_COLOR_INFO); + dw_printf ("Test payload functions...\n"); + + il2p_payload_properties_t ipp; + int e; + +// Examples in specification. + + e = il2p_payload_compute (&ipp, 100, 0); + assert (ipp.small_block_size == 100); + assert (ipp.large_block_size == 101); + assert (ipp.large_block_count == 0); + assert (ipp.small_block_count == 1); + assert (ipp.parity_symbols_per_block == 4); + + e = il2p_payload_compute (&ipp, 236, 0); + assert (ipp.small_block_size == 236); + assert (ipp.large_block_size == 237); + assert (ipp.large_block_count == 0); + assert (ipp.small_block_count == 1); + assert (ipp.parity_symbols_per_block == 8); + + e = il2p_payload_compute (&ipp, 512, 0); + assert (ipp.small_block_size == 170); + assert (ipp.large_block_size == 171); + assert (ipp.large_block_count == 2); + assert (ipp.small_block_count == 1); + assert (ipp.parity_symbols_per_block == 6); + + e = il2p_payload_compute (&ipp, 1023, 0); + assert (ipp.small_block_size == 204); + assert (ipp.large_block_size == 205); + assert (ipp.large_block_count == 3); + assert (ipp.small_block_count == 2); + assert (ipp.parity_symbols_per_block == 8); + +// Now try all possible sizes for Baseline FEC Parity. + + for (int n = 1; n <= IL2P_MAX_PAYLOAD_SIZE; n++) { + e = il2p_payload_compute (&ipp, n, 0); + //dw_printf ("bytecount=%d, smallsize=%d, largesize=%d, largecount=%d, smallcount=%d\n", n, + // ipp.small_block_size, ipp.large_block_size, + // ipp.large_block_count, ipp.small_block_count); + //fflush (stdout); + + assert (ipp.payload_block_count >= 1 && ipp.payload_block_count <= IL2P_MAX_PAYLOAD_BLOCKS); + assert (ipp.payload_block_count == ipp.small_block_count + ipp.large_block_count); + assert (ipp.small_block_count * ipp.small_block_size + + ipp.large_block_count * ipp.large_block_size == n); + assert (ipp.parity_symbols_per_block == 2 || + ipp.parity_symbols_per_block == 4 || + ipp.parity_symbols_per_block == 6 || + ipp.parity_symbols_per_block == 8); + + // Data and parity must fit in RS block size of 255. + // Size test does not apply if block count is 0. + assert (ipp.small_block_count == 0 || ipp.small_block_size + ipp.parity_symbols_per_block <= 255); + assert (ipp.large_block_count == 0 || ipp.large_block_size + ipp.parity_symbols_per_block <= 255); + } + +// All sizes for MAX FEC. + + for (int n = 1; n <= IL2P_MAX_PAYLOAD_SIZE; n++) { + e = il2p_payload_compute (&ipp, n, 1); // 1 for max fec. + //dw_printf ("bytecount=%d, smallsize=%d, largesize=%d, largecount=%d, smallcount=%d\n", n, + // ipp.small_block_size, ipp.large_block_size, + // ipp.large_block_count, ipp.small_block_count); + //fflush (stdout); + + assert (ipp.payload_block_count >= 1 && ipp.payload_block_count <= IL2P_MAX_PAYLOAD_BLOCKS); + assert (ipp.payload_block_count == ipp.small_block_count + ipp.large_block_count); + assert (ipp.small_block_count * ipp.small_block_size + + ipp.large_block_count * ipp.large_block_size == n); + assert (ipp.parity_symbols_per_block == 16); + + // Data and parity must fit in RS block size of 255. + // Size test does not apply if block count is 0. + assert (ipp.small_block_count == 0 || ipp.small_block_size + ipp.parity_symbols_per_block <= 255); + assert (ipp.large_block_count == 0 || ipp.large_block_size + ipp.parity_symbols_per_block <= 255); + } + +// Now let's try encoding payloads and extracting original again. +// This will also provide exercise for scrambling and Reed Solomon under more conditions. + + unsigned char original_payload[IL2P_MAX_PAYLOAD_SIZE]; + for (int n = 0; n < IL2P_MAX_PAYLOAD_SIZE; n++) { + original_payload[n] = n & 0xff; + } + for (int max_fec = 0; max_fec <= 1; max_fec++) { + for (int payload_length = 1; payload_length <= IL2P_MAX_PAYLOAD_SIZE; payload_length++) { + //dw_printf ("\n--------- max_fec = %d, payload_length = %d\n", max_fec, payload_length); + unsigned char encoded[IL2P_MAX_ENCODED_PAYLOAD_SIZE]; + int k = il2p_encode_payload (original_payload, payload_length, max_fec, encoded); + + //dw_printf ("payload length %d %s -> %d\n", payload_length, max_fec ? "M" : "", k); + assert (k > payload_length && k <= IL2P_MAX_ENCODED_PAYLOAD_SIZE); + + // Now extract. + + unsigned char extracted[IL2P_MAX_PAYLOAD_SIZE]; + int symbols_corrected = 0; + int e = il2p_decode_payload (encoded, payload_length, max_fec, extracted, &symbols_corrected); + //dw_printf ("e = %d, payload_length = %d\n", e, payload_length); + assert (e == payload_length); + + // if (memcmp (original_payload, extracted, payload_length) != 0) { + // dw_printf ("********** Received message not as expected. **********\n"); + // fx_hex_dump(extracted, payload_length); + // } + assert (memcmp (original_payload, extracted, payload_length) == 0); + } + } + (void)e; +} // end test_payload + + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test header examples found in protocol specification. +// +///////////////////////////////////////////////////////////////////////////////////////////// + +static void test_example_headers() +{ + +//----------- Example 1: AX.25 S-Frame -------------- + +// This frame sample only includes a 15 byte header, without PID field. +// Destination Callsign: ?KA2DEW-2 +// Source Callsign: ?KK4HEJ-7 +// N(R): 5 +// P/F: 1 +// C: 1 +// Control Opcode: 00 (Receive Ready) +// +// AX.25 data: +// 96 82 64 88 8a ae e4 96 96 68 90 8a 94 6f b1 +// +// IL2P Data Prior to Scrambling and RS Encoding: +// 2b a1 12 24 25 77 6b 2b 54 68 25 2a 27 +// +// IL2P Data After Scrambling and RS Encoding: +// 26 57 4d 57 f1 96 cc 85 42 e7 24 f7 2e 8a 97 + + text_color_set(DW_COLOR_INFO); + dw_printf ("Example 1: AX.25 S-Frame...\n"); + + static unsigned char example1[] = {0x96, 0x82, 0x64, 0x88, 0x8a, 0xae, 0xe4, 0x96, 0x96, 0x68, 0x90, 0x8a, 0x94, 0x6f, 0xb1}; + static unsigned char header1[] = {0x2b, 0xa1, 0x12, 0x24, 0x25, 0x77, 0x6b, 0x2b, 0x54, 0x68, 0x25, 0x2a, 0x27 }; + unsigned char header[IL2P_HEADER_SIZE]; + unsigned char sresult[32]; + memset (header, 0, sizeof(header)); + memset (sresult, 0, sizeof(sresult)); + unsigned char check[2]; + alevel_t alevel; + memset(&alevel, 0, sizeof(alevel)); + + packet_t pp = ax25_from_frame (example1, sizeof(example1), alevel); + assert (pp != NULL); + int e; + e = il2p_type_1_header (pp, 0, header); + assert (e == 0); + ax25_delete(pp); + + //dw_printf ("Example 1 header:\n"); + //for (int i = 0 ; i < sizeof(header); i++) { + // dw_printf (" %02x", header[i]); + //} + ///dw_printf ("\n"); + + assert (memcmp(header, header1, sizeof(header)) == 0); + + il2p_scramble_block (header, sresult, 13); + //dw_printf ("Expect scrambled 26 57 4d 57 f1 96 cc 85 42 e7 24 f7 2e\n"); + //for (int i = 0 ; i < sizeof(sresult); i++) { + // dw_printf (" %02x", sresult[i]); + //} + //dw_printf ("\n"); + + il2p_encode_rs (sresult, 13, 2, check); + + //dw_printf ("expect checksum = 8a 97\n"); + //dw_printf ("check = "); + //for (int i = 0 ; i < sizeof(check); i++) { + // dw_printf (" %02x", check[i]); + //} + //dw_printf ("\n"); + assert (check[0] == 0x8a); + assert (check[1] == 0x97); + +// Can we go from IL2P back to AX.25? + + pp = il2p_decode_header_type_1 (header, 0); + assert (pp != NULL); + + char dst_addr[AX25_MAX_ADDR_LEN]; + char src_addr[AX25_MAX_ADDR_LEN]; + + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, dst_addr); + ax25_get_addr_with_ssid (pp, AX25_SOURCE, src_addr); + + ax25_frame_type_t frame_type; + cmdres_t cr; // command or response. + char description[64]; + int pf; // Poll/Final. + int nr, ns; // Sequence numbers. + + frame_type = ax25_frame_type (pp, &cr, description, &pf, &nr, &ns); + (void)frame_type; +#if 1 + dw_printf ("%s(): %s>%s: %s\n", __func__, src_addr, dst_addr, description); +#endif +// TODO: compare binary. + ax25_delete (pp); + + dw_printf ("Example 1 header OK\n"); + + +// -------------- Example 2 - UI frame, no info part ------------------ + +// This is an AX.25 Unnumbered Information frame, such as APRS. +// Destination Callsign: ?CQ -0 +// Source Callsign: ?KK4HEJ-15 +// P/F: 0 +// C: 0 +// Control Opcode: 3 Unnumbered Information +// PID: 0xF0 No L3 +// +// AX.25 Data: +// 86 a2 40 40 40 40 60 96 96 68 90 8a 94 7f 03 f0 +// +// IL2P Data Prior to Scrambling and RS Encoding: +// 63 f1 40 40 40 00 6b 2b 54 28 25 2a 0f +// +// IL2P Data After Scrambling and RS Encoding: +// 6a ea 9c c2 01 11 fc 14 1f da 6e f2 53 91 bd + + + //dw_printf ("---------- example 2 ------------\n"); + static unsigned char example2[] = { 0x86, 0xa2, 0x40, 0x40, 0x40, 0x40, 0x60, 0x96, 0x96, 0x68, 0x90, 0x8a, 0x94, 0x7f, 0x03, 0xf0 }; + static unsigned char header2[] = { 0x63, 0xf1, 0x40, 0x40, 0x40, 0x00, 0x6b, 0x2b, 0x54, 0x28, 0x25, 0x2a, 0x0f }; + memset (header, 0, sizeof(header)); + memset (sresult, 0, sizeof(sresult)); + memset(&alevel, 0, sizeof(alevel)); + + pp = ax25_from_frame (example2, sizeof(example2), alevel); + assert (pp != NULL); + e = il2p_type_1_header (pp, 0, header); + assert (e == 0); + ax25_delete(pp); + + //dw_printf ("Example 2 header:\n"); + //for (int i = 0 ; i < sizeof(header); i++) { + // dw_printf (" %02x", header[i]); + //} + //dw_printf ("\n"); + + assert (memcmp(header, header2, sizeof(header2)) == 0); + + il2p_scramble_block (header, sresult, 13); + //dw_printf ("Expect scrambled 6a ea 9c c2 01 11 fc 14 1f da 6e f2 53\n"); + //for (int i = 0 ; i < sizeof(sresult); i++) { + // dw_printf (" %02x", sresult[i]); + //} + //dw_printf ("\n"); + + il2p_encode_rs (sresult, 13, 2, check); + + //dw_printf ("expect checksum = 91 bd\n"); + //dw_printf ("check = "); + //for (int i = 0 ; i < sizeof(check); i++) { + // dw_printf (" %02x", check[i]); + //} + //dw_printf ("\n"); + assert (check[0] == 0x91); + assert (check[1] == 0xbd); + +// Can we go from IL2P back to AX.25? + + pp = il2p_decode_header_type_1 (header, 0); + assert (pp != NULL); + + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, dst_addr); + ax25_get_addr_with_ssid (pp, AX25_SOURCE, src_addr); + + frame_type = ax25_frame_type (pp, &cr, description, &pf, &nr, &ns); + (void)frame_type; +#if 1 + dw_printf ("%s(): %s>%s: %s\n", __func__, src_addr, dst_addr, description); +#endif +// TODO: compare binary. + + ax25_delete (pp); +// TODO: more examples + + dw_printf ("Example 2 header OK\n"); + + +// -------------- Example 3 - I Frame ------------------ + +// This is an AX.25 I-Frame with 9 bytes of information after the 16 byte header. +// +// Destination Callsign: ?KA2DEW-2 +// Source Callsign: ?KK4HEJ-2 +// P/F: 1 +// C: 1 +// N(R): 5 +// N(S): 4 +// AX.25 PID: 0xCF TheNET +// IL2P Payload Byte Count: 9 +// +// AX.25 Data: +// 96 82 64 88 8a ae e4 96 96 68 90 8a 94 65 b8 cf 30 31 32 33 34 35 36 37 38 +// +// IL2P Scrambled and Encoded Data: +// 26 13 6d 02 8c fe fb e8 aa 94 2d 6a 34 43 35 3c 69 9f 0c 75 5a 38 a1 7f f3 fc + + + //dw_printf ("---------- example 3 ------------\n"); + static unsigned char example3[] = { 0x96, 0x82, 0x64, 0x88, 0x8a, 0xae, 0xe4, 0x96, 0x96, 0x68, 0x90, 0x8a, 0x94, 0x65, 0xb8, 0xcf, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38 }; + static unsigned char header3[] = { 0x2b, 0xe1, 0x52, 0x64, 0x25, 0x77, 0x6b, 0x2b, 0xd4, 0x68, 0x25, 0xaa, 0x22 }; + static unsigned char complete3[] = { 0x26, 0x13, 0x6d, 0x02, 0x8c, 0xfe, 0xfb, 0xe8, 0xaa, 0x94, 0x2d, 0x6a, 0x34, 0x43, 0x35, 0x3c, 0x69, 0x9f, 0x0c, 0x75, 0x5a, 0x38, 0xa1, 0x7f, 0xf3, 0xfc }; + memset (header, 0, sizeof(header)); + memset (sresult, 0, sizeof(sresult)); + memset(&alevel, 0, sizeof(alevel)); + + pp = ax25_from_frame (example3, sizeof(example3), alevel); + assert (pp != NULL); + e = il2p_type_1_header (pp, 0, header); + assert (e == 9); + ax25_delete(pp); + + //dw_printf ("Example 3 header:\n"); + //for (int i = 0 ; i < sizeof(header); i++) { + // dw_printf (" %02x", header[i]); + //} + //dw_printf ("\n"); + + assert (memcmp(header, header3, sizeof(header)) == 0); + + il2p_scramble_block (header, sresult, 13); + //dw_printf ("Expect scrambled 26 13 6d 02 8c fe fb e8 aa 94 2d 6a 34\n"); + //for (int i = 0 ; i < sizeof(sresult); i++) { + // dw_printf (" %02x", sresult[i]); + //} + //dw_printf ("\n"); + + il2p_encode_rs (sresult, 13, 2, check); + + //dw_printf ("expect checksum = 43 35\n"); + //dw_printf ("check = "); + //for (int i = 0 ; i < sizeof(check); i++) { + // dw_printf (" %02x", check[i]); + //} + //dw_printf ("\n"); + + assert (check[0] == 0x43); + assert (check[1] == 0x35); + + // That was only the header. We will get to the info part in a later test. + +// Can we go from IL2P back to AX.25? + + pp = il2p_decode_header_type_1 (header, 0); + assert (pp != NULL); + + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, dst_addr); + ax25_get_addr_with_ssid (pp, AX25_SOURCE, src_addr); + + frame_type = ax25_frame_type (pp, &cr, description, &pf, &nr, &ns); + (void)frame_type; +#if 1 + dw_printf ("%s(): %s>%s: %s\n", __func__, src_addr, dst_addr, description); +#endif +// TODO: compare binary. + + ax25_delete (pp); + dw_printf ("Example 3 header OK\n"); + +// Example 3 again, this time the Information part is included. + + pp = ax25_from_frame (example3, sizeof(example3), alevel); + assert (pp != NULL); + + int max_fec = 0; + unsigned char iout[IL2P_MAX_PACKET_SIZE]; + e = il2p_encode_frame (pp, max_fec, iout); + + //dw_printf ("expected for example 3:\n"); + //fx_hex_dump(complete3, sizeof(complete3)); + //dw_printf ("actual result for example 3:\n"); + //fx_hex_dump(iout, e); + // Does it match the example in the protocol spec? + assert (e == sizeof(complete3)); + assert (memcmp(iout, complete3, sizeof(complete3)) == 0); + ax25_delete (pp); + + dw_printf ("Example 3 with info OK\n"); + +} // end test_example_headers + + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test all of the frame types. +// +// Encode to IL2P format, decode, and verify that the result is the same as the original. +// +///////////////////////////////////////////////////////////////////////////////////////////// + + +static void enc_dec_compare (packet_t pp1) +{ + for (int max_fec = 0; max_fec <= 1; max_fec++) { + + unsigned char encoded[IL2P_MAX_PACKET_SIZE]; + int enc_len; + enc_len = il2p_encode_frame (pp1, max_fec, encoded); + assert (enc_len >= 0); + + packet_t pp2; + pp2 = il2p_decode_frame (encoded); + assert (pp2 != NULL); + +// Is it the same after encoding to IL2P and then decoding? + + int len1 = ax25_get_frame_len (pp1); + unsigned char *data1 = ax25_get_frame_data_ptr (pp1); + + int len2 = ax25_get_frame_len (pp2); + unsigned char *data2 = ax25_get_frame_data_ptr (pp2); + + if (len1 != len2 || memcmp(data1, data2, len1) != 0) { + + dw_printf ("\nEncode/Decode Error. Original:\n"); + ax25_hex_dump (pp1); + + dw_printf ("IL2P encoded as:\n"); + fx_hex_dump(encoded, enc_len); + + dw_printf ("Got turned into this:\n"); + ax25_hex_dump (pp2); + } + + assert (len1 == len2 && memcmp(data1, data2, len1) == 0); + + ax25_delete (pp2); + } +} + +static void all_frame_types(void) +{ + char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; + int num_addr = 2; + cmdres_t cr; + ax25_frame_type_t ftype; + int pf = 0; + int pid = 0xf0; + int modulo; + int nr, ns; + unsigned char *pinfo = NULL; + int info_len = 0; + packet_t pp; + + strcpy (addrs[0], "W2UB"); + strcpy (addrs[1], "WB2OSZ-12"); + num_addr = 2; + + text_color_set(DW_COLOR_INFO); + dw_printf ("Testing all frame types.\n"); + +/* U frame */ + + dw_printf ("\nU frames...\n"); + + for (ftype = frame_type_U_SABME; ftype <= frame_type_U_TEST; ftype++) { + + for (pf = 0; pf <= 1; pf++) { + + int cmin = 0, cmax = 1; + + switch (ftype) { + // 0 = response, 1 = command + case frame_type_U_SABME: cmin = 1; cmax = 1; break; + case frame_type_U_SABM: cmin = 1; cmax = 1; break; + case frame_type_U_DISC: cmin = 1; cmax = 1; break; + case frame_type_U_DM: cmin = 0; cmax = 0; break; + case frame_type_U_UA: cmin = 0; cmax = 0; break; + case frame_type_U_FRMR: cmin = 0; cmax = 0; break; + case frame_type_U_UI: cmin = 0; cmax = 1; break; + case frame_type_U_XID: cmin = 0; cmax = 1; break; + case frame_type_U_TEST: cmin = 0; cmax = 1; break; + default: break; // avoid compiler warning. + } + + for (cr = cmin; cr <= cmax; cr++) { + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct U frame, cr=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_u_frame (addrs, num_addr, cr, ftype, pf, pid, pinfo, info_len); + ax25_hex_dump (pp); + enc_dec_compare (pp); + ax25_delete (pp); + } + } + } + + +/* S frame */ + + //strcpy (addrs[2], "DIGI1-1"); + //num_addr = 3; + + dw_printf ("\nS frames...\n"); + + for (ftype = frame_type_S_RR; ftype <= frame_type_S_SREJ; ftype++) { + + for (pf = 0; pf <= 1; pf++) { + + modulo = 8; + nr = modulo / 2 + 1; + + // SREJ can only be response. + + for (cr = 0; cr <= (int)(ftype!=frame_type_S_SREJ); cr++) { + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct S frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, NULL, 0); + + ax25_hex_dump (pp); + enc_dec_compare (pp); + ax25_delete (pp); + } + + modulo = 128; + nr = modulo / 2 + 1; + + for (cr = 0; cr <= (int)(ftype!=frame_type_S_SREJ); cr++) { + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct S frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, NULL, 0); + + ax25_hex_dump (pp); + enc_dec_compare (pp); + ax25_delete (pp); + } + } + } + +/* SREJ is only S frame which can have information part. */ + + static unsigned char srej_info[] = { 1<<1, 2<<1, 3<<1, 4<<1 }; + + ftype = frame_type_S_SREJ; + for (pf = 0; pf <= 1; pf++) { + + modulo = 128; + nr = 127; + cr = cr_res; + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct Multi-SREJ S frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_s_frame (addrs, num_addr, cr, ftype, modulo, nr, pf, srej_info, (int)(sizeof(srej_info))); + + ax25_hex_dump (pp); + enc_dec_compare (pp); + ax25_delete (pp); + } + + +/* I frame */ + + dw_printf ("\nI frames...\n"); + + pinfo = (unsigned char*)"The rain in Spain stays mainly on the plain."; + info_len = strlen((char*)pinfo); + + for (pf = 0; pf <= 1; pf++) { + + modulo = 8; + nr = 0x55 & (modulo - 1); + ns = 0xaa & (modulo - 1); + + for (cr = 1; cr <= 1; cr++) { // can only be command + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct I frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_i_frame (addrs, num_addr, cr, modulo, nr, ns, pf, pid, pinfo, info_len); + + ax25_hex_dump (pp); + enc_dec_compare (pp); + ax25_delete (pp); + } + + modulo = 128; + nr = 0x55 & (modulo - 1); + ns = 0xaa & (modulo - 1); + + for (cr = 1; cr <= 1; cr++) { + + text_color_set(DW_COLOR_INFO); + dw_printf ("\nConstruct I frame, cmd=%d, ftype=%d, pid=0x%02x\n", cr, ftype, pid); + + pp = ax25_i_frame (addrs, num_addr, cr, modulo, nr, ns, pf, pid, pinfo, info_len); + + ax25_hex_dump (pp); + enc_dec_compare (pp); + ax25_delete (pp); + } + } + +} // end all_frame_types + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test bitstream tapped off from demodulator. +// +// 5 frames were sent to Nino TNC and a recording was made. +// This was demodulated and the resulting bit stream saved to a file. +// +// No automatic test here - must be done manually with audio recording. +// +///////////////////////////////////////////////////////////////////////////////////////////// + +static int decoding_bitstream = 0; + +static void decode_bitstream(void) +{ + dw_printf("-----\nReading il2p-bitstream.txt if available...\n"); + + FILE *fp = fopen ("il2p-bitstream.txt", "r"); + if (fp == NULL) { + dw_printf ("Bitstream test file not available.\n"); + return; + } + + decoding_bitstream = 1; + int save_previous = il2p_get_debug(); + il2p_set_debug (1); + + int ch; + while ( (ch = fgetc(fp)) != EOF) { + + if (ch == '0' || ch == '1') { + il2p_rec_bit (0, 0, 0, ch - '0'); + } + } + fclose(fp); + il2p_set_debug (save_previous); + decoding_bitstream = 0; + +} // end decode_bitstream + + + + +///////////////////////////////////////////////////////////////////////////////////////////// +// +// Test serialize / deserialize. +// +// This uses same functions used on the air. +// +///////////////////////////////////////////////////////////////////////////////////////////// + +static char addrs2[] = "AA1AAA-1>ZZ9ZZZ-9"; +static char addrs3[] = "AA1AAA-1>ZZ9ZZZ-9,DIGI*"; +static char text[] = + "'... As I was saying, that seems to be done right - though I haven't time to look it over thoroughly just now - and that shows that there are three hundred and sixty-four days when you might get un-birthday presents -'" + "\n" + "'Certainly,' said Alice." + "\n" + "'And only one for birthday presents, you know. There's glory for you!'" + "\n" + "'I don't know what you mean by \"glory\",' Alice said." + "\n" + "Humpty Dumpty smiled contemptuously. 'Of course you don't - till I tell you. I meant \"there's a nice knock-down argument for you!\"'" + "\n" + "'But \"glory\" doesn't mean \"a nice knock-down argument\",' Alice objected." + "\n" + "'When I use a word,' Humpty Dumpty said, in rather a scornful tone, 'it means just what I choose it to mean - neither more nor less.'" + "\n" + "'The question is,' said Alice, 'whether you can make words mean so many different things.'" + "\n" + "'The question is,' said Humpty Dumpty, 'which is to be master - that's all.'" + "\n" ; + + +static int rec_count = -1; // disable deserialized packet test. +static int polarity = 0; + +static void test_serdes (void) +{ + text_color_set(DW_COLOR_INFO); + dw_printf ("\nTest serialize / deserialize...\n"); + rec_count = 0; + + int max_fec = 1; + + // try combinations of header type, max_fec, polarity, errors. + + for (int hdr_type = 0; hdr_type <= 1; hdr_type++) { + char packet[1024]; + snprintf (packet, sizeof(packet), "%s:%s", hdr_type ? addrs2 : addrs3, text); + packet_t pp = ax25_from_text (packet, 1); + assert (pp != NULL); + + int chan = 0; + + + for (max_fec = 0; max_fec <= 1; max_fec++) { + for (polarity = 0; polarity <= 2; polarity++) { // 2 means throw in some errors. + int num_bits_sent = il2p_send_frame (chan, pp, max_fec, polarity); + dw_printf ("%d bits sent.\n", num_bits_sent); + + // Need extra bit at end to flush out state machine. + il2p_rec_bit (0, 0, 0, 0); + } + } + ax25_delete(pp); + } + + dw_printf ("Serdes receive count = %d\n", rec_count); + assert (rec_count == 12); + rec_count = -1; // disable deserialized packet test. +} + + +// Serializing calls this which then simulates the demodulator output. + +void tone_gen_put_bit (int chan, int data) +{ + il2p_rec_bit (chan, 0, 0, data); +} + +// This is called when a complete frame has been deserialized. + +void multi_modem_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, fec_type_t fec_type) +{ + if (rec_count < 0) return; // Skip check before serdes test. + + rec_count++; + + // Does it have the the expected content? + + unsigned char *pinfo; + int len = ax25_get_info(pp, &pinfo); + assert (len == strlen(text)); + assert (strcmp(text, (char*)pinfo) == 0); + + dw_printf ("Number of symbols corrected: %d\n", retries); + if (polarity == 2) { // expecting errors corrected. + assert (retries == 10); + } + else { // should be no errors. + assert (retries == 0); + } + + ax25_delete (pp); +} + +alevel_t demod_get_audio_level (int chan, int subchan) +{ + alevel_t alevel; + memset (&alevel, 0, sizeof(alevel)); + return (alevel); +} + +// end il2p_test.c \ No newline at end of file diff --git a/src/kiss.c b/src/kiss.c new file mode 100644 index 00000000..f93cb94c --- /dev/null +++ b/src/kiss.c @@ -0,0 +1,602 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2011, 2013, 2014, 2016, 2017 John Langner, WB2OSZ +// +// 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, see . +// + + + +/*------------------------------------------------------------------ + * + * Module: kiss.c + * + * Purpose: Act as a virtual KISS TNC for use by other packet radio applications. + * This file implements it with a pseudo terminal for Linux only. + * + * Description: It implements the KISS TNC protocol as described in: + * http://www.ka9q.net/papers/kiss.html + * + * Briefly, a frame is composed of + * + * * FEND (0xC0) + * * Contents - with special escape sequences so a 0xc0 + * byte in the data is not taken as end of frame. + * as part of the data. + * * FEND + * + * The first byte of the frame contains: + * + * * port number in upper nybble. + * * command in lower nybble. + * + * + * Commands from application recognized: + * + * _0 Data Frame AX.25 frame in raw format. + * + * _1 TXDELAY See explanation in xmit.c. + * + * _2 Persistence " " + * + * _3 SlotTime " " + * + * _4 TXtail " " + * Spec says it is obsolete but Xastir + * sends it and we respect it. + * + * _5 FullDuplex Ignored. + * + * _6 SetHardware TNC specific. + * + * FF Return Exit KISS mode. Ignored. + * + * + * Messages sent to client application: + * + * _0 Data Frame Received AX.25 frame in raw format. + * + * + * Platform differences: + * + * For the Linux case, + * We supply a pseudo terminal for use by other applications. + * + * Version 1.5: Split serial port version off into its own file. + * + *---------------------------------------------------------------*/ + + +#if __WIN32__ // Stub for Windows. + +#include "direwolf.h" +#include "kiss.h" + +void kisspt_init (struct misc_config_s *mc) +{ + return; +} + +void kisspt_set_debug (int n) +{ + return; +} + +void kisspt_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *kps, int client) +{ + return; +} + + +#else // Rest of file is for Linux only. + + +#include "direwolf.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + + +#include "tq.h" +#include "ax25_pad.h" +#include "textcolor.h" +#include "kiss.h" +#include "kiss_frame.h" +#include "xmit.h" + + +/* + * Accumulated KISS frame and state of decoder. + */ + +static kiss_frame_t kf; + + +/* + * These are for a Linux pseudo terminal. + */ + +static int pt_master_fd = -1; /* File descriptor for my end. */ + +static char pt_slave_name[32]; /* Pseudo terminal slave name */ + /* like /dev/pts/999 */ + + +/* + * Symlink to pseudo terminal name which changes. + */ + +#define TMP_KISSTNC_SYMLINK "/tmp/kisstnc" + + +static void * kisspt_listen_thread (void *arg); + + +static int kisspt_debug = 0; /* Print information flowing from and to client. */ + +void kisspt_set_debug (int n) +{ + kisspt_debug = n; +} + + +/* In server.c. Should probably move to some misc. function file. */ + +void hex_dump (unsigned char *p, int len); + + + + + +/*------------------------------------------------------------------- + * + * Name: kisspt_init + * + * Purpose: Set up a pseudo terminal acting as a virtual KISS TNC. + * + * + * Inputs: + * + * Outputs: + * + * Description: (1) Create a pseudo terminal for the client to use. + * (2) Start a new thread to listen for commands from client app + * so the main application doesn't block while we wait. + * + * + *--------------------------------------------------------------------*/ + +static int kisspt_open_pt (void); + + +void kisspt_init (struct misc_config_s *mc) +{ + + pthread_t kiss_pterm_listen_tid; + int e; + + memset (&kf, 0, sizeof(kf)); + +/* + * This reads messages from client. + */ + pt_master_fd = -1; + + if (mc->enable_kiss_pt) { + + pt_master_fd = kisspt_open_pt (); + + if (pt_master_fd != -1) { + e = pthread_create (&kiss_pterm_listen_tid, (pthread_attr_t*)NULL, kisspt_listen_thread, NULL); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + perror("Could not create kiss listening thread for Linux pseudo terminal"); + } + } + } + else { + //text_color_set(DW_COLOR_INFO); + //dw_printf ("Use -p command line option to enable KISS pseudo terminal.\n"); + } + + +#if DEBUG + text_color_set (DW_COLOR_DEBUG); + + dw_printf ("end of kisspt_init: pt_master_fd = %d\n", pt_master_fd); +#endif + +} + + +/* + * Returns fd for master side of pseudo terminal or -1 for error. + */ + +static int kisspt_open_pt (void) +{ + int fd; + char *pts; + struct termios ts; + int e; + + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kisspt_open_pt ( )\n"); +#endif + + fd = posix_openpt(O_RDWR|O_NOCTTY); + + if (fd == -1 + || grantpt (fd) == -1 + || unlockpt (fd) == -1 + || (pts = ptsname (fd)) == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR - Could not create pseudo terminal for KISS TNC.\n"); + return (-1); + } + + strlcpy (pt_slave_name, pts, sizeof(pt_slave_name)); + + e = tcgetattr (fd, &ts); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't get pseudo terminal attributes, err=%d\n", e); + perror ("pt tcgetattr"); + } + + cfmakeraw (&ts); + + ts.c_cc[VMIN] = 1; /* wait for at least one character */ + ts.c_cc[VTIME] = 0; /* no fancy timing. */ + + + e = tcsetattr (fd, TCSANOW, &ts); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't set pseudo terminal attributes, err=%d\n", e); + perror ("pt tcsetattr"); + } + +/* + * We had a problem here since the beginning. + * If no one was reading from the other end of the pseudo + * terminal, the buffer space would eventually fill up, + * the write here would block, and the receive decode + * thread would get stuck. + * + * March 2016 - A "select" was put before the read to + * solve a different problem. With that in place, we can + * now use non-blocking I/O and detect the buffer full + * condition here. + */ + + // text_color_set(DW_COLOR_DEBUG); + // dw_printf("Debug: Try using non-blocking mode for pseudo terminal.\n"); + + int flags = fcntl(fd, F_GETFL, 0); + e = fcntl (fd, F_SETFL, flags | O_NONBLOCK); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't set pseudo terminal to nonblocking, fcntl returns %d, errno = %d\n", e, errno); + perror ("pt fcntl"); + } + + text_color_set(DW_COLOR_INFO); + dw_printf("Virtual KISS TNC is available on %s\n", pt_slave_name); + + +#if 1 + // Sample code shows this. Why would we open it here? + // On Ubuntu, the slave side disappears after a few + // seconds if no one opens it. Same on Raspbian which + // is also based on Debian. + // Need to revisit this. + + int pt_slave_fd; + + pt_slave_fd = open(pt_slave_name, O_RDWR|O_NOCTTY); + + if (pt_slave_fd < 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Can't open %s\n", pt_slave_name); + perror (""); + return -1; + } +#endif + +/* + * The device name is not the same every time. + * This is inconvenient for the application because it might + * be necessary to change the device name in the configuration. + * Create a symlink, /tmp/kisstnc, so the application configuration + * does not need to change when the pseudo terminal name changes. + */ + + unlink (TMP_KISSTNC_SYMLINK); + + +// TODO: Is this removed when application exits? + + if (symlink (pt_slave_name, TMP_KISSTNC_SYMLINK) == 0) { + dw_printf ("Created symlink %s -> %s\n", TMP_KISSTNC_SYMLINK, pt_slave_name); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Failed to create symlink %s\n", TMP_KISSTNC_SYMLINK); + perror (""); + } + + return (fd); +} + + + +/*------------------------------------------------------------------- + * + * Name: kisspt_send_rec_packet + * + * Purpose: Send a received packet or text string to the client app. + * + * Inputs: chan - Channel number where packet was received. + * 0 = first, 1 = second if any. + * + * kiss_cmd - Usually KISS_CMD_DATA_FRAME but we can also have + * KISS_CMD_SET_HARDWARE when responding to a query. + * + * pp - Identifier for packet object. + * + * fbuf - Address of raw received frame buffer + * or a text string. + * + * flen - Length of raw received frame not including the FCS + * or -1 for a text string. + * + * kps, client - Not used for pseudo terminal. + * Here so that 3 related functions all have + * the same parameter list. + * + * Description: Send message to client. + * We really don't care if anyone is listening or not. + * I don't even know if we can find out. + * + *--------------------------------------------------------------------*/ + + +void kisspt_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *kps, int client) +{ + unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN + 2]; + int kiss_len; + int err; + + + if (pt_master_fd == -1) { + return; + } + + if (flen < 0) { + flen = strlen((char*)fbuf); + if (kisspt_debug) { + kiss_debug_print (TO_CLIENT, "Fake command prompt", fbuf, flen); + } + strlcpy ((char *)kiss_buff, (char *)fbuf, sizeof(kiss_buff)); + kiss_len = strlen((char *)kiss_buff); + } + else { + + unsigned char stemp[AX25_MAX_PACKET_LEN + 1]; + + if (flen > (int)(sizeof(stemp)) - 1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nPseudo Terminal KISS buffer too small. Truncated.\n\n"); + flen = (int)(sizeof(stemp)) - 1; + } + + stemp[0] = (chan << 4) | kiss_cmd; + memcpy (stemp+1, fbuf, flen); + + if (kisspt_debug >= 2) { + /* AX.25 frame with the CRC removed. */ + text_color_set(DW_COLOR_DEBUG); + dw_printf ("\n"); + dw_printf ("Packet content before adding KISS framing and any escapes:\n"); + hex_dump (fbuf, flen); + } + + kiss_len = kiss_encapsulate (stemp, flen+1, kiss_buff); + + /* This has KISS framing and escapes for sending to client app. */ + + if (kisspt_debug) { + kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len); + } + + } + + err = write (pt_master_fd, kiss_buff, (size_t)kiss_len); + + if (err == -1 && errno == EWOULDBLOCK) { + text_color_set (DW_COLOR_INFO); + dw_printf ("KISS SEND - Discarding message because no one is listening.\n"); + dw_printf ("This happens when you use the -p option and don't read from the pseudo terminal.\n"); + } + else if (err != kiss_len) + { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError sending KISS message to client application on pseudo terminal. fd=%d, len=%d, write returned %d, errno = %d\n\n", + pt_master_fd, kiss_len, err, errno); + perror ("pt write"); + } + +} /* kisspt_send_rec_packet */ + + + + +/*------------------------------------------------------------------- + * + * Name: kisspt_get + * + * Purpose: Read one byte from the KISS client app. + * + * Global In: pt_master_fd + * + * Returns: one byte (value 0 - 255) or terminate thread on error. + * + * Description: There is room for improvement here. Reading one byte + * at a time is inefficient. We could read a large block + * into a local buffer and return a byte from that most of the time. + * Is it worth the effort? I don't know. With GHz processors and + * the low data rate here it might not make a noticeable difference. + * + *--------------------------------------------------------------------*/ + + +static int kisspt_get (void) +{ + unsigned char ch; + + int n = 0; + fd_set fd_in, fd_ex; + int rc; + + while ( n == 0 ) { + +/* + * Since the beginning we've always had a couple annoying problems with + * the pseudo terminal KISS interface. + * When using "kissattach" we would sometimes get the error message: + * + * kissattach: Error setting line discipline: TIOCSETD: Device or resource busy + * Are you sure you have enabled MKISS support in the kernel + * or, if you made it a module, that the module is loaded? + * + * martinhpedersen came up with the interesting idea of putting in a "select" + * before the "read" and explained it like this: + * + * "Reading from master fd of the pty before the client has connected leads + * to trouble with kissattach. Use select to check if the slave has sent + * any data before trying to read from it." + * + * "This fix resolves the issue by not reading from the pty's master fd, until + * kissattach has opened and configured the slave. This is implemented using + * select() to wait for data before reading from the master fd." + * + * The submitted code looked like this: + * + * FD_ZERO(&fd_in); + * rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_in, NULL); + * + * That doesn't look right to me for a couple reasons. + * First, I would expect to use FD_SET for the fd. + * Second, using the same bit mask for two arguments doesn't seem + * like a good idea because select modifies them. + * When I tried running it, we don't get the failure message + * anymore but the select never returns so we can't read data from + * the KISS client app. + * + * I think this is what we want. + * + * Tested on Raspian (ARM) and Ubuntu (x86_64). + * We don't get the error from kissattach anymore. + */ + + FD_ZERO(&fd_in); + FD_SET(pt_master_fd, &fd_in); + + FD_ZERO(&fd_ex); + FD_SET(pt_master_fd, &fd_ex); + + rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_ex, NULL); + +#if 0 + text_color_set(DW_COLOR_DEBUG); + dw_printf ("select returns %d, errno=%d, fd=%d, fd_in=%08x, fd_ex=%08x\n", rc, errno, pt_master_fd, *((int*)(&fd_in)), *((int*)(&fd_in))); +#endif + + if (rc == 0) + { + continue; // When could we get a 0? + } + + if (rc == -1 + || (n = read(pt_master_fd, &ch, (size_t)1)) != 1) + { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError receiving KISS message from client application. Closing %s.\n\n", pt_slave_name); + perror (""); + + close (pt_master_fd); + + pt_master_fd = -1; + unlink (TMP_KISSTNC_SYMLINK); + pthread_exit (NULL); + } + } + +#if DEBUGx + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kisspt_get(%d) returns 0x%02x\n", fd, ch); +#endif + + return (ch); +} + + +/*------------------------------------------------------------------- + * + * Name: kisspt_listen_thread + * + * Purpose: Read messages from serial port KISS client application. + * + * Global In: + * + * Description: Reads bytes from the KISS client app and + * sends them to kiss_rec_byte for processing. + * + *--------------------------------------------------------------------*/ + +static void * kisspt_listen_thread (void *arg) +{ + unsigned char ch; + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kisspt_listen_thread ( %d )\n", fd); +#endif + + + while (1) { + ch = kisspt_get(); + kiss_rec_byte (&kf, ch, kisspt_debug, NULL, -1, kisspt_send_rec_packet); + } + + return (void *) 0; /* Unreachable but avoids compiler warning. */ +} + +#endif // Linux version + +/* end kiss.c */ diff --git a/src/kiss.h b/src/kiss.h new file mode 100644 index 00000000..1dc40daf --- /dev/null +++ b/src/kiss.h @@ -0,0 +1,24 @@ + +/* + * Name: kiss.h + * + * This is for the pseudo terminal KISS interface. + */ + + +#include "ax25_pad.h" /* for packet_t */ + +#include "config.h" + +#include "kiss_frame.h" // for struct kissport_status_s + + +void kisspt_init (struct misc_config_s *misc_config); + +void kisspt_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, + struct kissport_status_s *notused1, int notused2); + +void kisspt_set_debug (int n); + + +/* end kiss.h */ diff --git a/src/kiss_frame.c b/src/kiss_frame.c new file mode 100644 index 00000000..aa581dd2 --- /dev/null +++ b/src/kiss_frame.c @@ -0,0 +1,1023 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2013, 2014, 2017, 2023 John Langner, WB2OSZ +// +// 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, see . +// + + + +/*------------------------------------------------------------------ + * + * Module: kiss_frame.c + * + * Purpose: Common code used by Serial port and network versions of KISS protocol. + * + * Description: The KISS TNC protocol is described in http://www.ka9q.net/papers/kiss.html + * + * ( An extended form, to handle multiple TNCs on a single serial port. + * Not applicable for our situation. http://he.fi/pub/oh7lzb/bpq/multi-kiss.pdf ) + * + * Briefly, a frame is composed of + * + * * FEND (0xC0) + * * Contents - with special escape sequences so a 0xc0 + * byte in the data is not taken as end of frame. + * as part of the data. + * * FEND + * + * The first byte of the frame contains: + * + * * radio channel in upper nybble. + * (KISS doc uses "port" but I don't like that because it has too many meanings.) + * * command in lower nybble. + * + * + * Commands from application tp TNC: + * + * _0 Data Frame AX.25 frame in raw format. + * + * _1 TXDELAY See explanation in xmit.c. + * + * _2 Persistence " " + * + * _3 SlotTime " " + * + * _4 TXtail " " + * Spec says it is obsolete but Xastir + * sends it and we respect it. + * + * _5 FullDuplex Full Duplex. Transmit immediately without + * waiting for channel to be clear. + * + * _6 SetHardware TNC specific. + * + * _C XKISS extension - not supported. + * _E XKISS extension - not supported. + * + * FF Return Exit KISS mode. Ignored. + * + * + * Messages sent to client application: + * + * _0 Data Frame Received AX.25 frame in raw format. + * + * _6 SetHardware TNC specific. + * Usually a response to a query. + * + *---------------------------------------------------------------*/ + +#include "direwolf.h" + +#include +#include +#include +#include +#include +#include + +#include "ax25_pad.h" +#include "textcolor.h" +#include "kiss_frame.h" +#include "tq.h" +#include "xmit.h" +#include "version.h" +#include "kissnet.h" + + +/* In server.c. Should probably move to some misc. function file. */ +void hex_dump (unsigned char *p, int len); + +#ifdef KISSUTIL +void hex_dump (unsigned char *p, int len) +{ + int n, i, offset; + + offset = 0; + while (len > 0) { + n = len < 16 ? len : 16; + // FIXME: Is there some reason not to use dw_printf here? + printf (" %03x: ", offset); + for (i=0; i + * <0x0d> + * XFLOW OFF<0x0d> + * FULLDUP OFF<0x0d> + * KISS ON<0x0d> + * RESTART<0x0d> + * <0x03><0x03><0x03> + * TC 1<0x0d> + * TN 2,0<0x0d><0x0d><0x0d> + * XFLOW OFF<0x0d> + * FULLDUP OFF<0x0d> + * KISS ON<0x0d> + * RESTART<0x0d> + * + * This keeps repeating over and over and over and over again if + * it doesn't get any sort of response. + * + * Let's try to keep it happy by sending back a command prompt. + */ + + + + +void kiss_rec_byte (kiss_frame_t *kf, unsigned char ch, int debug, + struct kissport_status_s *kps, int client, + void (*sendfun)(int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *onlykps, int onlyclient)) +{ + + //dw_printf ("kiss_frame ( %c %02x ) \n", ch, ch); + + switch (kf->state) { + + case KS_SEARCHING: /* Searching for starting FEND. */ + default: + + if (ch == FEND) { + + /* Start of frame. But first print any collected noise for debugging. */ + + if (kf->noise_len > 0) { + if (debug) { + kiss_debug_print (FROM_CLIENT, "Rejected Noise", kf->noise, kf->noise_len); + } + kf->noise_len = 0; + } + + kf->kiss_len = 0; + kf->kiss_msg[kf->kiss_len++] = ch; + kf->state = KS_COLLECTING; + return; + } + + /* Noise to be rejected. */ + + if (kf->noise_len < MAX_NOISE_LEN) { + kf->noise[kf->noise_len++] = ch; + } + if (ch == '\r') { + if (debug) { + kiss_debug_print (FROM_CLIENT, "Rejected Noise", kf->noise, kf->noise_len); + kf->noise[kf->noise_len] = '\0'; + } + +#ifndef KISSUTIL + /* Try to appease client app by sending something back. */ + if (strcasecmp("restart\r", (char*)(kf->noise)) == 0 || + strcasecmp("reset\r", (char*)(kf->noise)) == 0) { + // first 2 parameters don't matter when length is -1 indicating text. + (*sendfun) (0, 0, (unsigned char *)"\xc0\xc0", -1, kps, client); + } + else { + (*sendfun) (0, 0, (unsigned char *)"\r\ncmd:", -1, kps, client); + } +#endif + kf->noise_len = 0; + } + return; + break; + + case KS_COLLECTING: /* Frame collection in progress. */ + + + if (ch == FEND) { + + unsigned char unwrapped[AX25_MAX_PACKET_LEN]; + int ulen; + + /* End of frame. */ + + if (kf->kiss_len == 0) { + /* Empty frame. Starting a new one. */ + kf->kiss_msg[kf->kiss_len++] = ch; + return; + } + if (kf->kiss_len == 1 && kf->kiss_msg[0] == FEND) { + /* Empty frame. Just go on collecting. */ + return; + } + + kf->kiss_msg[kf->kiss_len++] = ch; + if (debug) { + /* As received over the wire from client app. */ + kiss_debug_print (FROM_CLIENT, NULL, kf->kiss_msg, kf->kiss_len); + } + + ulen = kiss_unwrap (kf->kiss_msg, kf->kiss_len, unwrapped); + + if (debug >= 2) { + /* Append CRC to this and it goes out over the radio. */ + text_color_set(DW_COLOR_DEBUG); + dw_printf ("\n"); + dw_printf ("Packet content after removing KISS framing and any escapes:\n"); + /* Don't include the "type" indicator. */ + /* It contains the radio channel and type should always be 0 here. */ + hex_dump (unwrapped+1, ulen-1); + } + + kiss_process_msg (unwrapped, ulen, debug, kps, client, sendfun); + + kf->state = KS_SEARCHING; + return; + } + + if (kf->kiss_len < MAX_KISS_LEN) { + kf->kiss_msg[kf->kiss_len++] = ch; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS message exceeded maximum length.\n"); + } + return; + break; + } + + return; /* unreachable but suppress compiler warning. */ + +} /* end kiss_rec_byte */ + + + + +/*------------------------------------------------------------------- + * + * Name: kiss_process_msg + * + * Purpose: Process a message from the KISS client. + * + * Inputs: kiss_msg - Kiss frame with FEND and escapes removed. + * The first byte contains channel and command. + * + * kiss_len - Number of bytes including the command. + * + * debug - Debug option is selected. + * + * kps - Used only for TCP KISS. + * Should be NULL for pseudo terminal and serial port. + * + * client - Client app number for TCP KISS. + * Should be -1 for pseudo termal and serial port. + * + * sendfun - Function to send something to the client application. + * "Set Hardware" can send a response. + * + *-----------------------------------------------------------------*/ + +#ifndef KISSUTIL // All these ifdefs in here are a sign that this should be refactored. + // Should split this into multiple files. + // Some functions are only for the TNC end. + // Other functions are suitble for both TNC and client app. + +// This is used only by the TNC side. + +void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, struct kissport_status_s *kps, int client, + void (*sendfun)(int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *kps, int client)) +{ + int chan; + int cmd; + alevel_t alevel; + +// New in 1.7: +// We can have KISS TCP ports which convey only a single radio channel. +// This is to allow operation by applications which only know how to talk to single radio TNCs. + + if (kps != NULL && kps->chan != -1) { + // Ignore channel from KISS and substitute radio channel for that KISS TCP port. + chan = kps->chan; + } + else { + // Normal case of getting radio channel from the KISS frame. + chan = (kiss_msg[0] >> 4) & 0xf; + } + cmd = kiss_msg[0] & 0xf; + + switch (cmd) + { + case KISS_CMD_DATA_FRAME: /* 0 = Data Frame */ + + // kissnet_copy clobbers first byte but we don't care + // because we have already determined channel and command. + + kissnet_copy (kiss_msg, kiss_len, chan, cmd, kps, client); + + /* Note July 2017: There is a variant of of KISS, called SMACK, that assumes */ + /* a TNC can never have more than 8 channels. http://symek.de/g/smack.html */ + /* It uses the MSB to indicate that a checksum is added. I wonder if this */ + /* is why we sometimes hear about a request to transmit on channel 8. */ + /* Should we have a message that asks the user if SMACK is being used, */ + /* and if so, turn it off in the application configuration? */ + /* Our current default is a maximum of 6 channels but it is easily */ + /* increased by changing one number and recompiling. */ + +// Additional information, from Mike Playle, December 2018, for Issue #42 +// +// I came across this the other day with Xastir, and took a quick look. +// The problem is fixable without the kiss_frame.c hack, which doesn't help with Xastir anyway. +// +// Workaround +// +// After the kissattach command, put the interface into CRC mode "none" with a command like this: +// +// # kissparms -c 1 -p radio +// +// Analysis +// +// The source of this behaviour is the kernel's KISS implementation: +// +// https://elixir.bootlin.com/linux/v4.9/source/drivers/net/hamradio/mkiss.c#L489 +// +// It defaults to starting in state CRC_MODE_SMACK_TEST and ending up in mode CRC_NONE +// after the first two packets, which have their framing byte modified by this code in the process. +// It looks to me like deliberate behaviour on the kernel's part. +// +// Setting the CRC mode explicitly before sending any packets stops this state machine from running. +// +// Is this a bug? I don't know - that's up to you! Maybe it would make sense for Direwolf to set +// the CRC mode itself, or to expect this behaviour and ignore these flags on the first packets +// received from the Linux pty. +// +// This workaround seems sound to me, though, so perhaps this is just a documentation issue. + + +// Would it make sense to implement SMACK? I don't think so. +// Adding a checksum to the KISS data offers no benefit because it is very reliable. +// It violates the original protocol specification which states that 16 radio channels are possible. +// (Some times the term 'port' is used but I try to use 'channel' all the time because 'port' +// has too many other meanings. Serial port, TCP port, ...) +// SMACK imposes a limit of 8. That limit might have been OK back in 1991 but not now. +// There are people using more than 8 radio channels (using SDR not traditional radios) with direwolf. + + + /* Verify that the radio channel number is valid. */ + /* Any sort of medium should be OK here. */ + + if ((chan < 0 || chan >= MAX_CHANS || save_audio_config_p->chan_medium[chan] == MEDIUM_NONE) + && save_audio_config_p->chan_medium[chan] != MEDIUM_IGATE) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid transmit channel %d from KISS client app.\n", chan); + dw_printf ("\n"); + dw_printf ("Are you using AX.25 for Linux? It might be trying to use a modified\n"); + dw_printf ("version of KISS which uses the channel field differently than the\n"); + dw_printf ("original KISS protocol specification. The solution might be to use\n"); + dw_printf ("a command like \"kissparms -c 1 -p radio\" to set CRC none mode.\n"); + dw_printf ("Another way of doing this is pre-loading the \"kiss\" kernel module with CRC disabled:\n"); + dw_printf ("sudo /sbin/modprobe -q mkiss crc_force=1\n"); + + dw_printf ("\n"); + text_color_set(DW_COLOR_DEBUG); + kiss_debug_print (FROM_CLIENT, NULL, kiss_msg, kiss_len); + return; + } + + memset (&alevel, 0xff, sizeof(alevel)); + packet_t pp = ax25_from_frame (kiss_msg+1, kiss_len-1, alevel); + if (pp == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR - Invalid KISS data frame from client app.\n"); + } + else { + + /* How can we determine if it is an original or repeated message? */ + /* If there is at least one digipeater in the frame, AND */ + /* that digipeater has been used, it should go out quickly thru */ + /* the high priority queue. */ + /* Otherwise, it is an original for the low priority queue. */ + + if (ax25_get_num_repeaters(pp) >= 1 && + ax25_get_h(pp,AX25_REPEATER_1)) { + tq_append (chan, TQ_PRIO_0_HI, pp); + } + else { + tq_append (chan, TQ_PRIO_1_LO, pp); + } + } + break; + + case KISS_CMD_TXDELAY: /* 1 = TXDELAY */ + + if (kiss_len < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS ERROR: Missing value for TXDELAY command.\n"); + return; + } + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol set TXDELAY = %d (*10mS units = %d mS), chan %d\n", kiss_msg[1], kiss_msg[1] * 10, chan); + if (kiss_msg[1] < 4 || kiss_msg[1] > 100) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Are you sure you want such an extreme value for TXDELAY?\n"); + dw_printf ("See \"Radio Channel - Transmit Timing\" section of User Guide for explanation.\n"); + } + xmit_set_txdelay (chan, kiss_msg[1]); + break; + + case KISS_CMD_PERSISTENCE: /* 2 = Persistence */ + + if (kiss_len < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS ERROR: Missing value for PERSISTENCE command.\n"); + return; + } + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol set Persistence = %d, chan %d\n", kiss_msg[1], chan); + if (kiss_msg[1] < 5 || kiss_msg[1] > 250) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Are you sure you want such an extreme value for PERSIST?\n"); + dw_printf ("See \"Radio Channel - Transmit Timing\" section of User Guide for explanation.\n"); + } + xmit_set_persist (chan, kiss_msg[1]); + break; + + case KISS_CMD_SLOTTIME: /* 3 = SlotTime */ + + if (kiss_len < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS ERROR: Missing value for SLOTTIME command.\n"); + return; + } + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol set SlotTime = %d (*10mS units = %d mS), chan %d\n", kiss_msg[1], kiss_msg[1] * 10, chan); + if (kiss_msg[1] < 2 || kiss_msg[1] > 50) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Are you sure you want such an extreme value for SLOTTIME?\n"); + dw_printf ("See \"Radio Channel - Transmit Timing\" section of User Guide for explanation.\n"); + } + xmit_set_slottime (chan, kiss_msg[1]); + break; + + case KISS_CMD_TXTAIL: /* 4 = TXtail */ + + if (kiss_len < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS ERROR: Missing value for TXTAIL command.\n"); + return; + } + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol set TXtail = %d (*10mS units = %d mS), chan %d\n", kiss_msg[1], kiss_msg[1] * 10, chan); + if (kiss_msg[1] < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Setting TXTAIL so low is asking for trouble. You probably don't want to do this.\n"); + dw_printf ("See \"Radio Channel - Transmit Timing\" section of User Guide for explanation.\n"); + } + xmit_set_txtail (chan, kiss_msg[1]); + break; + + case KISS_CMD_FULLDUPLEX: /* 5 = FullDuplex */ + + if (kiss_len < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS ERROR: Missing value for FULLDUPLEX command.\n"); + return; + } + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol set FullDuplex = %d, chan %d\n", kiss_msg[1], chan); + xmit_set_fulldup (chan, kiss_msg[1]); + break; + + case KISS_CMD_SET_HARDWARE: /* 6 = TNC specific */ + + if (kiss_len < 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS ERROR: Missing value for SET HARDWARE command.\n"); + return; + } + kiss_msg[kiss_len] = '\0'; + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol set hardware \"%s\", chan %d\n", (char*)(kiss_msg+1), chan); + kiss_set_hardware (chan, (char*)(kiss_msg+1), debug, kps, client, sendfun); + break; + + case KISS_CMD_END_KISS: /* 15 = End KISS mode, channel should be 15. */ + /* Ignore it. */ + text_color_set(DW_COLOR_INFO); + dw_printf ("KISS protocol end KISS mode - Ignored.\n"); + break; + + default: + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS Invalid command %d\n", cmd); + kiss_debug_print (FROM_CLIENT, NULL, kiss_msg, kiss_len); + + text_color_set(DW_COLOR_INFO); + dw_printf ("Troubleshooting tip:\n"); + dw_printf ("Use \"-d kn\" option on direwolf command line to observe\n"); + dw_printf ("all communication with the client application.\n"); + + if (cmd == XKISS_CMD_DATA || cmd == XKISS_CMD_POLL) { + dw_printf ("\n"); + dw_printf ("It looks like you are trying to use the \"XKISS\" protocol which is not supported.\n"); + dw_printf ("Change your application settings to use standard \"KISS\" rather than some other variant.\n"); + dw_printf ("If you are using Winlink Express, configure like this:\n"); + dw_printf (" Packet TNC Type: KISS\n"); + dw_printf (" Packet TNC Model: NORMAL -- Using ACKMODE will cause this error.\n"); + dw_printf ("\n"); + } + break; + } + +} /* end kiss_process_msg */ + +#endif // ifndef KISSUTIL + + +/*------------------------------------------------------------------- + * + * Name: kiss_set_hardware + * + * Purpose: Process the "set hardware" command. + * + * Inputs: chan - channel, 0 - 15. + * + * command - All but the first byte. e.g. "TXBUF:99" + * Case sensitive. + * Will be modified so be sure caller doesn't care. + * + * debug - debug level. + * + * client - Client app number for TCP KISS. + * Needed so we can send any response to the right client app. + * Ignored for pseudo terminal and serial port. + * + * sendfun - Function to send something to the client application. + * + * This is the tricky part. We can have any combination of + * serial port, pseudo terminal, and multiple TCP clients. + * We need to send the response to same place where query came + * from. The function is different for each class of device + * and we need a client number for the TCP case because we + * can have multiple TCP KISS clients at the same time. + * + * + * Description: This is new in version 1.5. "Set hardware" was previously ignored. + * + * There are times when the client app might want to send configuration + * commands, such as modem speed, to the KISS TNC or inquire about its + * current state. + * + * The immediate motivation for adding this is that one application wants + * to know how many frames are currently in the transmit queue. This can + * be used for throttling of large transmissions and performing some action + * after the last frame has been sent. + * + * The original KISS protocol spec offers no guidance on what "Set Hardware" might look + * like. I'm aware of only two, drastically different, implementations: + * + * fldigi - http://www.w1hkj.com/FldigiHelp-3.22/kiss_command_page.html + * + * Everything is in human readable in both directions: + * + * COMMAND: [ parameter [ , parameter ... ] ] + * + * Lack of a parameter, in the client to TNC direction, is a query + * which should generate a response in the same format. + * + * Used by applications, http://www.w1hkj.com/FldigiHelp/kiss_host_prgs_page.html + * - BPQ32 + * - UIChar + * - YAAC + * + * mobilinkd - https://raw.githubusercontent.com/mobilinkd/tnc1/tnc2/bertos/net/kiss.c + * + * Single byte with the command / response code, followed by + * zero or more value bytes. + * + * Used by applications: + * - APRSdroid + * + * It would be beneficial to adopt one of them rather than doing something + * completely different. It might even be possible to recognize both. + * This might allow leveraging of other existing applications. + * + * Let's start with the easy to understand human readable format. + * + * Commands: (Client to TNC, with parameter(s) to set something.) + * + * none yet + * + * Queries: (Client to TNC, no parameters, generate a response.) + * + * Query Response Comment + * ----- -------- ------- + * + * TNC: TNC:DIREWOLF 9.9 9.9 represents current version. + * + * TXBUF: TXBUF:999 Number of bytes (not frames) in transmit queue. + * + *--------------------------------------------------------------------*/ + +#ifndef KISSUTIL + +static void kiss_set_hardware (int chan, char *command, int debug, struct kissport_status_s *kps, int client, + void (*sendfun)(int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *onlykps, int onlyclient)) +{ + char *param; + char response[100]; + + param = strchr (command, ':'); + if (param != NULL) { + *param = '\0'; + param++; + + if (strcmp(command, "TNC") == 0) { /* TNC - Identify software version. */ + + if (strlen(param) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS Set Hardware TNC: Did not expect a parameter.\n"); + } + + snprintf (response, sizeof(response), "DIREWOLF %d.%d", MAJOR_VERSION, MINOR_VERSION); + (*sendfun) (chan, KISS_CMD_SET_HARDWARE, (unsigned char *)response, strlen(response), kps, client); + } + + else if (strcmp(command, "TXBUF") == 0) { /* TXBUF - Number of bytes in transmit queue. */ + + if (strlen(param) > 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS Set Hardware TXBUF: Did not expect a parameter.\n"); + } + + int n = tq_count (chan, -1, "", "", 1); + snprintf (response, sizeof(response), "TXBUF:%d", n); + (*sendfun) (chan, KISS_CMD_SET_HARDWARE, (unsigned char *)response, strlen(response), kps, client); + } + + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS Set Hardware unrecognized command: %s.\n", command); + } + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS Set Hardware \"%s\" expected the form COMMAND:[parameter[,parameter...]]\n", command); + } + return; + +} /* end kiss_set_hardware */ + +#endif // ifndef KISSUTIL + + +/*------------------------------------------------------------------- + * + * Name: kiss_debug_print + * + * Purpose: Print message to/from client for debugging. + * + * Inputs: fromto - Direction of message. + * special - Comment if not a KISS frame. + * pmsg - Address of the message block. + * msg_len - Length of the message. + * + *--------------------------------------------------------------------*/ + + +void kiss_debug_print (fromto_t fromto, char *special, unsigned char *pmsg, int msg_len) +{ +#ifndef KISSUTIL + const char *direction [2] = { "from", "to" }; + const char *prefix [2] = { "<<<", ">>>" }; + const char *function[16] = { + "Data frame", "TXDELAY", "P", "SlotTime", + "TXtail", "FullDuplex", "SetHardware", "Invalid 7", + "Invalid 8", "Invalid 9", "Invalid 10", "Invalid 11", + "Invalid 12", "Invalid 13", "Invalid 14", "Return" }; +#endif + + text_color_set(DW_COLOR_DEBUG); + +#ifdef KISSUTIL + dw_printf ("From KISS TNC:\n"); +#else + dw_printf ("\n"); + if (special == NULL) { + unsigned char *p; /* to skip over FEND if present. */ + + p = pmsg; + if (*p == FEND) p++; + + dw_printf ("%s %s %s KISS client application, channel %d, total length = %d\n", + prefix[(int)fromto], function[p[0] & 0xf], direction[(int)fromto], + (p[0] >> 4) & 0xf, msg_len); + } + else { + dw_printf ("%s %s %s KISS client application, total length = %d\n", + prefix[(int)fromto], special, direction[(int)fromto], + msg_len); + } +#endif + hex_dump (pmsg, msg_len); + +} /* end kiss_debug_print */ + + +#endif + +#endif /* DECAMAIN */ + +/* Quick unit test for encapsulate & unwrap */ + +// $ gcc -DKISSTEST kiss_frame.c ; ./a +// Quick KISS test passed OK. + + +#if KISSTEST + + +int main () +{ + unsigned char din[512]; + unsigned char kissed[520]; + unsigned char dout[520]; + int klen; + int dlen; + int k; + + for (k = 0; k < 512; k++) { + if (k < 256) { + din[k] = k; + } + else { + din[k] = 511 - k; + } + } + + klen = kiss_encapsulate (din, 512, kissed); + assert (klen == 512 + 6); + + dlen = kiss_unwrap (kissed, klen, dout); + assert (dlen == 512); + assert (memcmp(din, dout, 512) == 0); + + dlen = kiss_unwrap (kissed+1, klen-1, dout); + assert (dlen == 512); + assert (memcmp(din, dout, 512) == 0); + + dw_printf ("Quick KISS test passed OK.\n"); + exit (EXIT_SUCCESS); +} + +#endif /* KISSTEST */ + +#endif /* WALK96 */ + +/* end kiss_frame.c */ diff --git a/src/kiss_frame.h b/src/kiss_frame.h new file mode 100644 index 00000000..941f5c01 --- /dev/null +++ b/src/kiss_frame.h @@ -0,0 +1,126 @@ + +/* kiss_frame.h */ + +#ifndef KISS_FRAME_H +#define KISS_FRAME_H + + +#include "audio.h" /* for struct audio_s */ + + +/* + * The first byte of a KISS frame has: + * channel in upper nybble. + * command in lower nybble. + */ + +#define KISS_CMD_DATA_FRAME 0 +#define KISS_CMD_TXDELAY 1 +#define KISS_CMD_PERSISTENCE 2 +#define KISS_CMD_SLOTTIME 3 +#define KISS_CMD_TXTAIL 4 +#define KISS_CMD_FULLDUPLEX 5 +#define KISS_CMD_SET_HARDWARE 6 +#define XKISS_CMD_DATA 12 // Not supported. http://he.fi/pub/oh7lzb/bpq/multi-kiss.pdf +#define XKISS_CMD_POLL 14 // Not supported. +#define KISS_CMD_END_KISS 15 + + + +/* + * Special characters used by SLIP protocol. + */ + +#define FEND 0xC0 +#define FESC 0xDB +#define TFEND 0xDC +#define TFESC 0xDD + + + +enum kiss_state_e { + KS_SEARCHING = 0, /* Looking for FEND to start KISS frame. */ + /* Must be 0 so we can simply zero whole structure to initialize. */ + KS_COLLECTING}; /* In process of collecting KISS frame. */ + + +#define MAX_KISS_LEN 2048 /* Spec calls for at least 1024. */ + /* Might want to make it longer to accommodate */ + /* maximum packet length. */ + +#define MAX_NOISE_LEN 100 + +typedef struct kiss_frame_s { + + enum kiss_state_e state; + + unsigned char kiss_msg[MAX_KISS_LEN]; + /* Leading FEND is optional. */ + /* Contains escapes and ending FEND. */ + int kiss_len; + + unsigned char noise[MAX_NOISE_LEN]; + int noise_len; + +} kiss_frame_t; + + +// This is used only for TCPKISS but it put in kissnet.h, +// there would be a circular dependency between the two header files. +// Each KISS TCP port has its own status block. + +struct kissport_status_s { + + struct kissport_status_s *pnext; // To next in list. + + volatile int arg2; // temp for passing second arg into + // kissnet_listen_thread + + int tcp_port; // default 8001 + + int chan; // Radio channel for this tcp port. + // -1 for all. + + // The default is a limit of 3 client applications at the same time. + // You can increase the limit by changing the line below. + // A larger number consumes more resources so don't go crazy by making it larger than needed. + // TODO: Should this be moved to direwolf.h so max number of audio devices + // client apps are in the same place? + +#define MAX_NET_CLIENTS 3 + + int client_sock[MAX_NET_CLIENTS]; + /* File descriptor for socket for */ + /* communication with client application. */ + /* Set to -1 if not connected. */ + /* (Don't use SOCKET type because it is unsigned.) */ + + kiss_frame_t kf[MAX_NET_CLIENTS]; + /* Accumulated KISS frame and state of decoder. */ +}; + + + +#ifndef KISSUTIL +void kiss_frame_init (struct audio_s *pa); +#endif + +int kiss_encapsulate (unsigned char *in, int ilen, unsigned char *out); + +int kiss_unwrap (unsigned char *in, int ilen, unsigned char *out); + +void kiss_rec_byte (kiss_frame_t *kf, unsigned char ch, int debug, struct kissport_status_s *kps, int client, + void (*sendfun)(int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *onlykps, int onlyclient)); + +typedef enum fromto_e { FROM_CLIENT=0, TO_CLIENT=1 } fromto_t; + +void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, struct kissport_status_s *kps, int client, + void (*sendfun)(int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *onlykps, int onlyclient)); + +void kiss_debug_print (fromto_t fromto, char *special, unsigned char *pmsg, int msg_len); + + +#endif // KISS_FRAME_H + + +/* end kiss_frame.h */ diff --git a/src/kissnet.c b/src/kissnet.c new file mode 100644 index 00000000..97094a08 --- /dev/null +++ b/src/kissnet.c @@ -0,0 +1,994 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// Copyright (C) 2011-2014, 2015, 2017, 2021 John Langner, WB2OSZ +// +// 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, see . +// + + +/*------------------------------------------------------------------ + * + * Module: kissnet.c + * + * Purpose: Provide service to other applications via KISS protocol via TCP socket. + * + * Input: + * + * Outputs: + * + * Description: This provides a TCP socket for communication with a client application. + * + * It implements the KISS TNS protocol as described in: + * http://www.ka9q.net/papers/kiss.html + * + * Briefly, a frame is composed of + * + * * FEND (0xC0) + * * Contents - with special escape sequences so a 0xc0 + * byte in the data is not taken as end of frame. + * as part of the data. + * * FEND + * + * The first byte of the frame contains: + * + * * port number in upper nybble. + * * command in lower nybble. + * + * + * Commands from application recognized: + * + * _0 Data Frame AX.25 frame in raw format. + * + * _1 TXDELAY See explanation in xmit.c. + * + * _2 Persistence " " + * + * _3 SlotTime " " + * + * _4 TXtail " " + * Spec says it is obsolete but Xastir + * sends it and we respect it. + * + * _5 FullDuplex Ignored. + * + * _6 SetHardware TNC specific. + * + * FF Return Exit KISS mode. Ignored. + * + * + * Messages sent to client application: + * + * _0 Data Frame Received AX.25 frame in raw format. + * + * + * + * + * References: Getting Started with Winsock + * http://msdn.microsoft.com/en-us/library/windows/desktop/bb530742(v=vs.85).aspx + * + * Future: Originally we had: + * KISS over serial port. + * AGW over socket. + * This is the two of them munged together and we end up with duplicate code. + * It would have been better to separate out the transport and application layers. + * Maybe someday. + * + *---------------------------------------------------------------*/ + +/* + Separate TCP ports per radio: + +An increasing number of people are using multiple radios. +direwolf is capable of handling many radio channels and +provides cross-band repeating, etc. +Maybe a single stereo audio interface is used for 2 radios. + + +------------+ tcp 8001, all channels +Radio A -------- | | -------------------------- Application A + | direwolf | +Radio B -------- | | -------------------------- Application B + +------------+ tcp 8001, all channels + +The KISS protocol has a 4 bit field for the TNC port (which I prefer to +call channel because port has too many different meanings). +direwolf handles this fine. However, most applications were written assuming +that a TNC could only talk to a single radio. On reception, they ignore the +channel in the KISS frame. For transmit, the channel is always set to 0. + +Many people are using the work-around of two separate instances of direwolf. + + +------------+ tcp 8001, KISS ch 0 +Radio A -------- | direwolf | -------------------------- Application A + +------------+ + + +------------+ tcp 8002, KISS ch 0 +Radio B -------- | direwolf | -------------------------- Application B + +------------+ + + +Or they might be using a single application that knows how to talk to multiple +single port TNCs. But they don't know how to multiplex multiple channels +thru a single KISS stream. + + +------------+ tcp 8001, KISS ch 0 +Radio A -------- | direwolf | ------------------------ + +------------+ \ + -- Application + +------------+ tcp 8002, KISS ch 0 / +Radio B -------- | direwolf | ------------------------ + +------------+ + +Using two different instances of direwolf means more complex configuration +and loss of cross-channel digipeating. It is possible to use a stereo +audio interface but some ALSA magic is required to make it look like two +independent virtual mono interfaces. + +In version 1.7, we add the capability of multiple KISS TCP ports, each for +a single radio channel. e.g. + +KISSPORT 8001 1 +KISSPORT 8002 2 + +Now can use a single instance of direwolf. + + + +------------+ tcp 8001, KISS ch 0 +Radio A -------- | | -------------------------- Application A + | direwolf | +Radio B -------- | | -------------------------- Application B + +------------+ tcp 8002, KISS ch 0 + +When receiving, the KISS channel is set to 0. + - only radio channel 1 would be sent over tcp port 8001. + - only radio channel 2 would be sent over tcp port 8001. + +When transmitting, the KISS channel is ignored. + - frames from tcp port 8001 are transmitted on radio channel 1. + - frames from tcp port 8002 are transmitted on radio channel 2. + +Of course, you could also use an application, capable of connecting to +multiple single radio TNCs. Separate TCP ports actually go to the +same direwolf instance. + +*/ + + +/* + * Native Windows: Use the Winsock interface. + * Linux: Use the BSD socket interface. + */ + + +#include "direwolf.h" // Sets _WIN32_WINNT for XP API level needed by ws2tcpip.h + + +#if __WIN32__ +#include +#include // _WIN32_WINNT must be set to 0x0501 before including this +#else +#include +#include +#include +#include +#include +#include +#endif + +#include +#include +#include +#include +#include + + +#include "tq.h" +#include "ax25_pad.h" +#include "textcolor.h" +#include "audio.h" +#include "kissnet.h" +#include "kiss_frame.h" +#include "xmit.h" + +void hex_dump (unsigned char *p, int len); // This should be in a .h file. + + + + + +// TODO: define in one place, use everywhere. +#if __WIN32__ +#define THREAD_F unsigned __stdcall +#else +#define THREAD_F void * +#endif + +static THREAD_F connect_listen_thread (void *arg); +static THREAD_F kissnet_listen_thread (void *arg); + + +static struct misc_config_s *s_misc_config_p; + + +// Each TCP port has its own status block. +// There is a variable number so use a linked list. + +static struct kissport_status_s *all_ports = NULL; + +static int kiss_debug = 0; /* Print information flowing from and to client. */ + +void kiss_net_set_debug (int n) +{ + kiss_debug = n; +} + + + +/*------------------------------------------------------------------- + * + * Name: kissnet_init + * + * Purpose: Set up a server to listen for connection requests from + * an application such as Xastir or APRSIS32. + * This is called once from the main program. + * + * Inputs: mc->kiss_port - TCP port for server. + * 0 means disable. New in version 1.2. + * + * Outputs: + * + * Description: This starts two threads: + * * to listen for a connection from client app. + * * to listen for commands from client app. + * so the main application doesn't block while we wait for these. + * + *--------------------------------------------------------------------*/ + +static void kissnet_init_one (struct kissport_status_s *kps); + +void kissnet_init (struct misc_config_s *mc) +{ + s_misc_config_p = mc; + + for (int i = 0; i < MAX_KISS_TCP_PORTS; i++) { + if (mc->kiss_port[i] != 0) { + struct kissport_status_s *kps = calloc(sizeof(struct kissport_status_s), 1); + if (kps == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } + + kps->tcp_port = mc->kiss_port[i]; + kps->chan = mc->kiss_chan[i]; + kissnet_init_one (kps); + + // Add to list. + kps->pnext = all_ports; + all_ports = kps; + } + } +} + + +static void kissnet_init_one (struct kissport_status_s *kps) +{ + int client; + +#if __WIN32__ + HANDLE connect_listen_th; + HANDLE cmd_listen_th[MAX_NET_CLIENTS]; +#else + pthread_t connect_listen_tid; + pthread_t cmd_listen_tid[MAX_NET_CLIENTS]; + int e; +#endif + + + + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kissnet_init ( tcp port %d, radio chan = %d )\n", kps->tcp_port, kps->chan); +#endif + + + for (client=0; clientclient_sock[client] = -1; + memset (&(kps->kf[client]), 0, sizeof(kps->kf[client])); + } + + if (kps->tcp_port == 0) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Disabled KISS network client port.\n"); + return; + } + +/* + * This waits for a client to connect and sets client_sock[n]. + */ +#if __WIN32__ + connect_listen_th = (HANDLE)_beginthreadex (NULL, 0, connect_listen_thread, (void *)kps, 0, NULL); + if (connect_listen_th == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not create KISS socket connect listening thread for tcp port %d, radio chan %d\n", kps->tcp_port, kps->chan); + return; + } +#else + e = pthread_create (&connect_listen_tid, NULL, connect_listen_thread, (void *)kps); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + perror("Could not create KISS socket connect listening thread"); + dw_printf ("for tcp port %d, radio chan %d\n", kps->tcp_port, kps->chan); + return; + } +#endif + +/* + * These read messages from client when client_sock[n] is valid. + * Currently we start up a separate thread for each potential connection. + * Possible later refinement. Start one now, others only as needed. + */ + for (client = 0; client < MAX_NET_CLIENTS; client++) { + + kps->arg2 = client; + +#if __WIN32__ + cmd_listen_th[client] = (HANDLE)_beginthreadex (NULL, 0, kissnet_listen_thread, (void*)kps, 0, NULL); + if (cmd_listen_th[client] == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not create KISS command listening thread for client %d\n", client); + return; + } +#else + e = pthread_create (&(cmd_listen_tid[client]), NULL, kissnet_listen_thread, (void *)kps); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not create KISS command listening thread for client %d\n", client); + // Replace add perror with better message handling. + perror(""); + return; + } +#endif + // Wait for new thread to get content of arg2 before reusing it for the next thread create. + + int timer = 0; + while (kps->arg2 >= 0) { + SLEEP_MS(10); + timer++; + if (timer > 100) { // 1 second - thread did not start + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS data listening thread did not start for tcp port %d, client slot %d\n", kps->tcp_port, client); + kps->arg2 = -1; // Keep moving along. + } + } + } +} + + +/*------------------------------------------------------------------- + * + * Name: connect_listen_thread + * + * Purpose: Wait for a connection request from an application. + * + * Inputs: arg - KISS port status block. + * + * Outputs: client_sock - File descriptor for communicating with client app. + * + * Description: Wait for connection request from client and establish + * communication. + * Note that the client can go away and come back again and + * re-establish communication without restarting this application. + * + *--------------------------------------------------------------------*/ + +static THREAD_F connect_listen_thread (void *arg) +{ + struct kissport_status_s *kps = arg; + +#if __WIN32__ + + struct addrinfo hints; + struct addrinfo *ai = NULL; + int err; + char tcp_port_str[12]; + + SOCKET listen_sock; + WSADATA wsadata; + + snprintf (tcp_port_str, sizeof(tcp_port_str), "%d", kps->tcp_port); +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("DEBUG: kissnet port = %d = '%s'\n", (int)(ptrdiff_t)arg, tcp_port_str); +#endif + err = WSAStartup (MAKEWORD(2,2), &wsadata); + if (err != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("WSAStartup failed: %d\n", err); + return (0); + } + + if (LOBYTE(wsadata.wVersion) != 2 || HIBYTE(wsadata.wVersion) != 2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Could not find a usable version of Winsock.dll\n"); + WSACleanup(); + //sleep (1); + return (0); + } + + memset (&hints, 0, sizeof(hints)); + hints.ai_family = AF_INET; + hints.ai_socktype = SOCK_STREAM; + hints.ai_protocol = IPPROTO_TCP; + hints.ai_flags = AI_PASSIVE; + + err = getaddrinfo(NULL, tcp_port_str, &hints, &ai); + if (err != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf("getaddrinfo failed: %d\n", err); + //sleep (1); + WSACleanup(); + return (0); + } + + listen_sock= socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol); + if (listen_sock == INVALID_SOCKET) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("connect_listen_thread: Socket creation failed, err=%d", WSAGetLastError()); + return (0); + } + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf("Binding to port %s ... \n", tcp_port_str); +#endif + + err = bind( listen_sock, ai->ai_addr, (int)ai->ai_addrlen); + if (err == SOCKET_ERROR) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Bind failed with error: %d\n", WSAGetLastError()); // TODO: provide corresponding text. + dw_printf("Some other application is probably already using port %s.\n", tcp_port_str); + dw_printf("Try using a different port number with KISSPORT in the configuration file.\n"); + freeaddrinfo(ai); + closesocket(listen_sock); + WSACleanup(); + return (0); + } + + freeaddrinfo(ai); + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf("opened KISS socket as fd (%d) on port (%s) for stream i/o\n", listen_sock, tcp_port_str ); +#endif + + while (1) { + + int client; + int c; + + client = -1; + for (c = 0; c < MAX_NET_CLIENTS && client < 0; c++) { + if (kps->client_sock[c] <= 0) { + client = c; + } + } + +/* + * Listen for connection if we have not reached maximum. + */ + if (client >= 0) { + + if(listen(listen_sock, MAX_NET_CLIENTS) == SOCKET_ERROR) + { + text_color_set(DW_COLOR_ERROR); + dw_printf("Listen failed with error: %d\n", WSAGetLastError()); + return (0); + } + + text_color_set(DW_COLOR_INFO); + if (kps->chan == -1) { + dw_printf("Ready to accept KISS TCP client application %d on port %s ...\n", client, tcp_port_str); + } + else { + dw_printf("Ready to accept KISS TCP client application %d on port %s (radio channel %d) ...\n", client, tcp_port_str, kps->chan); + } + + kps->client_sock[client] = accept(listen_sock, NULL, NULL); + + if (kps->client_sock[client] == -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Accept failed with error: %d\n", WSAGetLastError()); + closesocket(listen_sock); + WSACleanup(); + return (0); + } + + text_color_set(DW_COLOR_INFO); + if (kps->chan == -1) { + dw_printf("\nAttached to KISS TCP client application %d on port %s ...\n\n", client, tcp_port_str); + } + else { + dw_printf("\nAttached to KISS TCP client application %d on port %s (radio channel %d) ...\n\n", client, tcp_port_str, kps->chan); + } + + // Reset the state and buffer. + memset (&(kps->kf[client]), 0, sizeof(kps->kf[client])); + } + else { + SLEEP_SEC(1); /* wait then check again if more clients allowed. */ + } + } + + +#else /* End of Windows case, now Linux / Unix / Mac OSX. */ + + + struct sockaddr_in sockaddr; /* Internet socket address struct */ + socklen_t sockaddr_size = sizeof(struct sockaddr_in); + int listen_sock; + int bcopt = 1; + + listen_sock= socket(AF_INET,SOCK_STREAM,0); + if (listen_sock == -1) { + text_color_set(DW_COLOR_ERROR); + perror ("connect_listen_thread: Socket creation failed"); + return (NULL); + } + + /* Version 1.3 - as suggested by G8BPQ. */ + /* Without this, if you kill the application then try to run it */ + /* again quickly the port number is unavailable for a while. */ + /* Don't try doing the same thing On Windows; It has a different meaning. */ + /* http://stackoverflow.com/questions/14388706/socket-options-so-reuseaddr-and-so-reuseport-how-do-they-differ-do-they-mean-t */ + + setsockopt (listen_sock, SOL_SOCKET, SO_REUSEADDR, (const char *)&bcopt, 4); + + sockaddr.sin_addr.s_addr = INADDR_ANY; + sockaddr.sin_port = htons(kps->tcp_port); + sockaddr.sin_family = AF_INET; + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf("Binding to port %d ... \n", kps->tcp_port); +#endif + + if (bind(listen_sock,(struct sockaddr*)&sockaddr,sizeof(sockaddr)) == -1) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Bind failed with error: %d\n", errno); + dw_printf("%s\n", strerror(errno)); + dw_printf("Some other application is probably already using port %d.\n", kps->tcp_port); + dw_printf("Try using a different port number with KISSPORT in the configuration file.\n"); + return (NULL); + } + + getsockname( listen_sock, (struct sockaddr *)(&sockaddr), &sockaddr_size); + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf("opened KISS TCP socket as fd (%d) on port (%d) for stream i/o\n", listen_sock, ntohs(sockaddr.sin_port) ); +#endif + + while (1) { + + int client; + int c; + + client = -1; + for (c = 0; c < MAX_NET_CLIENTS && client < 0; c++) { + if (kps->client_sock[c] <= 0) { + client = c; + } + } + + if (client >= 0) { + + if(listen(listen_sock,MAX_NET_CLIENTS) == -1) + { + text_color_set(DW_COLOR_ERROR); + perror ("connect_listen_thread: Listen failed"); + return (NULL); + } + + text_color_set(DW_COLOR_INFO); + if (kps->chan == -1) { + dw_printf("Ready to accept KISS TCP client application %d on port %d ...\n", client, kps->tcp_port); + } + else { + dw_printf("Ready to accept KISS TCP client application %d on port %d (radio channel %d) ...\n", client, kps->tcp_port, kps->chan); + } + + kps->client_sock[client] = accept(listen_sock, (struct sockaddr*)(&sockaddr),&sockaddr_size); + + text_color_set(DW_COLOR_INFO); + if (kps->chan == -1) { + dw_printf("\nAttached to KISS TCP client application %d on port %d ...\n\n", client, kps->tcp_port); + } + else { + dw_printf("\nAttached to KISS TCP client application %d on port %d (radio channel %d) ...\n\n", client, kps->tcp_port, kps->chan); + } + + // Reset the state and buffer. + memset (&(kps->kf[client]), 0, sizeof(kps->kf[client])); + } + else { + SLEEP_SEC(1); /* wait then check again if more clients allowed. */ + } + } +#endif +} + + + + + +/*------------------------------------------------------------------- + * + * Name: kissnet_send_rec_packet + * + * Purpose: Send a packet, received over the radio, to the client app. + * + * Inputs: chan - Channel number where packet was received. + * 0 = first, 1 = second if any. + * +// TODO: add kiss_cmd + * + * fbuf - Address of raw received frame buffer + * or a text string. + * + * kiss_cmd - Usually KISS_CMD_DATA_FRAME but we can also have + * KISS_CMD_SET_HARDWARE when responding to a query. + * + * flen - Number of bytes for AX.25 frame. + * When called from kiss_rec_byte, flen will be -1 + * indicating a text string rather than frame content. + * This is used to fake out an application that thinks + * it is using a traditional TNC and tries to put it + * into KISS mode. + * + * onlykps - KISS TCP status block pointer or NULL. + * + * onlyclient - It is possible to have more than client attached + * at the same time with TCP KISS. + * Starting with version 1.7 we can have multiple TCP ports. + * When a frame is received from the radio we normally want it + * to go to all of the clients. + * In this case specify NULL for onlykps and -1 tcp client. + * When responding to a command from the client, we want + * to send only to that one client app. In this case + * a non NULL kps and onlyclient >= 0. + * + * Description: Send message to client(s) if connected. + * Disconnect from client, and notify user, if any error. + * + *--------------------------------------------------------------------*/ + +void kissnet_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, + struct kissport_status_s *onlykps, int onlyclient) +{ + unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN]; + int kiss_len; + int err; + +// Something received over the radio would normally be sent to all attached clients. +// However, there are times we want to send a response only to a particular client. +// In the case of a serial port or pseudo terminal, there is only one potential client. +// so the response would be sent to only one place. A new parameter has been added for this. + + for (struct kissport_status_s *kps = all_ports; kps != NULL; kps = kps->pnext) { + + if (onlykps == NULL || kps == onlykps) { + + for (int client = 0; client < MAX_NET_CLIENTS; client++) { + + if (onlyclient == -1 || client == onlyclient) { + + if (kps->client_sock[client] != -1) { + + if (flen < 0) { + +// A client app might think it is attached to a traditional TNC. +// It might try sending commands over and over again trying to get the TNC into KISS mode. +// We recognize this attempt and send it something to keep it happy. + + text_color_set(DW_COLOR_ERROR); + dw_printf ("KISS TCP: Something unexpected from client application.\n"); + dw_printf ("Is client app treating this like an old TNC with command mode?\n"); + dw_printf ("This can be caused by the application sending commands to put a\n"); + dw_printf ("traditional TNC into KISS mode. It is usually a harmless warning.\n"); + dw_printf ("For best results, configure for a KISS-only TNC to avoid this.\n"); + dw_printf ("In the case of APRSISCE/32, use \"Simply(KISS)\" rather than \"KISS.\"\n"); + + flen = strlen((char*)fbuf); + if (kiss_debug) { + kiss_debug_print (TO_CLIENT, "Fake command prompt", fbuf, flen); + } + strlcpy ((char *)kiss_buff, (char *)fbuf, sizeof(kiss_buff)); + kiss_len = strlen((char *)kiss_buff); + } + else { + unsigned char stemp[AX25_MAX_PACKET_LEN + 1]; + + assert (flen < (int)(sizeof(stemp))); + + // New in 1.7. + // Previously all channels were sent to everyone. + // We now have tcp ports which carry only a single radio channel. + // The application will see KISS channel 0 regardless of the radio channel. + + if (kps->chan == -1) { + // Normal case, all channels. + stemp[0] = (chan << 4) | kiss_cmd; + } + else if (kps->chan == chan) { + // Single radio channel for this port. Application sees 0. + stemp[0] = (0 << 4) | kiss_cmd; + } + else { + // Skip it. + continue; + } + + memcpy (stemp+1, fbuf, flen); + + if (kiss_debug >= 2) { + /* AX.25 frame with the CRC removed. */ + text_color_set(DW_COLOR_DEBUG); + dw_printf ("\n"); + dw_printf ("Packet content before adding KISS framing and any escapes:\n"); + hex_dump (fbuf, flen); + } + + kiss_len = kiss_encapsulate (stemp, flen+1, kiss_buff); + + /* This has the escapes and the surrounding FENDs. */ + + if (kiss_debug) { + kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len); + } + } + +#if __WIN32__ + err = SOCK_SEND(kps->client_sock[client], (char*)kiss_buff, kiss_len); + if (err == SOCKET_ERROR) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError %d sending message to KISS client application %d on port %d. Closing connection.\n\n", WSAGetLastError(), client, kps->tcp_port); + closesocket (kps->client_sock[client]); + kps->client_sock[client] = -1; + WSACleanup(); + } +#else + err = SOCK_SEND (kps->client_sock[client], kiss_buff, kiss_len); + if (err <= 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError %d sending message to KISS client application %d on port %d. Closing connection.\n\n", err, client, kps->tcp_port); + close (kps->client_sock[client]); + kps->client_sock[client] = -1; + } +#endif + } // frame length >= 0 + } // if all clients or the one specifie + } // for each client on the tcp port + } // if all ports or the one specified + } // for each tcp port + +} /* end kissnet_send_rec_packet */ + + +/*------------------------------------------------------------------- + * + * Name: kissnet_copy + * + * Purpose: Send data from one network KISS client to all others. + * + * Inputs: in_msg - KISS frame data without the framing or escapes. + * The first byte is channel and command (should be data). + * Caller no longer cares this byte. We will clobber it here. + * + * in_len - Number of bytes in above. + * + * chan - Channel. Use this instead of first byte of in_msg. + * + * cmd - KISS command nybble. + * Should be 0 because I'm expecting this only for data. + * + * from_client - Number of network (TCP) client instance. + * Should be 0, 1, 2, ... + * + * + * Global In: kiss_copy - From misc. configuration. + * This enables the feature. + * + * + * Description: Send message to any attached network KISS clients, other than the one where it came from. + * Enable this by putting KISSCOPY in the configuration file. + * Note that this applies only to network (TCP) KISS clients, not serial port, or pseudo terminal. + * + * + *--------------------------------------------------------------------*/ + + +void kissnet_copy (unsigned char *in_msg, int in_len, int chan, int cmd, struct kissport_status_s *from_kps, int from_client) +{ + unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN]; + int err; + + + if (s_misc_config_p->kiss_copy) { + + for (struct kissport_status_s *kps = all_ports; kps != NULL; kps = kps->pnext) { + + for (int client = 0; client < MAX_NET_CLIENTS; client++) { + + if ( ! ( kps == from_kps && client == from_client ) ) { // To all but origin. + + if (kps->client_sock[client] != -1) { + + if (kps-> chan == -1 || kps->chan == chan) { + + // Two different cases here: + // - The TCP port allows all channels, or + // - The TCP port allows only one channel. In this case set KISS channel to 0. + + if (kps->chan == -1) { + in_msg[0] = (chan << 4) | cmd; + } + else { + in_msg[0] = 0 | cmd; // set channel to zero. + } + + int kiss_len = kiss_encapsulate (in_msg, in_len, kiss_buff); + + /* This has the escapes and the surrounding FENDs. */ + + if (kiss_debug) { + kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len); + } + +#if __WIN32__ + err = SOCK_SEND(kps->client_sock[client], (char*)kiss_buff, kiss_len); + if (err == SOCKET_ERROR) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError %d copying message to KISS TCP port %d client %d application. Closing connection.\n\n", WSAGetLastError(), kps->tcp_port, client); + closesocket (kps->client_sock[client]); + kps->client_sock[client] = -1; + WSACleanup(); + } +#else + err = SOCK_SEND (kps->client_sock[client], kiss_buff, kiss_len); + if (err <= 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError copying message to KISS TCP port %d client %d application. Closing connection.\n\n", kps->tcp_port, client); + close (kps->client_sock[client]); + kps->client_sock[client] = -1; + } +#endif + } // Channel is allowed on this port. + } // socket is open + } // if origin and destination different. + } // loop over all KISS network clients for one port. + } // loop over all KISS TCP ports + } // Feature enabled. + +} /* end kissnet_copy */ + + + +/*------------------------------------------------------------------- + * + * Name: kissnet_listen_thread + * + * Purpose: Wait for KISS messages from an application. + * + * Inputs: arg - client number, 0 .. MAX_NET_CLIENTS-1 + * + * Outputs: client_sock[n] - File descriptor for communicating with client app. + * + * Description: Process messages from the client application. + * Note that the client can go away and come back again and + * re-establish communication without restarting this application. + * + *--------------------------------------------------------------------*/ + + +/* Return one byte (value 0 - 255) */ + + +static int kiss_get (struct kissport_status_s *kps, int client) +{ + + while (1) { + + while (kps->client_sock[client] <= 0) { + SLEEP_SEC(1); /* Not connected. Try again later. */ + } + + /* Just get one byte at a time. */ + + unsigned char ch; + int n = SOCK_RECV (kps->client_sock[client], (char *)(&ch), 1); + + if (n == 1) { +#if DEBUG9 + dw_printf (log_fp, "%02x %c %c", ch, + isprint(ch) ? ch : '.' , + (isupper(ch>>1) || isdigit(ch>>1) || (ch>>1) == ' ') ? (ch>>1) : '.'); + if (ch == FEND) fprintf (log_fp, " FEND"); + if (ch == FESC) fprintf (log_fp, " FESC"); + if (ch == TFEND) fprintf (log_fp, " TFEND"); + if (ch == TFESC) fprintf (log_fp, " TFESC"); + if (ch == '\r') fprintf (log_fp, " CR"); + if (ch == '\n') fprintf (log_fp, " LF"); + fprintf (log_fp, "\n"); + if (ch == FEND) fflush (log_fp); +#endif + return(ch); + } + + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nKISS client application %d on TCP port %d has gone away.\n\n", client, kps->tcp_port); +#if __WIN32__ + closesocket (kps->client_sock[client]); +#else + close (kps->client_sock[client]); +#endif + kps->client_sock[client] = -1; + } +} + + + +static THREAD_F kissnet_listen_thread (void *arg) +{ + struct kissport_status_s *kps = arg; + + int client = kps->arg2; + assert (client >= 0 && client < MAX_NET_CLIENTS); + + kps->arg2 = -1; // Indicates thread is running so + // arg2 can be reused for the next one. + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kissnet_listen_thread ( tcp_port = %d, client = %d, socket fd = %d )\n", kps->tcp_port, client, kps->client_sock[client]); +#endif + + + +// So why is kissnet_send_rec_packet mentioned here for incoming from the client app? +// The logic exists for the serial port case where the client might think it is +// attached to a traditional TNC. It might try sending commands over and over again +// trying to get the TNC into KISS mode. To keep it happy, we recognize this attempt +// and send it something to keep it happy. +// In the case of a serial port or pseudo terminal, there is only one potential client +// so the response would be sent to only one place. +// Starting in version 1.5, this now can have multiple attached clients. We wouldn't +// want to send the response to all of them. Actually, we should be providing only +// "Simply KISS" as some call it. + + + while (1) { + unsigned char ch = kiss_get(kps, client); + kiss_rec_byte (&(kps->kf[client]), ch, kiss_debug, kps, client, kissnet_send_rec_packet); + } + +#if __WIN32__ + return(0); +#else + return (THREAD_F) 0; /* Unreachable but avoids compiler warning. */ +#endif + +} /* end kissnet_listen_thread */ + +/* end kissnet.c */ diff --git a/src/kissnet.h b/src/kissnet.h new file mode 100644 index 00000000..469e4e63 --- /dev/null +++ b/src/kissnet.h @@ -0,0 +1,29 @@ + +/* + * Name: kissnet.h + */ + +#ifndef KISSNET_H +#define KISSNET_H + +#include "ax25_pad.h" /* for packet_t */ + +#include "config.h" + +#include "kiss_frame.h" + + + +void kissnet_init (struct misc_config_s *misc_config); + +void kissnet_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, + struct kissport_status_s *onlykps, int onlyclient); + +void kiss_net_set_debug (int n); + +void kissnet_copy (unsigned char *kiss_msg, int kiss_len, int chan, int cmd, struct kissport_status_s *from_kps, int from_client); + + +#endif // KISSNET_H + +/* end kissnet.h */ diff --git a/src/kissserial.c b/src/kissserial.c new file mode 100644 index 00000000..1ee5356a --- /dev/null +++ b/src/kissserial.c @@ -0,0 +1,503 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2011, 2013, 2014, 2016, 2017 John Langner, WB2OSZ +// +// 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, see . +// + + +/*------------------------------------------------------------------ + * + * Module: kissserial.c + * + * Purpose: Act as a virtual KISS TNC for use by other packet radio applications. + * This file provides the service by good old fashioned serial port. + * Other files implement a pseudo terminal or TCP KISS interface. + * + * Description: This implements the KISS TNC protocol as described in: + * http://www.ka9q.net/papers/kiss.html + * + * Briefly, a frame is composed of + * + * * FEND (0xC0) + * * Contents - with special escape sequences so a 0xc0 + * byte in the data is not taken as end of frame. + * as part of the data. + * * FEND + * + * The first byte of the frame contains: + * + * * port number in upper nybble. + * * command in lower nybble. + * + * Commands from application recognized: + * + * _0 Data Frame AX.25 frame in raw format. + * + * _1 TXDELAY See explanation in xmit.c. + * + * _2 Persistence " " + * + * _3 SlotTime " " + * + * _4 TXtail " " + * Spec says it is obsolete but Xastir + * sends it and we respect it. + * + * _5 FullDuplex Ignored. + * + * _6 SetHardware TNC specific. + * + * FF Return Exit KISS mode. Ignored. + * + * + * Messages sent to client application: + * + * _0 Data Frame Received AX.25 frame in raw format. + * + * + * Platform differences: + * + * This file implements KISS over a serial port. + * It should behave pretty much the same for both Windows and Linux. + * + * When running a client application on Windows, two applications + * can be connected together using a a "Null-modem emulator" + * such as com0com from http://sourceforge.net/projects/com0com/ + * + * (When running a client application, on the same host, with Linux, + * a pseudo terminal can be used for old applications. More modern + * applications will generally have AGW and/or KISS over TCP.) + * + * + * version 1.5: Split out from kiss.c, simplified, consistent for Windows and Linux. + * Add polling option for use with Bluetooth. + * + *---------------------------------------------------------------*/ + +#include "direwolf.h" + +#include +#include +#include +#include +#include +#include + + +#include "ax25_pad.h" +#include "textcolor.h" +#include "serial_port.h" +#include "kissserial.h" +#include "kiss_frame.h" +#include "xmit.h" + + +/* + * Save Configuration for later use. + */ + +static struct misc_config_s *g_misc_config_p; + +/* + * Accumulated KISS frame and state of decoder. + */ + +static kiss_frame_t kf; + + +/* + * The serial port device handle. + * MYFD... are defined in kissserial.h + */ + +static MYFDTYPE serialport_fd = MYFDERROR; + + + + +// TODO: define in one place, use everywhere. +#if __WIN32__ +#define THREAD_F unsigned __stdcall +#else +#define THREAD_F void * +#endif + +static THREAD_F kissserial_listen_thread (void *arg); + + +static int kissserial_debug = 0; /* Print information flowing from and to client. */ + +void kissserial_set_debug (int n) +{ + kissserial_debug = n; +} + + +/* In server.c. Should probably move to some misc. function file. */ + +void hex_dump (unsigned char *p, int len); + + + + +/*------------------------------------------------------------------- + * + * Name: kissserial_init + * + * Purpose: Set up a serial port acting as a virtual KISS TNC. + * + * Inputs: mc-> + * kiss_serial_port - Name of device for real or virtual serial port. + * kiss_serial_speed - Speed, bps, or 0 meaning leave it alone. + * kiss_serial_poll - When non-zero, poll each n seconds to see if + * device has appeared. + * + * Outputs: + * + * Description: (1) Open file descriptor for the device. + * (2) Start a new thread to listen for commands from client app + * so the main application doesn't block while we wait. + * + *--------------------------------------------------------------------*/ + + +void kissserial_init (struct misc_config_s *mc) +{ + +#if __WIN32__ + HANDLE kissserial_listen_th; +#else + pthread_t kissserial_listen_tid; + int e; +#endif + + g_misc_config_p = mc; + + memset (&kf, 0, sizeof(kf)); + + + if (strlen(g_misc_config_p->kiss_serial_port) > 0) { + + if (g_misc_config_p->kiss_serial_poll == 0) { + + // Normal case, try to open the serial port at start up time. + + serialport_fd = serial_port_open (g_misc_config_p->kiss_serial_port, g_misc_config_p->kiss_serial_speed); + + if (serialport_fd != MYFDERROR) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Opened %s for serial port KISS.\n", g_misc_config_p->kiss_serial_port); + } + else { + // An error message was already displayed. + } + } + else { + + // Polling case. Defer until read and device not opened. + text_color_set(DW_COLOR_INFO); + dw_printf ("Will be checking periodically for %s\n", g_misc_config_p->kiss_serial_port); + } + + if (g_misc_config_p->kiss_serial_poll != 0 || serialport_fd != MYFDERROR) { +#if __WIN32__ + kissserial_listen_th = (HANDLE)_beginthreadex (NULL, 0, kissserial_listen_thread, NULL, 0, NULL); + if (kissserial_listen_th == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Could not create kiss serial thread\n"); + return; + } +#else + e = pthread_create (&kissserial_listen_tid, NULL, kissserial_listen_thread, NULL); + if (e != 0) { + text_color_set(DW_COLOR_ERROR); + perror("Could not create kiss serial thread."); + } +#endif + } + } + + +#if DEBUG + text_color_set (DW_COLOR_DEBUG); + + dw_printf ("end of kiss_init: serialport_fd = %d, polling = %d\n", serialport_fd, g_misc_config_p->kiss_serial_poll); +#endif +} + + + + +/*------------------------------------------------------------------- + * + * Name: kissserial_send_rec_packet + * + * Purpose: Send a received packet or text string to the client app. + * + * Inputs: chan - Channel number where packet was received. + * 0 = first, 1 = second if any. + * + * kiss_cmd - Usually KISS_CMD_DATA_FRAME but we can also have + * KISS_CMD_SET_HARDWARE when responding to a query. + * + * pp - Identifier for packet object. + * + * fbuf - Address of raw received frame buffer + * or a text string. + * + * flen - Length of raw received frame not including the FCS + * or -1 for a text string. + * + * kps + * client - Not used for serial port version. + * Here so that 3 related functions all have + * the same parameter list. + * + * Description: Send message to client. + * We really don't care if anyone is listening or not. + * I don't even know if we can find out. + * + *--------------------------------------------------------------------*/ + + +void kissserial_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, + struct kissport_status_s *notused1, int notused2) +{ + unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN + 2]; + int kiss_len; + int err; + + +/* + * Quietly discard if we don't have open connection. + */ + if (serialport_fd == MYFDERROR) { + return; + } + + if (flen < 0) { + flen = strlen((char*)fbuf); + if (kissserial_debug) { + kiss_debug_print (TO_CLIENT, "Fake command prompt", fbuf, flen); + } + strlcpy ((char *)kiss_buff, (char *)fbuf, sizeof(kiss_buff)); + kiss_len = strlen((char *)kiss_buff); + } + else { + + unsigned char stemp[AX25_MAX_PACKET_LEN + 1]; + + if (flen > (int)(sizeof(stemp)) - 1) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nSerial Port KISS buffer too small. Truncated.\n\n"); + flen = (int)(sizeof(stemp)) - 1; + } + + stemp[0] = (chan << 4) | kiss_cmd; + memcpy (stemp+1, fbuf, flen); + + if (kissserial_debug >= 2) { + /* AX.25 frame with the CRC removed. */ + text_color_set(DW_COLOR_DEBUG); + dw_printf ("\n"); + dw_printf ("Packet content before adding KISS framing and any escapes:\n"); + hex_dump (fbuf, flen); + } + + kiss_len = kiss_encapsulate (stemp, flen+1, kiss_buff); + + /* This has KISS framing and escapes for sending to client app. */ + + if (kissserial_debug) { + kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len); + } + } + +/* + * This write can block on Windows if using the virtual null modem + * and nothing is connected to the other end. + * The solution is found in the com0com ReadMe file: + * + * Q. My application hangs during its startup when it sends anything to one paired + * COM port. The only way to unhang it is to start HyperTerminal, which is connected + * to the other paired COM port. I didn't have this problem with physical serial + * ports. + * A. Your application can hang because receive buffer overrun is disabled by + * default. You can fix the problem by enabling receive buffer overrun for the + * receiving port. Also, to prevent some flow control issues you need to enable + * baud rate emulation for the sending port. So, if your application use port CNCA0 + * and other paired port is CNCB0, then: + * + * 1. Launch the Setup Command Prompt shortcut. + * 2. Enter the change commands, for example: + * + * command> change CNCB0 EmuOverrun=yes + * command> change CNCA0 EmuBR=yes + */ + + err = serial_port_write (serialport_fd, (char*)kiss_buff, kiss_len); + + if (err != kiss_len) + { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nError sending KISS message to client application thru serial port.\n\n"); + serial_port_close (serialport_fd); + serialport_fd = MYFDERROR; + } + +} /* kissserial_send_rec_packet */ + + + +/*------------------------------------------------------------------- + * + * Name: kissserial_get + * + * Purpose: Read one byte from the KISS client app. + * + * Global In: serialport_fd + * + * Returns: one byte (value 0 - 255) or terminate thread on error. + * + * Description: There is room for improvement here. Reading one byte + * at a time is inefficient. We could read a large block + * into a local buffer and return a byte from that most of the time. + * Is it worth the effort? I don't know. With GHz processors and + * the low data rate here it might not make a noticeable difference. + * + *--------------------------------------------------------------------*/ + + +static int kissserial_get (void) +{ + int ch; // normally 0-255 but -1 for error. + + + if (g_misc_config_p->kiss_serial_poll == 0) { +/* + * Normal case, was opened at start up time. + */ + ch = serial_port_get1 (serialport_fd); + + if (ch < 0) { + + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nSerial Port KISS read error. Closing connection.\n\n"); + serial_port_close (serialport_fd); + serialport_fd = MYFDERROR; +#if __WIN32__ + ExitThread (0); +#else + pthread_exit (NULL); +#endif + } + +#if DEBUGx + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kissserial_get(%d) returns 0x%02x\n", fd, ch); +#endif + return (ch); + } + +/* + * Polling case. Wait until device is present and open. + */ + while (1) { + + if (serialport_fd != MYFDERROR) { + + // Open, try to read. + + ch = serial_port_get1 (serialport_fd); + + if (ch >= 0) { + return (ch); + } + + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nSerial Port KISS read error. Closing connection.\n\n"); + serial_port_close (serialport_fd); + serialport_fd = MYFDERROR; + } + else { + + // Not open. Wait for it to appear and try opening. + + struct stat buf; + + SLEEP_SEC (g_misc_config_p->kiss_serial_poll); + + if (stat(g_misc_config_p->kiss_serial_port, &buf) == 0) { + + // It's there now. Try to open. + + serialport_fd = serial_port_open (g_misc_config_p->kiss_serial_port, g_misc_config_p->kiss_serial_speed); + + if (serialport_fd != MYFDERROR) { + text_color_set(DW_COLOR_INFO); + dw_printf ("\nOpened %s for serial port KISS.\n\n", g_misc_config_p->kiss_serial_port); + + memset (&kf, 0, sizeof(kf)); // Start with clean state. + } + else { + // An error message was already displayed. + } + } + } + } + +} /* end kissserial_get */ + + + +/*------------------------------------------------------------------- + * + * Name: kissserial_listen_thread + * + * Purpose: Read messages from serial port KISS client application. + * + * Global In: serialport_fd + * + * Description: Reads bytes from the serial port KISS client app and + * sends them to kiss_rec_byte for processing. + * kiss_rec_byte is a common function used by all 3 KISS + * interfaces: serial port, pseudo terminal, and TCP. + * + *--------------------------------------------------------------------*/ + + +static THREAD_F kissserial_listen_thread (void *arg) +{ + unsigned char ch; + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + dw_printf ("kissserial_listen_thread ( %d )\n", fd); +#endif + + while (1) { + ch = kissserial_get(); + kiss_rec_byte (&kf, ch, kissserial_debug, NULL, -1, kissserial_send_rec_packet); + } + +#if __WIN32__ + return(0); +#else + return (THREAD_F) 0; /* Unreachable but avoids compiler warning. */ +#endif +} + +/* end kissserial.c */ diff --git a/src/kissserial.h b/src/kissserial.h new file mode 100644 index 00000000..44fb3c3f --- /dev/null +++ b/src/kissserial.h @@ -0,0 +1,23 @@ + +/* + * Name: kissserial.h + */ + + +#include "ax25_pad.h" /* for packet_t */ + +#include "config.h" + +#include "kiss_frame.h" + + +void kissserial_init (struct misc_config_s *misc_config); + +void kissserial_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int flen, + struct kissport_status_s *notused1, int notused2); + + +void kissserial_set_debug (int n); + + +/* end kissserial.h */ diff --git a/src/kissutil.c b/src/kissutil.c new file mode 100644 index 00000000..fcd86088 --- /dev/null +++ b/src/kissutil.c @@ -0,0 +1,955 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2017 John Langner, WB2OSZ +// +// 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, see . +// + + +/*------------------------------------------------------------------ + * + * Module: kissutil.c + * + * Purpose: Utility for talking to a KISS TNC. + * + * Description: Convert between KISS format and usual text representation. + * This might also serve as the starting point for an application + * that uses a KISS TNC. + * The TNC can be attached by TCP or a serial port. + * + * Usage: kissutil [ options ] + * + * Default is to connect to localhost:8001. + * See the "usage" functions at the bottom for details. + * + *---------------------------------------------------------------*/ + +#include "direwolf.h" // Sets _WIN32_WINNT for XP API level needed by ws2tcpip.h + +#if __WIN32__ + +#include +#include // _WIN32_WINNT must be set to 0x0501 before including this + +#else + +#include +#include +#include +#include + +#endif + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "ax25_pad.h" +#include "textcolor.h" +#include "serial_port.h" +#include "kiss_frame.h" +#include "dwsock.h" +#include "audio.h" // for DEFAULT_TXDELAY, etc. +#include "dtime_now.h" + + +// TODO: define in one place, use everywhere. +#if __WIN32__ +#define THREAD_F unsigned __stdcall +#else +#define THREAD_F void * +#endif + +#if __WIN32__ +#define DIR_CHAR "\\" +#else +#define DIR_CHAR "/" +#endif + +static THREAD_F tnc_listen_net (void *arg); +static THREAD_F tnc_listen_serial (void *arg); + +static void send_to_kiss_tnc (int chan, int cmd, char *data, int dlen); +static void hex_dump (unsigned char *p, int len); + +static void usage(void); +static void usage2(void); + + + + +/* Obtained from the command line. */ + +static char hostname[50] = "localhost"; /* -h option. */ + /* DNS host name or IPv4 address. */ + /* Some of the code is there for IPv6 but */ + /* it needs more work. */ + /* Defaults to "localhost" if not specified. */ + +static char port[30] = "8001"; /* -p option. */ + /* If it begins with a digit, it is considered */ + /* a TCP port number at the hostname. */ + /* Otherwise, we treat it as a serial port name. */ + +static int using_tcp = 1; /* Are we using TCP or serial port for TNC? */ + /* Use corresponding one of the next two. */ + /* This is derived from the first character of port. */ + +static int server_sock = -1; /* File descriptor for socket interface. */ + /* Set to -1 if not used. */ + /* (Don't use SOCKET type because it is unsigned.) */ + +static MYFDTYPE serial_fd = (MYFDTYPE)(-1); /* Serial port handle. */ + +static int serial_speed = 9600; /* -s option. */ + /* Serial port speed, bps. */ + +static int verbose = 0; /* -v option. */ + /* Display the KISS protocol in hexadecimal for troubleshooting. */ + +static char transmit_from[120] = ""; /* -f option */ + /* When specified, files are read from this directory */ + /* rather than using stdin. Each file is one or more */ + /* lines in the standard monitoring format. */ + +static char receive_output[120] = ""; /* -o option */ + /* When specified, each received frame is stored as a file */ + /* with a unique name here. */ + /* Directory must already exist; we won't create it. */ + +static char timestamp_format[60] = ""; /* -T option */ + /* Precede received frames with timestamp. */ + /* Command line option uses "strftime" format string. */ + + +#if __WIN32__ +#define THREAD_F unsigned __stdcall +#else +#define THREAD_F void * +#endif + +#if __WIN32__ + static HANDLE tnc_th; +#else + static pthread_t tnc_tid; +#endif + +static void process_input (char *stuff); + +/* Trim any CR, LF from the end of line. */ + +static void trim (char *stuff) +{ + char *p; + p = stuff + strlen(stuff) - 1; + while (strlen(stuff) > 0 && (*p == '\r' || *p == '\n')) { + *p = '\0'; + p--; + } +} /* end trim */ + + +/*------------------------------------------------------------------ + * + * Name: main + * + * Purpose: Attach to KISS TNC and exchange information. + * + * Usage: See "usage" functions at end. + * + *---------------------------------------------------------------*/ + +int main (int argc, char *argv[]) +{ + text_color_init (0); // Turn off text color. + // It could interfere with trying to pipe stdout to some other application. + +#if __WIN32__ + +#else + int e; + setlinebuf (stdout); // TODO: What is the Windows equivalent? +#endif + + +/* + * Extract command line args. + */ + while (1) { + int option_index = 0; + int c; + static struct option long_options[] = { + //{"future1", 1, 0, 0}, + //{"future2", 0, 0, 0}, + //{"future3", 1, 0, 'c'}, + {0, 0, 0, 0} + }; + + /* ':' following option character means arg is required. */ + + c = getopt_long(argc, argv, "h:p:s:vf:o:T:", + long_options, &option_index); + if (c == -1) + break; + + switch (c) { + + case 'h': /* -h for hostname. */ + strlcpy (hostname, optarg, sizeof(hostname)); + break; + + case 'p': /* -p for port, either TCP or serial device. */ + strlcpy (port, optarg, sizeof(port)); + break; + + case 's': /* -s for serial port speed. */ + serial_speed = atoi(optarg); + break; + + case 'v': /* -v for verbose. */ + verbose++; + break; + + case 'f': /* -f for transmit files directory. */ + strlcpy (transmit_from, optarg, sizeof(transmit_from)); + break; + + case 'o': /* -o for receive output directory. */ + strlcpy (receive_output, optarg, sizeof(receive_output)); + break; + + case 'T': /* -T for receive timestamp. */ + strlcpy (timestamp_format, optarg, sizeof(timestamp_format)); + break; + + case '?': + /* Unknown option message was already printed. */ + usage (); + break; + + default: + /* Should not be here. */ + text_color_set(DW_COLOR_DEBUG); + dw_printf("?? getopt returned character code 0%o ??\n", c); + usage (); + } + } /* end while(1) for options */ + + if (optind < argc) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Warning: Unused command line arguments are ignored.\n"); + } + +/* + * If receive queue directory was specified, make sure that it exists. + */ + if (strlen(receive_output) > 0) { + struct stat s; + + if (stat(receive_output, &s) == 0) { + if ( ! S_ISDIR(s.st_mode)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Receive queue location, %s, is not a directory.\n", receive_output); + exit (EXIT_FAILURE); + } + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Receive queue location, %s, does not exist.\n", receive_output); + exit (EXIT_FAILURE); + } + } + +/* If port begins with digit, consider it to be TCP. */ +/* Otherwise, treat as serial port name. */ + + using_tcp = isdigit(port[0]); + +#if __WIN32__ + if (using_tcp) { + tnc_th = (HANDLE)_beginthreadex (NULL, 0, tnc_listen_net, (void *)(ptrdiff_t)99, 0, NULL); + } + else { + tnc_th = (HANDLE)_beginthreadex (NULL, 0, tnc_listen_serial, (void *)99, 0, NULL); + } + if (tnc_th == NULL) { + printf ("Internal error: Could not create TNC listen thread.\n"); + exit (EXIT_FAILURE); + } +#else + if (using_tcp) { + e = pthread_create (&tnc_tid, NULL, tnc_listen_net, (void *)(ptrdiff_t)99); + } + else { + e = pthread_create (&tnc_tid, NULL, tnc_listen_serial, (void *)(ptrdiff_t)99); + } + if (e != 0) { + perror("Internal error: Could not create TNC listen thread."); + exit (EXIT_FAILURE); + } +#endif + +// Give the threads a little while to open the TNC connection before trying to use it. +// This was a problem when the transmit queue already existed when starting up. + + SLEEP_MS (500); + +/* + * Process keyboard or other input source. + */ + char stuff[AX25_MAX_PACKET_LEN]; + + if (strlen(transmit_from) > 0) { +/* + * Process and delete all files in specified directory. + * When done, sleep for a second and try again. + * This doesn't take them in any particular order. + * A future enhancement might sort by name or timestamp. + */ + while (1) { + DIR *dp; + struct dirent *ep; + + //text_color_set(DW_COLOR_DEBUG); + //dw_printf("Get directory listing...\n"); + + dp = opendir (transmit_from); + if (dp != NULL) { + while ((ep = readdir(dp)) != NULL) { + char path [300]; + FILE *fp; + if (ep->d_name[0] == '.') + continue; + + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Processing %s for transmit...\n", ep->d_name); + strlcpy (path, transmit_from, sizeof(path)); + strlcat (path, DIR_CHAR, sizeof(path)); + strlcat (path, ep->d_name, sizeof(path)); + fp = fopen (path, "r"); + if (fp != NULL) { + while (fgets(stuff, sizeof(stuff), fp) != NULL) { + trim (stuff); + text_color_set(DW_COLOR_DEBUG); + dw_printf ("%s\n", stuff); + // TODO: Don't delete file if errors encountered? + process_input (stuff); + } + fclose (fp); + unlink (path); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf("Can't open for read: %s\n", path); + } + } + closedir (dp); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf("Can't access transmit queue directory %s. Quitting.\n", transmit_from); + exit (EXIT_FAILURE); + } + SLEEP_SEC (1); + } + } + else { +/* + * Using stdin. + */ + while (fgets(stuff, sizeof(stuff), stdin) != NULL) { + process_input (stuff); + } + } + + return (EXIT_SUCCESS); + +} /* end main */ + + + +/*------------------------------------------------------------------- + * + * Name: process_input + * + * Purpose: Process frames/commands from user, either interactively or from files. + * + * Inputs: stuff - A frame is in usual format like SOURCE>DEST,DIGI:whatever. + * Commands begin with lower case letter. + * Note that it can be modified by this function. + * + * Later Enhancement: Return success/fail status. The transmit queue processing might want + * to preserve files that were not processed as expected. + * + *--------------------------------------------------------------------*/ + +static int parse_number (char *str, int de_fault) +{ + int n; + + while (isspace(*str)) { + str++; + } + if (strlen(str) == 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Missing number for KISS command. Using default %d.\n", de_fault); + return (de_fault); + } + n = atoi(str); + if (n < 0 || n > 255) { // must fit in a byte. + text_color_set(DW_COLOR_ERROR); + dw_printf ("Number for KISS command is out of range 0-255. Using default %d.\n", de_fault); + return (de_fault); + } + return (n); +} + +static void process_input (char *stuff) +{ + char *p; + int chan = 0; + +/* + * Remove any end of line character(s). + */ + trim (stuff); + +/* + * Optional prefix, like "[9]" or "[99]" to specify channel. + */ + p = stuff; + while (isspace(*p)) p++; + if (*p == '[') { + p++; + if (p[1] == ']') { + chan = atoi(p); + p += 2; + } + else if (p[2] == ']') { + chan = atoi(p); + p += 3; + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR! One or two digit channel number and ] was expected after [ at beginning of line.\n"); + usage2(); + return; + } + if (chan < 0 || chan > 15) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR! KISS channel number must be in range of 0 thru 15.\n"); + usage2(); + return; + } + while (isspace(*p)) p++; + } + +/* + * If it starts with upper case letter or digit, assume it is an AX.25 frame in monitor format. + * Lower case is a command (e.g. Persistence or set Hardware). + * Anything else, print explanation of what is expected. + */ + if (isupper(*p) || isdigit(*p)) { + + // Parse the "TNC2 monitor format" and convert to AX.25 frame. + + unsigned char frame_data[AX25_MAX_PACKET_LEN]; + packet_t pp = ax25_from_text (p, 1); + if (pp != NULL) { + int frame_len = ax25_pack (pp, frame_data); + send_to_kiss_tnc (chan, KISS_CMD_DATA_FRAME, (char*)frame_data, frame_len); + ax25_delete (pp); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR! Could not convert to AX.25 frame: %s\n", p); + } + } + else if (islower(*p)) { + char value; + + switch (*p) { + case 'd': // txDelay, 10ms units + value = parse_number(p+1, DEFAULT_TXDELAY); + send_to_kiss_tnc (chan, KISS_CMD_TXDELAY, &value, 1); + break; + case 'p': // Persistence + value = parse_number(p+1, DEFAULT_PERSIST); + send_to_kiss_tnc (chan, KISS_CMD_PERSISTENCE, &value, 1); + break; + case 's': // Slot time, 10ms units + value = parse_number(p+1, DEFAULT_SLOTTIME); + send_to_kiss_tnc (chan, KISS_CMD_SLOTTIME, &value, 1); + break; + case 't': // txTail, 10ms units + value = parse_number(p+1, DEFAULT_TXTAIL); + send_to_kiss_tnc (chan, KISS_CMD_TXTAIL, &value, 1); + break; + case 'f': // Full duplex + value = parse_number(p+1, 0); + send_to_kiss_tnc (chan, KISS_CMD_FULLDUPLEX, &value, 1); + break; + case 'h': // set Hardware + p++; + while (*p != '\0' && isspace(*p)) { p++; } + send_to_kiss_tnc (chan, KISS_CMD_SET_HARDWARE, p, strlen(p)); + break; + default: + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid command. Must be one of d p s t f h.\n"); + usage2 (); + break; + } + } + else { + usage2 (); + } + +} /* end process_input */ + + + + +/*------------------------------------------------------------------- + * + * Name: send_to_kiss_tnc + * + * Purpose: Encapsulate the data/command, into a KISS frame, and send to the TNC. + * + * Inputs: chan - channel number. + * + * cmd - KISS_CMD_DATA_FRAME, KISS_CMD_SET_HARDWARE, etc. + * + * data - Information for KISS frame. + * + * dlen - Number of bytes in data. + * + * Description: Encapsulate as KISS frame and send to TNC. + * + *--------------------------------------------------------------------*/ + +static void send_to_kiss_tnc (int chan, int cmd, char *data, int dlen) +{ + unsigned char temp[AX25_MAX_PACKET_LEN]; // We don't limit to 256 info bytes. + unsigned char kissed[AX25_MAX_PACKET_LEN*2]; + int klen; + + if (chan < 0 || chan > 15) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR - Invalid channel %d - must be in range 0 to 15.\n", chan); + chan = 0; + } + if (cmd < 0 || cmd > 15) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR - Invalid command %d - must be in range 0 to 15.\n", cmd); + cmd = 0; + } + if (dlen < 0 || dlen > (int)(sizeof(temp)-1)) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR - Invalid data length %d - must be in range 0 to %d.\n", dlen, (int)(sizeof(temp)-1)); + dlen = sizeof(temp)-1; + } + + temp[0] = (chan << 4) | cmd; + memcpy (temp+1, data, dlen); + + klen = kiss_encapsulate(temp, dlen+1, kissed); + + if (verbose) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Sending to KISS TNC:\n"); + hex_dump (kissed, klen); + } + + if (using_tcp) { + int rc = SOCK_SEND(server_sock, (char*)kissed, klen); + if (rc != klen) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR writing KISS frame to socket.\n"); + } + } + else { + int rc = serial_port_write (serial_fd, (char*)kissed, klen); + if (rc != klen) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR writing KISS frame to serial port.\n"); + //dw_printf ("DEBUG wanted %d, got %d\n", klen, rc); + } + } + +} /* end send_to_kiss_tnc */ + + +/*------------------------------------------------------------------- + * + * Name: tnc_listen_net + * + * Purpose: Connect to KISS TNC via TCP port. + * Print everything it sends to us. + * + * Inputs: arg - Currently not used. + * + * Global In: host + * port + * + * Global Out: server_sock - Needed to send to the TNC. + * + *--------------------------------------------------------------------*/ + +static THREAD_F tnc_listen_net (void *arg) +{ + int err; + char ipaddr_str[DWSOCK_IPADDR_LEN]; // Text form of IP address. + char data[4096]; + int allow_ipv6 = 0; // Maybe someday. + int debug = 0; + int client = 0; // Not used in this situation. + kiss_frame_t kstate; + + memset (&kstate, 0, sizeof(kstate)); + + err = dwsock_init (); + if (err < 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Network interface failure. Can't go on.\n"); + exit (EXIT_FAILURE); + } + +/* + * Connect to network KISS TNC. + */ + // For the IGate we would loop around and try to reconnect if the TNC + // goes away. We should probably do the same here. + + server_sock = dwsock_connect (hostname, port, "TCP KISS TNC", allow_ipv6, debug, ipaddr_str); + + if (server_sock == -1) { + text_color_set(DW_COLOR_ERROR); + // Should have been a message already. What else is there to say? + exit (EXIT_FAILURE); + } + +/* + * Print what we get from TNC. + */ + int len; + + while ((len = SOCK_RECV (server_sock, (char*)(data), sizeof(data))) > 0) { + int j; + for (j = 0; j < len; j++) { + + // Feed in one byte at a time. + // kiss_process_msg is called when a complete frame has been accumulated. + + // When verbose is specified, we get debug output like this: + // + // <<< Data frame from KISS client application, port 0, total length = 46 + // 000: c0 00 82 a0 88 ae 62 6a e0 ae 84 64 9e a6 b4 ff ......bj...d.... + // ... + // It says "from KISS client application" because it was written + // on the assumption it was being used in only one direction. + // Not worried enough about it to do anything at this time. + + kiss_rec_byte (&kstate, data[j], verbose, NULL, client, NULL); + } + } + + text_color_set(DW_COLOR_ERROR); + dw_printf ("Read error from TCP KISS TNC. Terminating.\n"); + exit (EXIT_FAILURE); + +} /* end tnc_listen_net */ + + +/*------------------------------------------------------------------- + * + * Name: tnc_listen_serial + * + * Purpose: Connect to KISS TNC via serial port. + * Print everything it sends to us. + * + * Inputs: arg - Currently not used. + * + * Global In: port + * serial_speed + * + * Global Out: serial_fd - Need for sending to the TNC. + * + *--------------------------------------------------------------------*/ + +static THREAD_F tnc_listen_serial (void *arg) +{ + int client = 0; + kiss_frame_t kstate; + + memset (&kstate, 0, sizeof(kstate)); + + serial_fd = serial_port_open (port, serial_speed); + + if (serial_fd == MYFDERROR) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Unable to connect to KISS TNC serial port %s.\n", port); +#if __WIN32__ +#else + // More detail such as "permission denied" or "no such device" + dw_printf("%s\n", strerror(errno)); +#endif + exit (EXIT_FAILURE); + } + +/* + * Read and print. + */ + while (1) { + int ch; + + ch = serial_port_get1(serial_fd); + + if (ch < 0) { + dw_printf("Read error from serial port KISS TNC.\n"); + exit (EXIT_FAILURE); + } + + // Feed in one byte at a time. + // kiss_process_msg is called when a complete frame has been accumulated. + + kiss_rec_byte (&kstate, ch, verbose, NULL, client, NULL); + } + +} /* end tnc_listen_serial */ + + + +/*------------------------------------------------------------------- + * + * Name: kiss_process_msg + * + * Purpose: Process a frame from the KISS TNC. + * This is called when a complete frame has been accumulated. + * In this case, we simply print it. + * + * Inputs: kiss_msg - Kiss frame with FEND and escapes removed. + * The first byte contains channel and command. + * + * kiss_len - Number of bytes including the command. + * + * debug - Debug option is selected. + * + * client - Not used in this case. + * + * sendfun - Not used in this case. + * + *-----------------------------------------------------------------*/ + +void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, struct kissport_status_s *kps, int client, + void (*sendfun)(int chan, int kiss_cmd, unsigned char *fbuf, int flen, struct kissport_status_s *onlykps, int onlyclient)) +{ + int chan; + int cmd; + packet_t pp; + alevel_t alevel; + + chan = (kiss_msg[0] >> 4) & 0xf; + cmd = kiss_msg[0] & 0xf; + + switch (cmd) + { + case KISS_CMD_DATA_FRAME: /* 0 = Data Frame */ + + memset (&alevel, 0, sizeof(alevel)); + pp = ax25_from_frame (kiss_msg+1, kiss_len-1, alevel); + if (pp == NULL) { + text_color_set(DW_COLOR_ERROR); + printf ("ERROR - Invalid KISS data frame from TNC.\n"); + } + else { + char prefix[120]; // Channel and optional timestamp. + // Like [0] or [2 12:34:56] + + char addrs[AX25_MAX_ADDRS*AX25_MAX_ADDR_LEN]; // Like source>dest,digi,...,digi: + unsigned char *pinfo; + int info_len; + + if (strlen(timestamp_format) > 0) { + char ts[100]; + timestamp_user_format (ts, sizeof(ts), timestamp_format); + snprintf (prefix, sizeof(prefix), "[%d %s]", chan, ts); + } + else { + snprintf (prefix, sizeof(prefix), "[%d]", chan); + } + + ax25_format_addrs (pp, addrs); + + info_len = ax25_get_info (pp, &pinfo); + + text_color_set(DW_COLOR_REC); + + dw_printf ("%s %s", prefix, addrs); // [channel] Addresses followed by : + + // Safe print will replace any unprintable characters with + // hexadecimal representation. + + ax25_safe_print ((char *)pinfo, info_len, 0); + dw_printf ("\n"); +#if __WIN32__ + fflush (stdout); +#endif + +/* + * Add to receive queue directory if specified. + * File name will be based on current local time. + * If you want UTC, just set an environment variable like this: + * + * TZ=UTC kissutil ... + */ + if (strlen(receive_output) > 0) { + char fname [30]; + char path [300]; + FILE *fp; + + timestamp_filename (fname, (int)sizeof(fname)); + + strlcpy (path, receive_output, sizeof(path)); + strlcat (path, DIR_CHAR, sizeof(path)); + strlcat (path, fname, sizeof(path)); + + text_color_set(DW_COLOR_DEBUG); + dw_printf ("Save received frame to %s\n", path); + fp = fopen (path, "w"); + if (fp != NULL) { + fprintf (fp, "%s %s%s\n", prefix, addrs, pinfo); + fclose (fp); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Unable to open for write: %s\n", path); + } + } + + ax25_delete (pp); + } + break; + + case KISS_CMD_SET_HARDWARE: /* 6 = TNC specific */ + + kiss_msg[kiss_len] = '\0'; + text_color_set(DW_COLOR_REC); + // Display as "h ..." for in/out symmetry. + // Use safe print here? + dw_printf ("[%d] h %s\n", chan, (char*)(kiss_msg+1)); + break; + +/* + * The rest should only go TO the TNC and not come FROM it. + */ + case KISS_CMD_TXDELAY: /* 1 = TXDELAY */ + case KISS_CMD_PERSISTENCE: /* 2 = Persistence */ + case KISS_CMD_SLOTTIME: /* 3 = SlotTime */ + case KISS_CMD_TXTAIL: /* 4 = TXtail */ + case KISS_CMD_FULLDUPLEX: /* 5 = FullDuplex */ + case KISS_CMD_END_KISS: /* 15 = End KISS mode, port should be 15. */ + default: + + text_color_set(DW_COLOR_ERROR); + printf ("Unexpected KISS command %d, channel %d\n", cmd, chan); + break; + } + +} /* end kiss_process_msg */ + + +// TODO: We have multiple copies of this. Move to some misc file. + +void hex_dump (unsigned char *p, int len) +{ + int n, i, offset; + + offset = 0; + while (len > 0) { + n = len < 16 ? len : 16; + printf (" %03x: ", offset); + for (i=0; i= 1) { @@ -124,9 +149,12 @@ void latitude_to_str (double dlat, int ambiguity, char *slat) * Inputs: dlong - Floating point degrees. * ambiguity - If 1, 2, 3, or 4, blank out that many trailing digits. * - * Outputs: slat - String in format dddmm.mm[NS] - * Should always be exactly 9 characters + NUL. - * + * Outputs: slong - String in format dddmm.mm[NS] + * Must always be exactly 9 characters + NUL. + * Put in leading zeros if necessary. + * We must have exactly dddmm.mm and hemisphere because + * the APRS position report has fixed width fields. + * Trailing digits can be blanked for position ambiguity. * Returns: None * *----------------------------------------------------------------*/ @@ -167,7 +195,12 @@ void longitude_to_str (double dlong, int ambiguity, char *slong) ideg++; } + // Assumes slong can hold 9 characters + nul. + // Degrees must be exactly 3 digits, with leading zero, if needed. + + // FIXME: Should pass in sizeof slong and use snprintf sprintf (slong, "%03d%s%c", ideg, smin, hemi); + /* * The spec says position ambiguity in latitude also * applies to longitude automatically. @@ -341,6 +374,7 @@ void latitude_to_nmea (double dlat, char *slat, char *hemi) ideg++; } + // FIXME: Should pass in sizeof slat and use snprintf sprintf (slat, "%02d%s", ideg, smin); } /* end latitude_to_str */ @@ -402,6 +436,7 @@ void longitude_to_nmea (double dlong, char *slong, char *hemi) ideg++; } + // FIXME: Should pass in sizeof slong and use snprintf sprintf (slong, "%03d%s", ideg, smin); } /* end longitude_to_nmea */ @@ -555,6 +590,92 @@ double ll_distance_km (double lat1, double lon1, double lat2, double lon2) } +/*------------------------------------------------------------------ + * + * Function: ll_bearing_deg + * + * Purpose: Calculate bearing between two locations. + * + * Inputs: lat1, lon1 - starting location, in degrees. + * lat2, lon2 - destination location + * + * Returns: Initial Bearing, in degrees. + * The calculation produces Range +- 180 degrees. + * But I think that 0 - 360 would be more customary? + * + *------------------------------------------------------------------*/ + +double ll_bearing_deg (double lat1, double lon1, double lat2, double lon2) +{ + double b; + + lat1 *= M_PI / 180; + lon1 *= M_PI / 180; + lat2 *= M_PI / 180; + lon2 *= M_PI / 180; + + b = atan2 (sin(lon2-lon1) * cos(lat2), + cos(lat1) * sin(lat2) - sin(lat1) * cos(lat2) * cos(lon2-lon1)); + + b *= 180 / M_PI; + if (b < 0) b += 360; + + return (b); +} + + +/*------------------------------------------------------------------ + * + * Function: ll_dest_lat + * ll_dest_lon + * + * Purpose: Calculate the destination location given a starting point, + * distance, and bearing, + * + * Inputs: lat1, lon1 - starting location, in degrees. + * dist - distance in km. + * bearing - direction in degrees. Shouldn't matter + * if it is in +- 180 or 0 to 360 range. + * + * Returns: New latitude or longitude. + * + *------------------------------------------------------------------*/ + +double ll_dest_lat (double lat1, double lon1, double dist, double bearing) +{ + double lat2; + + lat1 *= M_PI / 180; // Everything to radians. + lon1 *= M_PI / 180; + bearing *= M_PI / 180; + + lat2 = asin(sin(lat1) * cos(dist/R) + cos(lat1) * sin(dist/R) * cos(bearing)); + + lat2 *= 180 / M_PI; // Back to degrees. + + return (lat2); +} + +double ll_dest_lon (double lat1, double lon1, double dist, double bearing) +{ + double lon2; + double lat2; + + lat1 *= M_PI / 180; // Everything to radians. + lon1 *= M_PI / 180; + bearing *= M_PI / 180; + + lat2 = asin(sin(lat1) * cos(dist/R) + cos(lat1) * sin(dist/R) * cos(bearing)); + + lon2 = lon1 + atan2(sin(bearing) * sin(dist/R) * cos(lat1), cos(dist/R) - sin(lat1) * sin(lat2)); + + lon2 *= 180 / M_PI; // Back to degrees. + + return (lon2); +} + + + /*------------------------------------------------------------------ * * Function: ll_from_grid_square @@ -776,6 +897,7 @@ int main (int argc, char *argv[]) int errors = 0; int ok; double dlat, dlon; + double d, b; /* Latitude to APRS format. */ @@ -805,6 +927,14 @@ int main (int argc, char *argv[]) latitude_to_str (45.999830, 4, result); if (strcmp(result, "45 . N") != 0) { errors++; dw_printf ("Error 1.8: Did not expect \"%s\"\n", result); } + // Test for leading zeros for small values. Result must be fixed width. + + latitude_to_str (0.016666666, 0, result); + if (strcmp(result, "0001.00N") != 0) { errors++; dw_printf ("Error 1.9: Did not expect \"%s\"\n", result); } + + latitude_to_str (-1.999999, 0, result); + if (strcmp(result, "0200.00S") != 0) { errors++; dw_printf ("Error 1.10: Did not expect \"%s\"\n", result); } + /* Longitude to APRS format. */ longitude_to_str (45.25, 0, result); @@ -833,6 +963,15 @@ int main (int argc, char *argv[]) longitude_to_str (45.999830, 4, result); if (strcmp(result, "045 . E") != 0) { errors++; dw_printf ("Error 2.8: Did not expect \"%s\"\n", result); } + // Test for leading zeros for small values. Result must be fixed width. + + longitude_to_str (0.016666666, 0, result); + if (strcmp(result, "00001.00E") != 0) { errors++; dw_printf ("Error 2.9: Did not expect \"%s\"\n", result); } + + longitude_to_str (-1.999999, 0, result); + if (strcmp(result, "00200.00W") != 0) { errors++; dw_printf ("Error 2.10: Did not expect \"%s\"\n", result); } + + /* Compressed format. */ /* Protocol spec example has <*e7 but I got <*e8 due to rounding rather than truncation to integer. */ @@ -860,6 +999,55 @@ int main (int argc, char *argv[]) // to be continued for others... NMEA... +/* Distance & bearing - Take a couple examples from other places and see if we get similar results. */ + + // http://www.movable-type.co.uk/scripts/latlong.html + + d = ll_distance_km (35., 45., 35., 135.); + b = ll_bearing_deg (35., 45., 35., 135.); + + if (d < 7862 || d > 7882) { errors++; dw_printf ("Error 5.1: Did not expect distance %.1f\n", d); } + + if (b < 59.7 || b > 60.3) { errors++; dw_printf ("Error 5.2: Did not expect bearing %.1f\n", b); } + + // Sydney to Kinsale. https://woodshole.er.usgs.gov/staffpages/cpolloni/manitou/ccal.htm + + d = ll_distance_km (-33.8688, 151.2093, 51.7059, -8.5222); + b = ll_bearing_deg (-33.8688, 151.2093, 51.7059, -8.5222); + + if (d < 17435 || d > 17455) { errors++; dw_printf ("Error 5.3: Did not expect distance %.1f\n", d); } + + if (b < 327-1 || b > 327+1) { errors++; dw_printf ("Error 5.4: Did not expect bearing %.1f\n", b); } + + +/* + * More distance and bearing. + * Here we will start at some location1 (lat1,lon1) and go some distance (d1) at some bearing (b1). + * This results in a new location2 (lat2, lon2). + * We then calculate the distance and bearing from location1 to location2 and compare with the intention. + */ + int lat1, lon1, d1 = 10, b1; + double lat2, lon2, d2, b2; + + for (lat1 = -60; lat1 <= 60; lat1 += 30) { + for (lon1 = -180; lon1 <= 180; lon1 +=30) { + for (b1 = 0; b1 < 360; b1 += 15) { + + lat2 = ll_dest_lat ((double)lat1, (double)lon1, (double)d1, (double)b1); + lon2 = ll_dest_lon ((double)lat1, (double)lon1, (double)d1, (double)b1); + + d2 = ll_distance_km ((double)lat1, (double)lon1, lat2, lon2); + b2 = ll_bearing_deg ((double)lat1, (double)lon1, lat2, lon2); + if (b2 > 359.9 && b2 < 360.1) b2 = 0; + + // must be within 0.1% of distance and 0.1 degree. + if (d2 < 0.999 * d1 || d2 > 1.001 * d1) { errors++; dw_printf ("Error 5.8: lat1=%d, lon2=%d, d1=%d, b1=%d, d2=%.2f\n", lat1, lon1, d1, b1, d2); } + if (b2 < b1 - 0.1 || b2 > b1 + 0.1) { errors++; dw_printf ("Error 5.9: lat1=%d, lon2=%d, d1=%d, b1=%d, b2=%.2f\n", lat1, lon1, d1, b1, b2); } + } + } + } + + /* Maidenhead locator to lat/long. */ diff --git a/latlong.h b/src/latlong.h similarity index 100% rename from latlong.h rename to src/latlong.h diff --git a/ll2utm.c b/src/ll2utm.c similarity index 100% rename from ll2utm.c rename to src/ll2utm.c diff --git a/log.c b/src/log.c similarity index 62% rename from log.c rename to src/log.c index fa86c3e2..d7ef544b 100644 --- a/log.c +++ b/src/log.c @@ -28,6 +28,13 @@ * unreadable, format, write separated properties into * CSV format for easy reading and later processing. * + * There are two alternatives here. + * + * -L logfile Specify full file path. + * + * -l logdir Daily names will be created here. + * + * Use one or the other but not both. * *------------------------------------------------------------------*/ @@ -44,6 +51,10 @@ #include #include +#if __WIN32__ +#include // for _mkdir() +#endif + #include "ax25_pad.h" #include "textcolor.h" #include "decode_aprs.h" @@ -91,26 +102,39 @@ static void quote_for_csv (char *out, size_t outsize, const char *in) { * * Purpose: Initialization at start of application. * - * Inputs: path - Path of log file directory. + * Inputs: daily_names - True if daily names should be generated. + * In this case path is a directory. + * When false, path would be the file name. + * + * path - Log file name or just directory. * Use "." for current directory. * Empty string disables feature. * - * Global Out: g_log_dir - Save directory here for later use. + * Global Out: g_daily_names - True if daily names should be generated. + * + * g_log_path - Save directory or full name here for later use. + * * g_log_fp - File pointer for writing. + * Note that file is kept open. + * We don't open/close for every new item. + * * g_open_fname - Name of currently open file. + * Applicable only when g_daily_names is true. * *------------------------------------------------------------------*/ -static char g_log_dir[80]; +static int g_daily_names; +static char g_log_path[80]; static FILE *g_log_fp; static char g_open_fname[20]; -void log_init (char *path) +void log_init (int daily_names, char *path) { struct stat st; - strlcpy (g_log_dir, "", sizeof(g_log_dir)); + g_daily_names = daily_names; + strlcpy (g_log_path, "", sizeof(g_log_path)); g_log_fp = NULL; strlcpy (g_open_fname, "", sizeof(g_open_fname)); @@ -118,42 +142,57 @@ void log_init (char *path) return; } - if (stat(path,&st) == 0) { - // Exists, but is it a directory? - if (S_ISDIR(st.st_mode)) { - // Specified directory exists. - strlcpy (g_log_dir, path, sizeof(g_log_dir)); + if (g_daily_names) { + +// Original strategy. Automatic daily file names. + + if (stat(path,&st) == 0) { + // Exists, but is it a directory? + if (S_ISDIR(st.st_mode)) { + // Specified directory exists. + strlcpy (g_log_path, path, sizeof(g_log_path)); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Log file location \"%s\" is not a directory.\n", path); + dw_printf ("Using current working directory \".\" instead.\n"); + strlcpy (g_log_path, ".", sizeof(g_log_path)); + } } else { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Log file location \"%s\" is not a directory.\n", path); - dw_printf ("Using current working directory \".\" instead.\n"); - strlcpy (g_log_dir, ".", sizeof(g_log_dir)); - } - } - else { - // Doesn't exist. Try to create it. - // parent directory must exist. - // We don't create multiple levels like "mkdir -p" + // Doesn't exist. Try to create it. + // parent directory must exist. + // We don't create multiple levels like "mkdir -p" #if __WIN32__ - if (_mkdir (path) == 0) { + if (_mkdir (path) == 0) { #else - if (mkdir (path, 0777) == 0) { + if (mkdir (path, 0777) == 0) { #endif - // Success. - text_color_set(DW_COLOR_INFO); - dw_printf ("Log file location \"%s\" has been created.\n", path); - strlcpy (g_log_dir, path, sizeof(g_log_dir)); - } - else { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Failed to create log file location \"%s\".\n", path); - dw_printf ("%s\n", strerror(errno)); - dw_printf ("Using current working directory \".\" instead.\n"); - strlcpy (g_log_dir, ".", sizeof(g_log_dir)); + // Success. + text_color_set(DW_COLOR_INFO); + dw_printf ("Log file location \"%s\" has been created.\n", path); + strlcpy (g_log_path, path, sizeof(g_log_path)); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Failed to create log file location \"%s\".\n", path); + dw_printf ("%s\n", strerror(errno)); + dw_printf ("Using current working directory \".\" instead.\n"); + strlcpy (g_log_path, ".", sizeof(g_log_path)); + } } } -} + else { + +// Added in version 1.5. Single file. +// Typically logrotate would be used to keep size under control. + + text_color_set(DW_COLOR_INFO); + dw_printf ("Log file is \"%s\"\n", path); + strlcpy (g_log_path, path, sizeof(g_log_path)); + } + +} /* end log_init */ @@ -177,72 +216,120 @@ void log_init (char *path) void log_write (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, retry_t retries) { - time_t now; // make 'now' a parameter so we can process historical data ??? - char fname[20]; + time_t now; struct tm tm; - if (strlen(g_log_dir) == 0) return; + if (strlen(g_log_path) == 0) return; - // Generate the file name from current date, UTC. - - now = time(NULL); + now = time(NULL); // Get current time. (void)gmtime_r (&now, &tm); +// FIXME: https://github.com/wb2osz/direwolf/issues/473 - // Microsoft doesn't recognize %F as equivalent to %Y-%m-%d + if (g_daily_names) { - strftime (fname, sizeof(fname), "%Y-%m-%d.log", &tm); +// Original strategy. Automatic daily file names. - // Close current file if name has changed + char fname[20]; - if (g_log_fp != NULL && strcmp(fname, g_open_fname) != 0) { - log_term (); - } + // Generate the file name from current date, UTC. + // Why UTC rather than local time? I don't recall the reasoning. + // It's been there a few years and no on complained so leave it alone for now. - // Open for append if not already open. + // Microsoft doesn't recognize %F as equivalent to %Y-%m-%d - if (g_log_fp == NULL) { - char full_path[120]; - struct stat st; - int already_there; + strftime (fname, sizeof(fname), "%Y-%m-%d.log", &tm); - strlcpy (full_path, g_log_dir, sizeof(full_path)); + // Close current file if name has changed + + if (g_log_fp != NULL && strcmp(fname, g_open_fname) != 0) { + log_term (); + } + + // Open for append if not already open. + + if (g_log_fp == NULL) { + char full_path[120]; + struct stat st; + int already_there; + + strlcpy (full_path, g_log_path, sizeof(full_path)); #if __WIN32__ - strlcat (full_path, "\\", sizeof(full_path)); + strlcat (full_path, "\\", sizeof(full_path)); #else - strlcat (full_path, "/", sizeof(full_path)); + strlcat (full_path, "/", sizeof(full_path)); #endif - strlcat (full_path, fname, sizeof(full_path)); + strlcat (full_path, fname, sizeof(full_path)); - // See if it already exists. - // This is used later to write a header if it did not exist already. + // See if file already exists and not empty. + // This is used later to write a header if it did not exist already. - already_there = stat(full_path,&st) == 0; + already_there = (stat(full_path,&st) == 0) && (st.st_size > 0); - text_color_set(DW_COLOR_INFO); - dw_printf("Opening log file \"%s\".\n", fname); + text_color_set(DW_COLOR_INFO); + dw_printf("Opening log file \"%s\".\n", fname); - g_log_fp = fopen (full_path, "a"); + g_log_fp = fopen (full_path, "a"); - if (g_log_fp != NULL) { - strlcpy (g_open_fname, fname, sizeof(g_open_fname)); - } - else { - text_color_set(DW_COLOR_ERROR); - dw_printf("Can't open log file \"%s\" for write.\n", full_path); - dw_printf ("%s\n", strerror(errno)); - strlcpy (g_open_fname, "", sizeof(g_open_fname)); - return; - } + if (g_log_fp != NULL) { + strlcpy (g_open_fname, fname, sizeof(g_open_fname)); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf("Can't open log file \"%s\" for write.\n", full_path); + dw_printf ("%s\n", strerror(errno)); + strlcpy (g_open_fname, "", sizeof(g_open_fname)); + return; + } - // Write a header suitable for importing into a spreadsheet - // only if this will be the first line. + // Write a header suitable for importing into a spreadsheet + // only if this will be the first line. - if ( ! already_there) { - fprintf (g_log_fp, "chan,utime,isotime,source,heard,level,error,dti,name,symbol,latitude,longitude,speed,course,altitude,frequency,offset,tone,system,status,comment\n"); + if ( ! already_there) { + fprintf (g_log_fp, "chan,utime,isotime,source,heard,level,error,dti,name,symbol,latitude,longitude,speed,course,altitude,frequency,offset,tone,system,status,telemetry,comment\n"); + } + } + } + else { + +// Added in version 1.5. Single file. + + // Open for append if not already open. + + if (g_log_fp == NULL) { + struct stat st; + int already_there; + + // See if file already exists and not empty. + // This is used later to write a header if it did not exist already. + + already_there = (stat(g_log_path,&st) == 0) && (st.st_size > 0); + + text_color_set(DW_COLOR_INFO); + dw_printf("Opening log file \"%s\"\n", g_log_path); + + g_log_fp = fopen (g_log_path, "a"); + + if (g_log_fp == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Can't open log file \"%s\" for write.\n", g_log_path); + dw_printf ("%s\n", strerror(errno)); + strlcpy (g_log_path, "", sizeof(g_log_path)); + return; + } + + // Write a header suitable for importing into a spreadsheet + // only if this will be the first line. + + if ( ! already_there) { + fprintf (g_log_fp, "chan,utime,isotime,source,heard,level,error,dti,name,symbol,latitude,longitude,speed,course,altitude,frequency,offset,tone,system,status,telemetry,comment\n"); + } } } + +// Add line to file if it is now open. + if (g_log_fp != NULL) { char itime[24]; @@ -258,7 +345,7 @@ void log_write (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, retry_ char sstatus[40]; char stelemetry[200]; char scomment[256]; - char alevel_text[32]; + char alevel_text[40]; @@ -439,7 +526,13 @@ void log_term (void) if (g_log_fp != NULL) { text_color_set(DW_COLOR_INFO); - dw_printf("Closing log file \"%s\".\n", g_open_fname); + + if (g_daily_names) { + dw_printf("Closing log file \"%s\".\n", g_open_fname); + } + else { + dw_printf("Closing log file \"%s\".\n", g_log_path); + } fclose (g_log_fp); diff --git a/log.h b/src/log.h similarity index 86% rename from log.h rename to src/log.h index fc6ca441..3afb6b17 100644 --- a/log.h +++ b/src/log.h @@ -10,7 +10,7 @@ -void log_init (char *path); +void log_init (int daily_names, char *path); void log_write (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, retry_t retries); diff --git a/log2gpx.c b/src/log2gpx.c similarity index 97% rename from log2gpx.c rename to src/log2gpx.c index 15b9835e..15d389f4 100644 --- a/log2gpx.c +++ b/src/log2gpx.c @@ -258,8 +258,6 @@ static void read_csv(FILE *fp) float speed = UNKNOWN_VALUE; float course = UNKNOWN_VALUE; float alt = UNKNOWN_VALUE; - double freq = UNKNOWN_VALUE; - int offset = UNKNOWN_VALUE; char stemp[16], desc[32], comment[256]; if (pspeed != NULL && strlen(pspeed) > 0) { @@ -275,7 +273,7 @@ static void read_csv(FILE *fp) /* combine freq/offset/tone into one description string. */ if (pfreq != NULL && strlen(pfreq) > 0) { - freq = atof(pfreq); + double freq = atof(pfreq); snprintf (desc, sizeof(desc), "%.3f MHz", freq); } else { @@ -283,7 +281,7 @@ static void read_csv(FILE *fp) } if (poffset != NULL && strlen(poffset) > 0) { - offset = atoi(poffset); + int offset = atoi(poffset); if (offset != 0 && offset % 1000 == 0) { snprintf (stemp, sizeof(stemp), "%+dM", offset / 1000); } @@ -325,10 +323,10 @@ static void read_csv(FILE *fp) things[num_things].speed = speed; things[num_things].course = course; things[num_things].alt = alt; - strncpy (things[num_things].time, pisotime, sizeof(things[num_things].time)); - strncpy (things[num_things].name, pname, sizeof(things[num_things].name)); - strncpy (things[num_things].desc, desc, sizeof(things[num_things].desc)); - strncpy (things[num_things].comment, comment, sizeof(things[num_things].comment)); + strlcpy (things[num_things].time, pisotime, sizeof(things[num_things].time)); + strlcpy (things[num_things].name, pname, sizeof(things[num_things].name)); + strlcpy (things[num_things].desc, desc, sizeof(things[num_things].desc)); + strlcpy (things[num_things].comment, comment, sizeof(things[num_things].comment)); num_things++; } @@ -543,4 +541,4 @@ static void process_things (int first, int last) } printf (" %s\n", safe_name); printf (" \n"); -} \ No newline at end of file +} diff --git a/mgn_icon.h b/src/mgn_icon.h similarity index 97% rename from mgn_icon.h rename to src/mgn_icon.h index 4563cce1..c870bc0e 100644 --- a/mgn_icon.h +++ b/src/mgn_icon.h @@ -208,9 +208,9 @@ static const char mgn_alternate_symtab[SYMTAB_SIZE][3] = { MGN_park, // ; 27 Park/Picnic area MGN_default, // < 28 ADVISORY (one WX flag) MGN_default, // = 29 APRStt Touchtone (DTMF users) - MGN_default, // > 30 OVERLAYED CAR + MGN_default, // > 30 OVERLAID CAR MGN_tourist_info, // ? 31 INFO Kiosk (Blue box with ?) - MGN_default, // @ 32 HURICANE/Trop-Storm + MGN_default, // @ 32 HURRICANE/Trop-Storm MGN_box, // A 33 overlayBOX DTMF & RFID & XO MGN_default, // B 34 Blwng Snow (& future codes) MGN_boating, // C 35 Coast Guard @@ -220,7 +220,7 @@ static const char mgn_alternate_symtab[SYMTAB_SIZE][3] = { MGN_default, // G 39 Snow Shwr (& future ovrlys) MGN_default, // H 40 Haze (& Overlay Hazards) MGN_default, // I 41 Rain Shower - MGN_default, // J 42 Lightening (& future ovrlys) + MGN_default, // J 42 Lightning (& future ovrlys) MGN_default, // K 43 Kenwood HT (W) MGN_lighthouse, // L 44 Lighthouse MGN_default, // M 45 MARS (A=Army,N=Navy,F=AF) @@ -263,12 +263,12 @@ static const char mgn_alternate_symtab[SYMTAB_SIZE][3] = { MGN_default, // r 82 Restrooms MGN_default, // s 83 OVERLAY SHIP/boat (top view) MGN_default, // t 84 Tornado - MGN_default, // u 85 OVERLAYED TRUCK - MGN_default, // v 86 OVERLAYED Van + MGN_default, // u 85 OVERLAID TRUCK + MGN_default, // v 86 OVERLAID Van MGN_default, // w 87 Flooding MGN_wreck, // x 88 Wreck or Obstruction ->X<- MGN_default, // y 89 Skywarn - MGN_default, // z 90 OVERLAYED Shelter + MGN_default, // z 90 OVERLAID Shelter MGN_default, // { 91 Fog (& future ovrly codes) MGN_default, // | 92 TNC Stream Switch MGN_default, // } 93 diff --git a/mheard.c b/src/mheard.c similarity index 86% rename from mheard.c rename to src/mheard.c index 37163c52..f11c68f0 100644 --- a/mheard.c +++ b/src/mheard.c @@ -102,7 +102,7 @@ typedef struct mheard_s { double dlat, dlon; // Last position. G_UNKNOWN for unknown. - int msp; // Allow message sender positon report. + int msp; // Allow message sender position report. // When non zero, an IS>RF position report is allowed. // Then decremented. @@ -243,9 +243,9 @@ static void mheard_dump (void) int i; mheard_t *mptr; time_t now = time(NULL); - char stuff[80]; - char rf[16]; // hours:minutes - char is[16]; + char stuff[120]; + char rf[20]; // hours:minutes + char is[20]; char position[40]; mheard_t *station[MAXDUMP]; int num_stations = 0; @@ -336,6 +336,52 @@ void mheard_save_rf (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, r */ hops = ax25_get_heard(pp) - AX25_SOURCE; +/* + * Consider the following scenario: + * + * (1) We hear AA1PR-9 by a path of 4 digipeaters. + * Looking closer, it's probably only two because there are left over WIDE1-0 and WIDE2-0. + * + * Digipeater WIDE2 (probably N3LLO-3) audio level = 72(19/15) [NONE] _|||||___ + * [0.3] AA1PR-9>APY300,K1EQX-7,WIDE1,N3LLO-3,WIDE2*,ARISS::ANSRVR :cq hotg vt aprsthursday{01<0x0d> + * ----- ----- + * + * (2) APRS-IS sends a response to us. + * + * [ig>tx] ANSRVR>APWW11,KJ4ERJ-15*,TCPIP*,qAS,KJ4ERJ-15::AA1PR-9 :N:HOTG 161 Messages Sent{JL} + * + * (3) Here is our analysis of whether it should be sent to RF. + * + * Was message addressee AA1PR-9 heard in the past 180 minutes, with 2 or fewer digipeater hops? + * No, AA1PR-9 was last heard over the radio with 4 digipeater hops 0 minutes ago. + * + * The wrong hop count caused us to drop a packet that should have been transmitted. + * We could put in a hack to not count the "WIDEn-0" addresses. + * That is not correct because other prefixes could be used and we don't know + * what they are for other digipeaters. + * I think the best solution is to simply ignore the hop count. + * Maybe next release will have a major cleanup. + */ + + // HACK - Reduce hop count by number of used WIDEn-0 addresses. + + if (hops > 1) { + for (int k = 0; k < ax25_get_num_repeaters(pp); k++) { + char digi[AX25_MAX_ADDR_LEN]; + ax25_get_addr_no_ssid (pp, AX25_REPEATER_1 + k, digi); + int ssid = ax25_get_ssid (pp, AX25_REPEATER_1 + k); + int used = ax25_get_h (pp, AX25_REPEATER_1 + k); + + //text_color_set(DW_COLOR_DEBUG); + //dw_printf ("Examining %s-%d used=%d.\n", digi, ssid, used); + + if (used && strlen(digi) == 5 && strncmp(digi, "WIDE", 4) == 0 && isdigit(digi[4]) && ssid == 0) { + hops--; + //text_color_set(DW_COLOR_DEBUG); + //dw_printf ("Decrease hop count to %d for problematic %s.\n", hops, digi); + } + } + } mptr = mheard_ptr(source); if (mptr == NULL) { @@ -350,6 +396,11 @@ void mheard_save_rf (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, r } mptr = calloc(sizeof(mheard_t),1); + if (mptr == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } strlcpy (mptr->callsign, source, sizeof(mptr->callsign)); mptr->count = 1; mptr->chan = chan; @@ -429,6 +480,7 @@ void mheard_save_rf (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, r * N1HKO-10>APJI40,TCPIP*,qAC,N1HKO-JS:APWW10,WIDE1-1,WIDE2-1,qAS,K1RI:/221700h/9AmAT3PQ3S,WIDE1-1,WIDE2-1,qAR,W1TG-1:`c)@qh\>/"50}TinyTrak4 Mobile + * WHO-IS>APJIW4,TCPIP*,qAC,AE5PL-JF::WB2OSZ :C/Billerica Amateur Radio Society/MA/United States{XF}WO * * Notice how the final address in the header might not * be a valid AX.25 address. We see a 9 character address @@ -438,6 +490,7 @@ void mheard_save_rf (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, r * a clue about the journey taken but I don't think we care here. * * All we should care about here is the the source address. + * Note that the source address might not adhere to the AX.25 format. * * Description: * @@ -445,21 +498,25 @@ void mheard_save_rf (int chan, decode_aprs_t *A, packet_t pp, alevel_t alevel, r void mheard_save_is (char *ptext) { - packet_t pp; time_t now = time(NULL); char source[AX25_MAX_ADDR_LEN]; - mheard_t *mptr; + +#if 1 +// It is possible that source won't adhere to the AX.25 restrictions. +// So we simply extract the source address, as text, from the beginning rather than +// using ax25_from_text() and ax25_get_addr_with_ssid(). + + memset (source, 0, sizeof(source)); + memcpy (source, ptext, sizeof(source)-1); + char *g = strchr(source, '>'); + if (g != NULL) *g = '\0'; + +#else /* - * Try to parse it into a packet object. - * This will contain "q constructs" and we might see an address - * with two alphnumeric characters in the SSID so we must use - * the non-strict parsing. - * - * Bug: Up to 8 digipeaters are allowed in radio format. - * There is a potential of finding a larger number here. + * Keep this here in case I want to revive it to get location. */ - pp = ax25_from_text(ptext, 0); + packet_t pp = ax25_from_text(ptext, 0); if (pp == NULL) { if (mheard_debug) { @@ -470,13 +527,17 @@ void mheard_save_is (char *ptext) return; } - ax25_get_addr_with_ssid (pp, AX25_SOURCE, source); + //////ax25_get_addr_with_ssid (pp, AX25_SOURCE, source); +#endif - mptr = mheard_ptr(source); + mheard_t *mptr = mheard_ptr(source); if (mptr == NULL) { int i; /* * Not heard before. Add it. + * Observation years later: + * Hmmmm. I wonder why I did not store the location if available. + * An earlier example has an APRSdroid station reporting location without using [ham] RF. */ if (mheard_debug) { @@ -485,6 +546,11 @@ void mheard_save_is (char *ptext) } mptr = calloc(sizeof(mheard_t),1); + if (mptr == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } strlcpy (mptr->callsign, source, sizeof(mptr->callsign)); mptr->count = 1; mptr->last_heard_is = now; @@ -527,7 +593,9 @@ void mheard_save_is (char *ptext) mheard_dump (); } +#if 0 ax25_delete (pp); +#endif } /* end mheard_save_is */ @@ -549,7 +617,7 @@ void mheard_save_is (char *ptext) * 8 for RF_CNT. * * time_limit - Include only stations heard within this many minutes. - * Typically 30 or 60. + * Typically 180. * * Returns: Number to be used in the statistics report. * @@ -627,7 +695,7 @@ int mheard_count (int max_hops, int time_limit) * callsign - Callsign for station. * * time_limit - Include only stations heard within this many minutes. - * Typically 30 or 60. + * Typically 180. * * max_hops - Include only stations heard with this number of * digipeater hops or less. For reporting, we might use: @@ -768,7 +836,7 @@ void mheard_set_msp (char *callsign, int num) * * Inputs: callsign - Callsign for station which sent the "message." * - * Returns: The cound for the specified station. + * Returns: The count for the specified station. * 0 if not found. * *------------------------------------------------------------------*/ diff --git a/mheard.h b/src/mheard.h similarity index 100% rename from mheard.h rename to src/mheard.h diff --git a/morse.c b/src/morse.c similarity index 95% rename from morse.c rename to src/morse.c index df0038d8..b61e75cb 100644 --- a/morse.c +++ b/src/morse.c @@ -70,7 +70,7 @@ static const struct morse_s { { 'F', "..-." }, { 'G', "--." }, { 'H', "...." }, - { 'I', "." }, + { 'I', ".." }, { 'J', ".---" }, { 'K', "-.-" }, { 'L', ".-.." }, @@ -312,7 +312,12 @@ static void morse_tone (int chan, int tu, int wpm) { int f1_change_per_sample; // How much to advance phase for each audio sample. - assert (save_audio_config_p->achan[chan].valid); + + if (save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid channel %d for sending Morse Code.\n", chan); + return; + } tone_phase = 0; @@ -360,7 +365,11 @@ static void morse_quiet (int chan, int tu, int wpm) { int nsamples; int j; - assert (save_audio_config_p->achan[chan].valid); + if (save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid channel %d for sending Morse Code.\n", chan); + return; + } nsamples = (int) ((TIME_UNITS_TO_MS(tu,wpm) * (float)save_audio_config_p->adev[a].samples_per_sec / 1000.) + 0.5); @@ -395,7 +404,11 @@ static void morse_quiet_ms (int chan, int ms) { int nsamples; int j; - assert (save_audio_config_p->achan[chan].valid); + if (save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Invalid channel %d for sending Morse Code.\n", chan); + return; + } nsamples = (int) ((ms * (float)save_audio_config_p->adev[a].samples_per_sec / 1000.) + 0.5); diff --git a/morse.h b/src/morse.h similarity index 100% rename from morse.h rename to src/morse.h diff --git a/multi_modem.c b/src/multi_modem.c similarity index 70% rename from multi_modem.c rename to src/multi_modem.c index fd53b48a..d2382f1a 100644 --- a/multi_modem.c +++ b/src/multi_modem.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2013, 2014, 2015, 2016, 2019 John Langner, WB2OSZ // // 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 @@ -69,10 +69,21 @@ * different fixup attempts. * Set limit on number of packets in fix up later queue. * + * New in version 1.6: + * + * FX.25. Previously a delay of a couple bits (or more accurately + * symbols) was fine because the decoders took about the same amount of time. + * Now, we can have an additional delay of up to 64 check bytes and + * some filler in the data portion. We can't simply wait that long. + * With normal AX.25 a couple frames can come and go during that time. + * We want to delay the duplicate removal while FX.25 block reception + * is going on. + * *------------------------------------------------------------------*/ //#define DEBUG 1 -#define DIGIPEATER_C + +#define DIGIPEATER_C // Why? #include "direwolf.h" @@ -80,7 +91,7 @@ #include #include #include -#include +#include #include "ax25_pad.h" #include "textcolor.h" @@ -89,6 +100,10 @@ #include "hdlc_rec.h" #include "hdlc_rec2.h" #include "dlq.h" +#include "fx25.h" +#include "version.h" +#include "ais.h" + // Properties of the radio channels. @@ -101,7 +116,13 @@ static struct audio_s *save_audio_config_p; static struct { packet_t packet_p; alevel_t alevel; - retry_t retries; + float speed_error; + fec_type_t fec_type; // Type of FEC: none(0), fx25, il2p + retry_t retries; // For the old "fix bits" strategy, this is the + // number of bits that were modified to get a good CRC. + // It would be 0 to something around 4. + // For FX.25, it is the number of corrected. + // This could be from 0 thru 32. int age; unsigned int crc; int score; @@ -152,7 +173,7 @@ void multi_modem_init (struct audio_s *pa) hdlc_rec_init (save_audio_config_p); for (chan=0; chanachan[chan].valid) { + if (save_audio_config_p->chan_medium[chan] == MEDIUM_RADIO) { if (save_audio_config_p->achan[chan].baud <= 0) { text_color_set(DW_COLOR_ERROR); dw_printf("Internal error, chan=%d, %s, %d\n", chan, __FILE__, __LINE__); @@ -169,97 +190,6 @@ void multi_modem_init (struct audio_s *pa) } -#if 0 - -//Add a crc to the end of the queue and returns the numbers of CRC stored in the queue -int crc_queue_append (unsigned int crc, unsigned int chan) { - crc_t plast; -// crc_t plast1; - crc_t pnext; - crc_t new_crc; - - unsigned int nb_crc = 1; - if (chan>=MAX_CHANS) { - return -1; - } - new_crc = (crc_t) malloc (10*sizeof(struct crc_s)); - if (!new_crc) - return -1; - new_crc->crc = crc; - new_crc->nextp = NULL; - if (crc_queue_of_last_to_app[chan] == NULL) { - crc_queue_of_last_to_app[chan] = new_crc; - nb_crc = 1; - } - else { - nb_crc = 2; - plast = crc_queue_of_last_to_app[chan]; - pnext = plast->nextp; - while (pnext != NULL) { - nb_crc++; - plast = pnext; - pnext = pnext->nextp; - } - plast->nextp = new_crc; - } -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf("Out crc_queue_append nb_crc = %d\n", nb_crc); -#endif - return nb_crc; - - -} - -//Remove the crc from the top of the queue -unsigned int crc_queue_remove (unsigned int chan) { - - unsigned int res; -// crc_t plast; -// crc_t pnext; -#if DEBUG - text_color_set(DW_COLOR_DEBUG); - dw_printf("In crc_queue_remove\n"); -#endif - crc_t removed_crc; - if (chan>=MAX_CHANS) { - return 0; - } - removed_crc = crc_queue_of_last_to_app[chan]; - if (removed_crc == NULL) { - return 0; - } - else { - - crc_queue_of_last_to_app[chan] = removed_crc->nextp; - res = removed_crc->crc; - free(removed_crc); - - } - return res; - -} - -unsigned char is_crc_in_queue(unsigned int chan, unsigned int crc) { - crc_t plast; - crc_t pnext; - - if (crc_queue_of_last_to_app[chan] == NULL) { - return 0; - } - else { - plast = crc_queue_of_last_to_app[chan]; - do { - pnext = plast->nextp; - if (plast->crc == crc) { - return 1; - } - plast = pnext; - } while (pnext != NULL); - } - return 0; -} -#endif /* if 0 */ /*------------------------------------------------------------------------------ * @@ -292,38 +222,50 @@ unsigned char is_crc_in_queue(unsigned int chan, unsigned int crc) { * *------------------------------------------------------------------------------*/ +static float dc_average[MAX_CHANS]; + +int multi_modem_get_dc_average (int chan) +{ + // Scale to +- 200 so it will like the deviation measurement. + + return ( (int) ((float)(dc_average[chan]) * (200.0f / 32767.0f) ) ); +} __attribute__((hot)) void multi_modem_process_sample (int chan, int audio_sample) { int d; int subchan; - static int i = 0; /* for interleaving among multiple demodulators. */ -// TODO: temp debug, remove this. +// Accumulate an average DC bias level. +// Shouldn't happen with a soundcard but could with mistuned SDR. - assert (save_audio_config_p->achan[chan].num_subchan > 0 && save_audio_config_p->achan[chan].num_subchan <= MAX_SUBCHANS); - assert (save_audio_config_p->achan[chan].num_slicers > 0 && save_audio_config_p->achan[chan].num_slicers <= MAX_SLICERS); + dc_average[chan] = dc_average[chan] * 0.999f + (float)audio_sample * 0.001f; - /* Formerly one loop. */ - /* 1.2: We can feed one demodulator but end up with multiple outputs. */ - +// Issue 128. Someone ran into this. - if (save_audio_config_p->achan[chan].interleave > 1) { + //assert (save_audio_config_p->achan[chan].num_subchan > 0 && save_audio_config_p->achan[chan].num_subchan <= MAX_SUBCHANS); + //assert (save_audio_config_p->achan[chan].num_slicers > 0 && save_audio_config_p->achan[chan].num_slicers <= MAX_SLICERS); -// TODO: temp debug, remove this. + if (save_audio_config_p->achan[chan].num_subchan <= 0 || save_audio_config_p->achan[chan].num_subchan > MAX_SUBCHANS || + save_audio_config_p->achan[chan].num_slicers <= 0 || save_audio_config_p->achan[chan].num_slicers > MAX_SLICERS) { - assert (save_audio_config_p->achan[chan].interleave == save_audio_config_p->achan[chan].num_subchan); - demod_process_sample(chan, i, audio_sample); - i++; - if (i >= save_audio_config_p->achan[chan].interleave) i = 0; + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR! Something is seriously wrong in %s %s.\n", __FILE__, __func__); + dw_printf ("chan = %d, num_subchan = %d [max %d], num_slicers = %d [max %d]\n", chan, + save_audio_config_p->achan[chan].num_subchan, MAX_SUBCHANS, + save_audio_config_p->achan[chan].num_slicers, MAX_SLICERS); + dw_printf ("Please report this message and include a copy of your configuration file.\n"); + exit (EXIT_FAILURE); } - else { - /* Send same thing to all. */ - for (d = 0; d < save_audio_config_p->achan[chan].num_subchan; d++) { - demod_process_sample(chan, d, audio_sample); - } + + /* Formerly one loop. */ + /* 1.2: We can feed one demodulator but end up with multiple outputs. */ + + /* Send same thing to all. */ + for (d = 0; d < save_audio_config_p->achan[chan].num_subchan; d++) { + demod_process_sample(chan, d, audio_sample); } for (subchan = 0; subchan < save_audio_config_p->achan[chan].num_subchan; subchan++) { @@ -334,7 +276,12 @@ void multi_modem_process_sample (int chan, int audio_sample) if (candidate[chan][subchan][slice].packet_p != NULL) { candidate[chan][subchan][slice].age++; if (candidate[chan][subchan][slice].age > process_age[chan]) { - pick_best_candidate (chan); + if (fx25_rec_busy(chan)) { + candidate[chan][subchan][slice].age = 0; + } + else { + pick_best_candidate (chan); + } } } } @@ -359,94 +306,15 @@ void multi_modem_process_sample (int chan, int audio_sample) * (Special case, use negative to skip * display of audio level line. * Use -2 to indicate DTMF message.) - * retries - Level of bit correction used. - * + * retries - Level of correction used. + * fec_type - none(0), fx25, il2p * * Description: Add to list of candidates. Best one will be picked later. * *--------------------------------------------------------------------*/ -/* - - It gets a little more complicated when we try fixing frames - with imperfect CRCs. - - Changing of adjacent bits is quick and done immediately. These - all come in at nearly the same time. The processing of two - separated bits can take a long time and is handled in the - background by another thread. These could come in seconds later. - - We need a way to remove duplicates. I think these are the - two cases we need to consider. - - (1) Same result as earlier no error or adjacent bit errors. - - ____||||_ - 0.0: ptr=00000000 - 0.1: ptr=00000000 - 0.2: ptr=00000000 - 0.3: ptr=00000000 - 0.4: ptr=009E5540, retry=0, age=295, crc=9458, score=5024 - 0.5: ptr=0082F008, retry=0, age=294, crc=9458, score=5026 *** - 0.6: ptr=009CE560, retry=0, age=293, crc=9458, score=5026 - 0.7: ptr=009CEE08, retry=0, age=293, crc=9458, score=5024 - 0.8: ptr=00000000 - - ___._____ - 0.0: ptr=00000000 - 0.1: ptr=00000000 - 0.2: ptr=00000000 - 0.3: ptr=009E5540, retry=4, age=295, crc=9458, score=1000 *** - 0.4: ptr=00000000 - 0.5: ptr=00000000 - 0.6: ptr=00000000 - 0.7: ptr=00000000 - 0.8: ptr=00000000 - - (2) Only results from adjusting two non-adjacent bits. - - - ||||||||_ - 0.0: ptr=022EBA08, retry=0, age=289, crc=5acd, score=5042 - 0.1: ptr=022EA8B8, retry=0, age=290, crc=5acd, score=5048 - 0.2: ptr=022EB160, retry=0, age=290, crc=5acd, score=5052 - 0.3: ptr=05BD0048, retry=0, age=291, crc=5acd, score=5054 *** - 0.4: ptr=04FE0048, retry=0, age=292, crc=5acd, score=5054 - 0.5: ptr=05E10048, retry=0, age=294, crc=5acd, score=5052 - 0.6: ptr=053D0048, retry=0, age=294, crc=5acd, score=5048 - 0.7: ptr=02375558, retry=0, age=295, crc=5acd, score=5042 - 0.8: ptr=00000000 - - _______._ - 0.0: ptr=00000000 - 0.1: ptr=00000000 - 0.2: ptr=00000000 - 0.3: ptr=00000000 - 0.4: ptr=00000000 - 0.5: ptr=00000000 - 0.6: ptr=00000000 - 0.7: ptr=02375558, retry=4, age=295, crc=5fc5, score=1000 *** - 0.8: ptr=00000000 - - ________. - 0.0: ptr=00000000 - 0.1: ptr=00000000 - 0.2: ptr=00000000 - 0.3: ptr=00000000 - 0.4: ptr=00000000 - 0.5: ptr=00000000 - 0.6: ptr=00000000 - 0.7: ptr=00000000 - 0.8: ptr=02375558, retry=4, age=295, crc=5fc5, score=1000 *** - - - These can both be covered by keepin the last CRC and dropping - duplicates. In theory we could get another frame in between with - a slow computer so the complete solution would be to remember more - than one. -*/ - -void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned char *fbuf, int flen, alevel_t alevel, retry_t retries) + +void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned char *fbuf, int flen, alevel_t alevel, retry_t retries, fec_type_t fec_type) { packet_t pp; @@ -455,20 +323,50 @@ void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned c assert (subchan >= 0 && subchan < MAX_SUBCHANS); assert (slice >= 0 && slice < MAX_SUBCHANS); - pp = ax25_from_frame (fbuf, flen, alevel); +// Special encapsulation for AIS & EAS so they can be treated normally pretty much everywhere else. + + if (save_audio_config_p->achan[chan].modem_type == MODEM_AIS) { + char nmea[256]; + ais_to_nmea (fbuf, flen, nmea, sizeof(nmea)); + + char monfmt[276]; + snprintf (monfmt, sizeof(monfmt), "AIS>%s%1d%1d:{%c%c%s", APP_TOCALL, MAJOR_VERSION, MINOR_VERSION, USER_DEF_USER_ID, USER_DEF_TYPE_AIS, nmea); + pp = ax25_from_text (monfmt, 1); + + // alevel gets in there somehow making me question why it is passed thru here. + } + else if (save_audio_config_p->achan[chan].modem_type == MODEM_EAS) { + char monfmt[300]; // EAS SAME message max length is 268 + + snprintf (monfmt, sizeof(monfmt), "EAS>%s%1d%1d:{%c%c%s", APP_TOCALL, MAJOR_VERSION, MINOR_VERSION, USER_DEF_USER_ID, USER_DEF_TYPE_EAS, fbuf); + pp = ax25_from_text (monfmt, 1); + + // alevel gets in there somehow making me question why it is passed thru here. + } + else { + pp = ax25_from_frame (fbuf, flen, alevel); + } + multi_modem_process_rec_packet (chan, subchan, slice, pp, alevel, retries, fec_type); +} + +// TODO: Eliminate function above and move code elsewhere? + +void multi_modem_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, fec_type_t fec_type) +{ if (pp == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Unexpected internal problem, %s %d\n", __FILE__, __LINE__); return; /* oops! why would it fail? */ } - /* - * If only one demodulator/slicer, push it thru and forget about all this foolishness. + * If only one demodulator/slicer, and no FX.25 in progress, + * push it thru and forget about all this foolishness. */ if (save_audio_config_p->achan[chan].num_subchan == 1 && - save_audio_config_p->achan[chan].num_slicers == 1) { + save_audio_config_p->achan[chan].num_slicers == 1 && + ! fx25_rec_busy(chan)) { int drop_it = 0; @@ -489,7 +387,7 @@ void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned c ax25_delete (pp); } else { - dlq_rec_frame (chan, subchan, slice, pp, alevel, retries, ""); + dlq_rec_frame (chan, subchan, slice, pp, alevel, fec_type, retries, ""); } return; } @@ -499,13 +397,17 @@ void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned c * Otherwise, save them up for a few bit times so we can pick the best. */ if (candidate[chan][subchan][slice].packet_p != NULL) { - /* Oops! Didn't expect it to be there. */ + /* Plain old AX.25: Oops! Didn't expect it to be there. */ + /* FX.25: Quietly replace anything already there. It will have priority. */ ax25_delete (candidate[chan][subchan][slice].packet_p); candidate[chan][subchan][slice].packet_p = NULL; } + assert (pp != NULL); + candidate[chan][subchan][slice].packet_p = pp; candidate[chan][subchan][slice].alevel = alevel; + candidate[chan][subchan][slice].fec_type = fec_type; candidate[chan][subchan][slice].retries = retries; candidate[chan][subchan][slice].age = 0; candidate[chan][subchan][slice].crc = ax25_m_m_crc(pp); @@ -542,6 +444,9 @@ static void pick_best_candidate (int chan) int best_n, best_score; char spectrum[MAX_SUBCHANS*MAX_SLICERS+1]; int n, j, k; + if (save_audio_config_p->achan[chan].num_slicers < 1) { + save_audio_config_p->achan[chan].num_slicers = 1; + } int num_bars = save_audio_config_p->achan[chan].num_slicers * save_audio_config_p->achan[chan].num_subchan; memset (spectrum, 0, sizeof(spectrum)); @@ -555,6 +460,15 @@ static void pick_best_candidate (int chan) if (candidate[chan][j][k].packet_p == NULL) { spectrum[n] = '_'; } + else if (candidate[chan][j][k].fec_type != fec_type_none) { // FX.25 or IL2P + // FIXME: using retries both as an enum and later int too. + if ((int)(candidate[chan][j][k].retries) <= 9) { + spectrum[n] = '0' + candidate[chan][j][k].retries; + } + else { + spectrum[n] = '+'; + } + } // AX.25 below else if (candidate[chan][j][k].retries == RETRY_NONE) { spectrum[n] = '|'; } @@ -565,21 +479,29 @@ static void pick_best_candidate (int chan) spectrum[n] = '.'; } - /* Begining score depends on effort to get a valid frame CRC. */ + /* Beginning score depends on effort to get a valid frame CRC. */ if (candidate[chan][j][k].packet_p == NULL) { candidate[chan][j][k].score = 0; } else { - /* Originally, this produced 0 for the PASSALL case. */ - /* This didn't work so well when looking for the best score. */ - /* Around 1.3 dev H, we add an extra 1 in here so the minimum */ - /* score should now be 1 for anything received. */ + if (candidate[chan][j][k].fec_type != fec_type_none) { + candidate[chan][j][k].score = 9000 - 100 * candidate[chan][j][k].retries; // has FEC + } + else { + /* Originally, this produced 0 for the PASSALL case. */ + /* This didn't work so well when looking for the best score. */ + /* Around 1.3 dev H, we add an extra 1 in here so the minimum */ + /* score should now be 1 for anything received. */ - candidate[chan][j][k].score = RETRY_MAX * 1000 - ((int)candidate[chan][j][k].retries * 1000) + 1; + candidate[chan][j][k].score = RETRY_MAX * 1000 - ((int)candidate[chan][j][k].retries * 1000) + 1; + } } } + // FIXME: IL2p & FX.25 don't have CRC calculated. Must fill it in first. + + /* Bump it up slightly if others nearby have the same CRC. */ for (n = 0; n < num_bars; n++) { @@ -632,8 +554,9 @@ static void pick_best_candidate (int chan) candidate[chan][j][k].packet_p); } else { - dw_printf ("%d.%d.%d: ptr=%p, retry=%d, age=%3d, crc=%04x, score=%d %s\n", chan, j, k, + dw_printf ("%d.%d.%d: ptr=%p, fec_type=%d, retry=%d, age=%3d, crc=%04x, score=%d %s\n", chan, j, k, candidate[chan][j][k].packet_p, + (int)(candidate[chan][j][k].fec_type), (int)(candidate[chan][j][k].retries), candidate[chan][j][k].age, candidate[chan][j][k].crc, @@ -688,9 +611,11 @@ static void pick_best_candidate (int chan) candidate[chan][j][k].packet_p = NULL; } else { + assert (candidate[chan][j][k].packet_p != NULL); dlq_rec_frame (chan, j, k, candidate[chan][j][k].packet_p, candidate[chan][j][k].alevel, + candidate[chan][j][k].fec_type, (int)(candidate[chan][j][k].retries), spectrum); diff --git a/multi_modem.h b/src/multi_modem.h similarity index 62% rename from multi_modem.h rename to src/multi_modem.h index 77c5b6d5..51c3cde5 100644 --- a/multi_modem.h +++ b/src/multi_modem.h @@ -14,6 +14,11 @@ void multi_modem_init (struct audio_s *pmodem); void multi_modem_process_sample (int c, int audio_sample); -void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned char *fbuf, int flen, alevel_t alevel, retry_t retries); +int multi_modem_get_dc_average (int chan); + +// Deprecated. Replace with ...packet +void multi_modem_process_rec_frame (int chan, int subchan, int slice, unsigned char *fbuf, int flen, alevel_t alevel, retry_t retries, fec_type_t fec_type); + +void multi_modem_process_rec_packet (int chan, int subchan, int slice, packet_t pp, alevel_t alevel, retry_t retries, fec_type_t fec_type); #endif diff --git a/pfilter.c b/src/pfilter.c similarity index 71% rename from pfilter.c rename to src/pfilter.c index 53604101..35767a67 100644 --- a/pfilter.c +++ b/src/pfilter.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2015, 2016, 2023 John Langner, WB2OSZ // // 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 @@ -116,7 +116,7 @@ typedef struct pfstate_s { /* * Are we processing APRS or connected mode? - * This determines whch types of filters are available. + * This determines which types of filters are available. */ int is_aprs; @@ -220,7 +220,7 @@ int pfilter (int from_chan, int to_chan, char *filter, packet_t pp, int is_aprs) pfstate.from_chan = from_chan; pfstate.to_chan = to_chan; - /* Copy filter string, changing any control characers to spaces. */ + /* Copy filter string, changing any control characters to spaces. */ strlcpy (pfstate.filter_str, filter, sizeof(pfstate.filter_str)); @@ -235,7 +235,7 @@ int pfilter (int from_chan, int to_chan, char *filter, packet_t pp, int is_aprs) pfstate.is_aprs = is_aprs; if (is_aprs) { - decode_aprs (&pfstate.decoded, pp, 1); + decode_aprs (&pfstate.decoded, pp, 1, NULL); } next_token(&pfstate); @@ -546,7 +546,8 @@ static int parse_filter_spec (pfstate_t *pf) /* b - budlist */ else if (pf->token_str[0] == 'b' && ispunct(pf->token_str[1])) { - /* Budlist - source address */ + /* Budlist - AX.25 source address */ + /* Could be different than source encapsulated by 3rd party header. */ char addr[AX25_MAX_ADDR_LEN]; ax25_get_addr_with_ssid (pf->pp, AX25_SOURCE, addr); result = filt_bodgu (pf, addr); @@ -572,7 +573,7 @@ static int parse_filter_spec (pfstate_t *pf) else if (pf->token_str[0] == 'd' && ispunct(pf->token_str[1])) { int n; - // loop on all digipeaters + // Loop on all AX.25 digipeaters. result = 0; for (n = AX25_REPEATER_1; result == 0 && n < ax25_get_num_addr (pf->pp); n++) { // Consider only those with the H (has-been-used) bit set. @@ -599,7 +600,7 @@ static int parse_filter_spec (pfstate_t *pf) else if (pf->token_str[0] == 'v' && ispunct(pf->token_str[1])) { int n; - // loop on all digipeaters (mnemonic Via) + // loop on all AX.25 digipeaters (mnemonic Via) result = 0; for (n = AX25_REPEATER_1; result == 0 && n < ax25_get_num_addr (pf->pp); n++) { // This is different than the previous "d" filter. @@ -623,10 +624,15 @@ static int parse_filter_spec (pfstate_t *pf) } } -/* g - Addressee of message. */ +/* g - Addressee of message. e.g. "BLN*" for bulletins. */ else if (pf->token_str[0] == 'g' && ispunct(pf->token_str[1])) { - if (ax25_get_dti(pf->pp) == ':') { + if (pf->decoded.g_message_subtype == message_subtype_message || + pf->decoded.g_message_subtype == message_subtype_ack || + pf->decoded.g_message_subtype == message_subtype_rej || + pf->decoded.g_message_subtype == message_subtype_bulletin || + pf->decoded.g_message_subtype == message_subtype_nws || + pf->decoded.g_message_subtype == message_subtype_directed_query) { result = filt_bodgu (pf, pf->decoded.g_addressee); if (s_debug >= 2) { @@ -643,7 +649,7 @@ static int parse_filter_spec (pfstate_t *pf) } } -/* u - unproto (destination) */ +/* u - unproto (AX.25 destination) */ else if (pf->token_str[0] == 'u' && ispunct(pf->token_str[1])) { /* Probably want to exclude mic-e types */ @@ -668,7 +674,7 @@ static int parse_filter_spec (pfstate_t *pf) } } -/* t - type: position, weather, etc. */ +/* t - packet type: position, weather, telemetry, etc. */ else if (pf->token_str[0] == 't' && ispunct(pf->token_str[1])) { @@ -728,7 +734,7 @@ static int parse_filter_spec (pfstate_t *pf) (void) ax25_get_info (pf->pp, (unsigned char **)(&infop)); text_color_set(DW_COLOR_DEBUG); - if (*infop == ':' && ! is_telem_metadata(infop)) { + if (pf->decoded.g_packet_type == packet_type_message) { dw_printf (" %s returns %s for message to %s\n", pf->token_str, bool2text(result), pf->decoded.g_addressee); } else { @@ -776,7 +782,7 @@ static int parse_filter_spec (pfstate_t *pf) * -1 = error detected * * Description: Same function is used for all of these because they are so similar. - * Look for exact match to any of the specifed strings. + * Look for exact match to any of the specified strings. * All of them allow wildcarding with single * at the end. * *------------------------------------------------------------------------------*/ @@ -832,31 +838,17 @@ static int filt_bodgu (pfstate_t *pf, char *arg) * 0 = no * -1 = error detected * - * Description: The filter is based the type filtering described here: + * Description: The filter is loosely based the type filtering described here: * http://www.aprs-is.net/javAPRSFilter.aspx * - * Most of these simply check the first byte of the information part. - * Trying to detect NWS information is a little trickier. + * Mostly use g_packet_type and g_message_subtype from decode_aprs. + * + * References: * http://www.aprs-is.net/WX/ - * http://wxsvr.aprs.net.au/protocol-new.html + * http://wxsvr.aprs.net.au/protocol-new.html (has disappeared) * *------------------------------------------------------------------------------*/ -/* Telemetry metadata is a special case of message. */ -/* We want to categorize it as telemetry rather than message. */ - -int is_telem_metadata (char *infop) -{ - if (*infop != ':') return (0); - if (strlen(infop) < 16) return (0); - if (strncmp(infop+10, ":PARM.", 6) == 0) return (1); - if (strncmp(infop+10, ":UNIT.", 6) == 0) return (1); - if (strncmp(infop+10, ":EQNS.", 6) == 0) return (1); - if (strncmp(infop+10, ":BITS.", 6) == 0) return (1); - return (0); -} - - static int filt_t (pfstate_t *pf) { char src[AX25_MAX_ADDR_LEN]; @@ -873,83 +865,60 @@ static int filt_t (pfstate_t *pf) switch (*f) { case 'p': /* Position */ - if (*infop == '!') return (1); - if (*infop == '\'') return (1); - if (*infop == '/') return (1); - if (*infop == '=') return (1); - if (*infop == '@') return (1); - if (*infop == '`') return (1); + if (pf->decoded.g_packet_type == packet_type_position) return(1); break; case 'o': /* Object */ - if (*infop == ';') return (1); + if (pf->decoded.g_packet_type == packet_type_object) return(1); break; case 'i': /* Item */ - if (*infop == ')') return (1); + if (pf->decoded.g_packet_type == packet_type_item) return(1); break; - case 'm': /* Message */ - if (*infop == ':' && ! is_telem_metadata(infop)) return (1); + case 'm': // Any "message." + if (pf->decoded.g_packet_type == packet_type_message) return(1); break; case 'q': /* Query */ - if (*infop == '?') return (1); + if (pf->decoded.g_packet_type == packet_type_query) return(1); + break; + + case 'c': /* station Capabilities - my extension */ + /* Most often used for IGate statistics. */ + if (pf->decoded.g_packet_type == packet_type_capabilities) return(1); break; case 's': /* Status */ - if (*infop == '>') return (1); + if (pf->decoded.g_packet_type == packet_type_status) return(1); break; - case 't': /* Telemetry */ - if (*infop == 'T') return (1); - if (is_telem_metadata(infop)) return (1); + case 't': /* Telemetry data or metadata */ + if (pf->decoded.g_packet_type == packet_type_telemetry) return(1); break; case 'u': /* User-defined */ - if (*infop == '{') return (1); + if (pf->decoded.g_packet_type == packet_type_userdefined) return(1); + break; + + case 'h': /* has third party Header - my extension */ + if (pf->decoded.g_has_thirdparty_header) return (1); break; case 'w': /* Weather */ - if (*infop == '@') return (1); - if (*infop == '*') return (1); - if (*infop == '_') return (1); - /* '$' is normally raw GPS. Check for special case. */ - if (strncmp(infop, "$ULTW", 5) == 0) return (1); + if (pf->decoded.g_packet_type == packet_type_weather) return(1); + + /* Positions !=/@ with symbol code _ are weather. */ + /* Object with _ symbol is also weather. APRS protocol spec page 66. */ + // Can't use *infop because it would not work with 3rd party header. - /* TODO: Positions !=/@ can be weather. */ - /* Need to check for _ symbol. */ + if ((pf->decoded.g_packet_type == packet_type_position || + pf->decoded.g_packet_type == packet_type_object) && pf->decoded.g_symbol_code == '_') return (1); break; case 'n': /* NWS format */ -/* - * This is the interesting case. - * The source must be exactly 6 upper case letters, no SSID. - */ - if (strlen(src) != 6) break; - if (! isupper(src[0])) break; - if (! isupper(src[1])) break; - if (! isupper(src[2])) break; - if (! isupper(src[3])) break; - if (! isupper(src[4])) break; - if (! isupper(src[5])) break; -/* - * We can have a "message" with addressee starting with NWS, SKY, or BOM (Australian version.) - */ - if (strncmp(infop, ":NWS", 4) == 0) return (1); - if (strncmp(infop, ":SKY", 4) == 0) return (1); - if (strncmp(infop, ":BOM", 4) == 0) return (1); -/* - * Or we can have an object. - * It's not exactly clear how to distiguish this from other objects. - * It looks like the first 3 characters of the source should be the same - * as the first 3 characters of the addressee. - */ - if (infop[0] == ';' && - infop[1] == src[0] && - infop[2] == src[1] && - infop[3] == src[2]) return (1); + if (pf->decoded.g_packet_type == packet_type_nws) return(1); break; default: @@ -965,6 +934,7 @@ static int filt_t (pfstate_t *pf) + /*------------------------------------------------------------------------------ * * Name: filt_r @@ -1059,15 +1029,51 @@ static int filt_r (pfstate_t *pf, char *sdist) * * Description: * + * s/pri + * s/pri/alt + * s/pri/alt/ + * s/pri/alt/over + * * "pri" is zero or more symbols from the primary symbol set. + * Symbol codes are any printable ASCII character other than | or ~. + * (Zero symbols here would be sensible only if later alt part is specified.) * "alt" is one or more symbols from the alternate symbol set. - * "over" is overlay characters. Overlays apply only to the alternate symbol set. + * "over" is overlay characters for the alternate symbol set. + * Only upper case letters, digits, and \ are allowed here. + * If the last part is not specified, any overlay or lack of overlay, is ignored. + * If the last part is specified, only the listed overlays will match. + * An explicit lack of overlay is represented by the \ character. * * Examples: - * s/-> Allow house and car from primary symbol table. - * s//# Allow alternate table digipeater, with or without overlay. - * s//#/\ Allow alternate table digipeater, only if no overlay. - * s//#/SL1 Allow alternate table digipeater, with overlay S, L, or 1 + * s/O Balloon. + * s/-> House or car from primary symbol table. + * + * s//# Alternate table digipeater, with or without overlay. + * s//#/\ Alternate table digipeater, only if no overlay. + * s//#/SL1 Alternate table digipeater, with overlay S, L, or 1. + * s//#/SL\ Alternate table digipeater, with S, L, or no overlay. + * + * s/s/s Any variation of watercraft. Either symbol table. With or without overlay. + * s/s/s/ Ship or ship sideview, only if no overlay. + * s//s/J Jet Ski. + * + * What if you want to use the / symbol when / is being used as a delimiter here? Recall that you + * can use some other special character after the initial lower case letter and this becomes the + * delimiter for the rest of the specification. + * + * Examples: + * + * s:/ Red Dot. + * s::/ Waypoint Destination, with or without overlay. + * s:/:/ Either Red Dot or Waypoint Destination. + * s:/:/: Either Red Dot or Waypoint Destination, no overlay. + * + * Bad example: + * + * Someone tried using this to include ballons: s/'/O/-/#/_ + * probably following the buddy filter pattern of / between each alternative. + * There should be an error message because it has more than 3 delimiter characters. + * * *------------------------------------------------------------------------------*/ @@ -1075,8 +1081,9 @@ static int filt_s (pfstate_t *pf) { char str[MAX_TOKEN_LEN]; char *cp; - char sep[2]; - char *pri, *alt, *over; + char sep[2]; // Delimiter character. Typically / but it could be different. + char *pri = NULL, *alt = NULL, *over = NULL, *extra = NULL; + char *x; strlcpy (str, pf->token_str, sizeof(str)); @@ -1084,58 +1091,140 @@ static int filt_s (pfstate_t *pf) sep[1] = '\0'; cp = str + 2; -// TODO: check here. + +// First, separate the parts and do a strict syntax check. pri = strsep (&cp, sep); - if (pri == NULL) { + if (pri != NULL) { + + // Zero length is acceptable if alternate symbol(s) specified. Will check that later. + + for (x = pri; *x != '\0'; x++) { + if ( ! isprint(*x) || *x == '|' || *x == '~') { + print_error (pf, "Symbol filter, primary must be printable ASCII character(s) other than | or ~."); + return (-1); + } + } + + alt = strsep (&cp, sep); + + if (alt != NULL) { + + // Zero length after second / would be pointless. + + if (strlen(alt) == 0) { + print_error (pf, "Nothing specified for alternate symbol table."); + return (-1); + } + + for (x = alt; *x != '\0'; x++) { + if ( ! isprint(*x) || *x == '|' || *x == '~') { + print_error (pf, "Symbol filter, alternate must be printable ASCII character(s) other than | or ~."); + return (-1); + } + } + + over = strsep (&cp, sep); + + if (over != NULL) { + + // Zero length is acceptable and is not the same as missing. + + for (x = over; *x != '\0'; x++) { + if ( (! isupper(*x)) && (! isdigit(*x)) && *x != '\\') { + print_error (pf, "Symbol filter, overlay must be upper case letter, digit, or \\."); + return (-1); + } + } + + extra = strsep (&cp, sep); + + if (extra != NULL) { + print_error (pf, "More than 3 delimiter characters in Symbol filter."); + return (-1); + } + } + } + else { + // No alt part is OK if at least one primary symbol was specified. + if (strlen(pri) == 0) { + print_error (pf, "No symbols specified for Symbol filter."); + return (-1); + } + } + } + else { print_error (pf, "Missing arguments for Symbol filter."); return (-1); } - if (pf->decoded.g_symbol_table == '/' && strchr(pri, pf->decoded.g_symbol_code) != NULL) { - /* Found in primary symbols. All done. */ - return (1); + +// This applies only for Position, Object, Item. +// decode_aprs() should set symbol code to space to mean undefined. + + if (pf->decoded.g_symbol_code == ' ') { + return (0); + } + + +// Look for Primary symbols. + + if (pf->decoded.g_symbol_table == '/') { + if (pri != NULL && strlen(pri) > 0) { + return (strchr(pri, pf->decoded.g_symbol_code) != NULL); + } } - alt = strsep (&cp, sep); if (alt == NULL) { return (0); } - if (strlen(alt) == 0) { - /* We have s/.../ */ - print_error (pf, "Missing alternate symbols for Symbol filter."); - return (-1); - } //printf ("alt=\"%s\" sym='%c'\n", alt, pf->decoded.g_symbol_code); - if (strchr(alt, pf->decoded.g_symbol_code) == NULL) { - /* Not found in alternate symbols. Reject. */ - return (0); - } +// Look for Alternate symbols. - over = strsep (&cp, sep); - if (over == NULL) { - /* alternate, with or without overlay. */ - return (pf->decoded.g_symbol_table != '/'); - } + if (strchr(alt, pf->decoded.g_symbol_code) != NULL) { + + // We have a match but that might not be enough. + // We must see if there was an overlay part specified. + + if (over != NULL) { - // printf ("over=\"%s\" table='%c'\n", over, pf->decoded.g_symbol_table); + if (strlen(over) > 0) { - if (strlen(over) == 0) { - return (pf->decoded.g_symbol_table == '\\'); + // Non-zero length overlay part was specified. + // Need to match one of them. + + return (strchr(over, pf->decoded.g_symbol_table) != NULL); + } + else { + + // Zero length overlay part was specified. + // We must have no overlay, i.e. table is \. + + return (pf->decoded.g_symbol_table == '\\'); + } + } + else { + + // No check of overlay part. Just make sure it is not primary table. + + return (pf->decoded.g_symbol_table != '/'); + } } - return (strchr(over, pf->decoded.g_symbol_table) != NULL); -} + return (0); + +} /* end filt_s */ /*------------------------------------------------------------------------------ * * Name: filt_i * - * Purpose: IGate messaging default behavior. + * Purpose: IGate messaging filter. + * This would make sense only for IS>RF direction. * * Inputs: pf - Pointer to current state information. * token_str should contain something of format: @@ -1147,7 +1236,7 @@ static int filt_s (pfstate_t *pf) * -1 = error detected * * Description: Selection is based on time since last heard on RF, and distance - * in terms of digipeater hops and/or phyiscal location. + * in terms of digipeater hops and/or physical location. * * i/time * i/time/hops @@ -1155,7 +1244,7 @@ static int filt_s (pfstate_t *pf) * * * "time" is maximum number of minutes since message addressee was last heard. - * This is required. + * This is required. APRS-IS uses 3 hours so that would be a good value here. * * "hops" is maximum number of digpeater hops. (i.e. 0 for heard directly). * If hops is not specified, the maximum transmit digipeater hop count, @@ -1164,20 +1253,61 @@ static int filt_s (pfstate_t *pf) * The rest is distanced, in kilometers, from given point. * * Examples: - * i/60/0 Heard in past 60 minutes directly. + * i/180/0 Heard in past 3 hours directly. * i/45 Past 45 minutes, default max digi hops. - * i/30/3 Default time, max 3 digi hops. - * i/30/8/42.6/-71.3/50. + * i/180/3 Default time (3 hours), max 3 digi hops. + * i/180/8/42.6/-71.3/50. * * * It only makes sense to use this for the IS>RF direction. * The basic idea is that we want to transmit a "message" only if the * addressee has been heard recently and is not too far away. * + * That is so we can distinguish messages addressed to a specific + * station, and other sundry uses of the addressee field. + * * After passing along a "message" we will also allow the next * position report from the sender of the "message." * That is done somewhere else. We are not concerned with it here. * + * IMHO, the rules here are too restrictive. + * + * The APRS-IS would send a "message" to my IGate only if the addressee + * has been heard nearby recently. 180 minutes, I believe. + * Why would I not want to transmit it? + * + * Discussion: In retrospect, I think this is far too complicated. + * In a future release, I think at options other than time should be removed. + * Messages have more value than most packets. Why reduce the chance of successful delivery? + * + * Consider the following scenario: + * + * (1) We hear AA1PR-9 by a path of 4 digipeaters. + * Looking closer, it's probably only two because there are left over WIDE1-0 and WIDE2-0. + * + * Digipeater WIDE2 (probably N3LLO-3) audio level = 72(19/15) [NONE] _|||||___ + * [0.3] AA1PR-9>APY300,K1EQX-7,WIDE1,N3LLO-3,WIDE2*,ARISS::ANSRVR :cq hotg vt aprsthursday{01<0x0d> + * + * (2) APRS-IS sends a response to us. + * + * [ig>tx] ANSRVR>APWW11,KJ4ERJ-15*,TCPIP*,qAS,KJ4ERJ-15::AA1PR-9 :N:HOTG 161 Messages Sent{JL} + * + * (3) Here is our analysis of whether it should be sent to RF. + * + * Was message addressee AA1PR-9 heard in the past 180 minutes, with 2 or fewer digipeater hops? + * No, AA1PR-9 was last heard over the radio with 4 digipeater hops 0 minutes ago. + * + * The wrong hop count caused us to drop a packet that should have been transmitted. + * We could put in a hack to not count the "WIDE*-0" addresses. + * That is not correct because other prefixes could be used and we don't know + * what they are for other digipeaters. + * I think the best solution is to simply ignore the hop count. + * + * Release 1.7: I got overly ambitious and now realize this is just giving people too much + * "rope to hang themselves," drop messages unexpectedly, and accidentally break messaging. + * Change documentation to mention only the time limit. + * The other functionality will be undocumented and maybe disappear over time. + * *------------------------------------------------------------------------------*/ static int filt_i (pfstate_t *pf) @@ -1186,7 +1316,15 @@ static int filt_i (pfstate_t *pf) char *cp; char sep[2]; char *v; - int heardtime = 30; + +// http://lists.tapr.org/pipermail/aprssig_lists.tapr.org/2020-July/048656.html +// Default of 3 hours should be good. +// One might question why to have a time limit at all. Messages are very rare +// the the APRS-IS wouldn't be sending it to me unless the addressee was in the +// vicinity recently. +// TODO: Should produce a warning if a user specified filter does not include "i". + + int heardtime = 180; // 3 hours * 60 min/hr = 180 minutes #if PFTEST int maxhops = 2; #else @@ -1197,9 +1335,9 @@ static int filt_i (pfstate_t *pf) double km = G_UNKNOWN; - char src[AX25_MAX_ADDR_LEN]; - char *infop = NULL; - int info_len; + //char src[AX25_MAX_ADDR_LEN]; + //char *infop = NULL; + //int info_len; //char *f; //char addressee[AX25_MAX_ADDR_LEN]; @@ -1273,20 +1411,7 @@ static int filt_i (pfstate_t *pf) * Get source address and info part. * Addressee has already been extracted into pf->decoded.g_addressee. */ - - memset (src, 0, sizeof(src)); - ax25_get_addr_with_ssid (pf->pp, AX25_SOURCE, src); - info_len = ax25_get_info (pf->pp, (unsigned char **)(&infop)); - - if (infop == NULL) return (0); - if (info_len < 1) return (0); - -// Determine packet type. We are interested only in "message." -// Telemetry metadata is not considered a message. - - if (*infop != ':') return (0); - if (is_telem_metadata(infop)) return (0); - + if (pf->decoded.g_packet_type != packet_type_message) return(0); #if defined(PFTEST) || defined(DIGITEST) // TODO: test functionality too, not just syntax. @@ -1323,10 +1448,10 @@ static int filt_i (pfstate_t *pf) * * Maybe we could compromise here and say the sender must have been heard directly. * It sent the message currently being processed so we must have heard it very recently, i.e. in - * the past minute, rather than the usual 30 or 60 minutes for the addressee. + * the past minute, rather than the usual 180 minutes for the addressee. */ - was_heard = mheard_was_recently_nearby ("source", src, 1, 0, G_UNKNOWN, G_UNKNOWN, G_UNKNOWN); + was_heard = mheard_was_recently_nearby ("source", pf->decoded.g_src, 1, 0, G_UNKNOWN, G_UNKNOWN, G_UNKNOWN); if (was_heard) return (0); @@ -1480,24 +1605,34 @@ int main () pftest (112, "t/t", "WM1X>APU25N:@210147z4235.39N/07106.58W_359/000g000t027r000P000p000h89b10234/WX REPORT {UIV32N}<0x0d>", 0); pftest (113, "t/w", "WM1X>APU25N:@210147z4235.39N/07106.58W_359/000g000t027r000P000p000h89b10234/WX REPORT {UIV32N}<0x0d>", 1); - /* Telemetry metadata is a special case of message. */ + /* Telemetry metadata should not be classified as message. */ pftest (114, "t/t", "KJ4SNT>APMI04::KJ4SNT :PARM.Vin,Rx1h,Dg1h,Eff1h,Rx10m,O1,O2,O3,O4,I1,I2,I3,I4", 1); pftest (115, "t/m", "KJ4SNT>APMI04::KJ4SNT :PARM.Vin,Rx1h,Dg1h,Eff1h,Rx10m,O1,O2,O3,O4,I1,I2,I3,I4", 0); pftest (116, "t/t", "KB1GKN-10>APRX27,UNCAN,WIDE1*:T#491,4.9,0.3,25.0,0.0,1.0,00000000", 1); + /* Bulletins should not be considered to be messages. Was bug in 1.6. */ + pftest (117, "t/m", "A>B::W1AW :test", 1); + pftest (118, "t/m", "A>B::BLN :test", 0); + pftest (119, "t/m", "A>B::NWS :test", 0); - pftest (120, "t/p", "CWAPID>APRS::NWS-TTTTT:DDHHMMz,ADVISETYPE,zcs{seq#", 0); + // https://www.aprs-is.net/WX/ + pftest (121, "t/p", "CWAPID>APRS::NWS-TTTTT:DDHHMMz,ADVISETYPE,zcs{seq#", 0); pftest (122, "t/p", "CWAPID>APRS::SKYCWA :DDHHMMz,ADVISETYPE,zcs{seq#", 0); pftest (123, "t/p", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", 0); pftest (124, "t/n", "CWAPID>APRS::NWS-TTTTT:DDHHMMz,ADVISETYPE,zcs{seq#", 1); pftest (125, "t/n", "CWAPID>APRS::SKYCWA :DDHHMMz,ADVISETYPE,zcs{seq#", 1); - pftest (126, "t/n", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", 1); + //pftest (126, "t/n", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", 1); pftest (127, "t/", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", 0); - pftest (130, "r/42.6/-71.3/10", "WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); - pftest (131, "r/42.6/-71.3/10", "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32", 0); + pftest (128, "t/c", "S0RCE>DEST:DEST:DEST:}WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); + pftest (131, "t/c", "S0RCE>DEST:}WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); + + pftest (140, "r/42.6/-71.3/10", "WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); + pftest (141, "r/42.6/-71.3/10", "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32", 0); - pftest (140, "( t/t & b/WB2OSZ ) | ( t/o & ! r/42.6/-71.3/1 )", "WB2OSZ>APDW12:;home *111111z4237.14N/07120.83W-Chelmsford MA", 1); + pftest (145, "( t/t & b/WB2OSZ ) | ( t/o & ! r/42.6/-71.3/1 )", "WB2OSZ>APDW12:;home *111111z4237.14N/07120.83W-Chelmsford MA", 1); pftest (150, "s/->", "WB2OSZ-5>APDW12:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); pftest (151, "s/->", "WB2OSZ-5>APDW12:!4237.14N/07120.83W-PHG7140Chelmsford MA", 1); @@ -1515,13 +1650,28 @@ int main () pftest (160, "s//#/LS1", "WB2OSZ-5>APDW12:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); pftest (161, "s//#/LS1", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W#PHG7140Chelmsford MA", 0); pftest (162, "s//#/LS1", "WB2OSZ-5>APDW12:!4237.14N/07120.83W#PHG7140Chelmsford MA", 0); + pftest (163, "s//#/LS\\", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W#PHG7140Chelmsford MA", 1); + + pftest (170, "s:/", "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA", 1); + pftest (171, "s:/", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA", 0); + pftest (172, "s::/", "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA", 0); + pftest (173, "s::/", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA", 1); + pftest (174, "s:/:/", "WB2OSZ-5>APDW12:!4237.14N/07120.83W/PHG7140Chelmsford MA", 1); + pftest (175, "s:/:/", "WB2OSZ-5>APDW12:!4237.14N\\07120.83W/PHG7140Chelmsford MA", 1); + pftest (176, "s:/:/", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1); + pftest (177, "s:/:/:X", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1); + + // FIXME: Different on Windows and 64 bit Linux. + //pftest (178, "s:/:/:", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 1); - pftest (170, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); - pftest (171, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1*,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); - pftest (172, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); - pftest (173, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3*,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); - pftest (174, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4*:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); - pftest (175, "v/DIGI9/DIGI2", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); + pftest (179, "s:/:/:\\", "WB2OSZ-5>APDW12:!4237.14NX07120.83W/PHG7140Chelmsford MA", 0); + + pftest (180, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); + pftest (181, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1*,DIGI2,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); + pftest (182, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); + pftest (183, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3*,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); + pftest (184, "v/DIGI2/DIGI3", "WB2OSZ-5>APDW12,DIGI1,DIGI2,DIGI3,DIGI4*:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); + pftest (185, "v/DIGI9/DIGI2", "WB2OSZ-5>APDW12,DIGI1,DIGI2*,DIGI3,DIGI4:!4237.14NS07120.83W#PHG7140Chelmsford MA", 0); /* Test error reporting. */ @@ -1532,25 +1682,65 @@ int main () pftest (203, "t/w t/w", "CWAPID>APRS:;CWAttttz *DDHHMMzLATLONICONADVISETYPE{seq#", -1); pftest (204, "r/42.6/-71.3", "WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32", -1); - pftest (220, "i/30/8/42.6/-71.3/50", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); - pftest (222, "i/30/8/42.6/-71.3/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); - pftest (223, "i/30/8/42.6/-71.3", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); - pftest (224, "i/30/8/42.6/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); - pftest (225, "i/30/8/42.6", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); - pftest (226, "i/30/8/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); - pftest (227, "i/30/8", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); - - // FIXME: behaves differently on Windows and Linux + pftest (210, "i/30/8/42.6/-71.3/50", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); + pftest (212, "i/30/8/42.6/-71.3/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); + pftest (213, "i/30/8/42.6/-71.3", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); + pftest (214, "i/30/8/42.6/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); + pftest (215, "i/30/8/42.6", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); + pftest (216, "i/30/8/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); + pftest (217, "i/30/8", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); + + // FIXME: behaves differently on Windows and Linux. Why? + // On Windows we have our own version of strsep because it's not in the MS library. + // It must behave differently than the Linux version when nothing follows the last separator. //pftest (228, "i/30/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); pftest (229, "i/30", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); - pftest (230, "i/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); - -// TODO: to be continued... + pftest (230, "i/30", "X>X:}WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); + pftest (231, "i/", "WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", -1); + + // Besure bulletins and telemetry metadata don't get included. + pftest (234, "i/30", "KJ4SNT>APMI04::KJ4SNT :PARM.Vin,Rx1h,Dg1h,Eff1h,Rx10m,O1,O2,O3,O4,I1,I2,I3,I4", 0); + pftest (235, "i/30", "A>B::BLN :test", 0); + + pftest (240, "s/", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + pftest (241, "s/'/O/-/#/_", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + pftest (242, "s/O/O/c", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + pftest (243, "s/O/O/1/2", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + pftest (244, "s/O/|/1", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + pftest (245, "s//", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + pftest (246, "s///", "WB2OSZ-5>APDW12:!4237.14N/07120.83WOPHG7140Chelmsford MA", -1); + + // Third party header - done properly in 1.7. + // Packet filter t/h is no longer a mutually exclusive packet type. + // Now it is an independent attribute and the encapsulated part is evaluated. + + pftest (250, "o/home", "A>B:}WB2OSZ>APDW12,WIDE1-1,WIDE2-1:;home *111111z4237.14N/07120.83W-Chelmsford MA", 1); + pftest (251, "t/p", "A>B:}W1WRA-7>TRSY3T,WIDE1-1,WIDE2-1:`c-:l!hK\\>\"4b}=<0x0d>", 1); + pftest (252, "i/180", "A>B:}WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); + pftest (253, "t/m", "A>B:}WB2OSZ-5>APDW14::W2UB :Happy Birthday{001", 1); + pftest (254, "r/42.6/-71.3/10", "A>B:}WB2OSZ-5>APDW12,WIDE1-1,WIDE2-1:!4237.14NS07120.83W#PHG7140Chelmsford MA", 1); + pftest (254, "r/42.6/-71.3/10", "A>B:}WA1PLE-5>APWW10,W1MHL,N8VIM,WIDE2*:@022301h4208.75N/07115.16WoAPRS-IS for Win32", 0); + pftest (255, "t/h", "KB1GKN-10>APRX27,UNCAN,WIDE1*:T#491,4.9,0.3,25.0,0.0,1.0,00000000", 0); + pftest (256, "t/h", "A>B:}KB1GKN-10>APRX27,UNCAN,WIDE1*:T#491,4.9,0.3,25.0,0.0,1.0,00000000", 1); + pftest (258, "t/t", "A>B:}KB1GKN-10>APRX27,UNCAN,WIDE1*:T#491,4.9,0.3,25.0,0.0,1.0,00000000", 1); + pftest (259, "t/t", "A>B:}KJ4SNT>APMI04::KJ4SNT :PARM.Vin,Rx1h,Dg1h,Eff1h,Rx10m,O1,O2,O3,O4,I1,I2,I3,I4", 1); + + pftest (270, "g/BLN*", "WB2OSZ>APDW17::BLN1xxxxx:bulletin text", 1); + pftest (271, "g/BLN*", "A>B:}WB2OSZ>APDW17::BLN1xxxxx:bulletin text", 1); + pftest (272, "g/BLN*", "A>B:}WB2OSZ>APDW17::W1AW :xxxx", 0); + + pftest (273, "g/NWS*", "WB2OSZ>APDW17::NWS-xxxxx:weather bulletin", 1); + pftest (274, "g/NWS*", "A>B:}WB2OSZ>APDW17::NWS-xxxxx:weather bulletin", 1); + pftest (275, "g/NWS*", "A>B:}WB2OSZ>APDW17::W1AW :xxxx", 0); + +// TODO: add b/ with 3rd party header. + +// TODO: to be continued... directed query ... if (error_count > 0) { text_color_set (DW_COLOR_ERROR); - dw_printf ("\nPacket Filtering Test - FAILED!\n"); + dw_printf ("\nPacket Filtering Test - FAILED! %d errors\n", error_count); exit (EXIT_FAILURE); } text_color_set (DW_COLOR_REC); diff --git a/pfilter.h b/src/pfilter.h similarity index 100% rename from pfilter.h rename to src/pfilter.h diff --git a/ptt.c b/src/ptt.c similarity index 81% rename from ptt.c rename to src/ptt.c index da5429a9..5187f1df 100644 --- a/ptt.c +++ b/src/ptt.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2013, 2014, 2015, 2016, 2017 John Langner, WB2OSZ +// Copyright (C) 2011, 2013, 2014, 2015, 2016, 2017, 2023 John Langner, WB2OSZ // // 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 @@ -52,6 +52,9 @@ * * Handle more complicated gpio node names for CubieBoard, etc. * + * Version 1.5: Ability to use GPIO pins of CM108/CM119 for PTT signal. + * + * * References: http://www.robbayer.com/files/serial-win.pdf * * https://www.kernel.org/doc/Documentation/gpio.txt @@ -59,18 +62,31 @@ *---------------------------------------------------------------*/ /* - Idea for future enhancement: + A growing number of people have been asking about support for the DMK URI, + RB-USB RIM, etc. - A growing number of people have been asking about support for the DMK URI. - This uses a C-Media CM108/CM119 with one interesting addition, a GPIO + These use a C-Media CM108/CM119 with an interesting addition, a GPIO pin is used to drive PTT. Here is some related information. DMK URI: http://www.dmkeng.com/URI_Order_Page.htm http://dmkeng.com/images/URI%20Schematic.pdf + + RB-USB RIM: + + http://www.repeater-builder.com/products/usb-rim-lite.html http://www.repeater-builder.com/voip/pdf/cm119-datasheet.pdf + RA-35: + + http://www.masterscommunications.com/products/radio-adapter/ra35.html + + DINAH: + + https://hamprojects.info/dinah/ + + Homebrew versions of the same idea: http://images.ohnosec.org/usbfob.pdf @@ -83,6 +99,7 @@ http://docs.allstarlink.org/drupal/ http://soundmodem.sourcearchive.com/documentation/0.16-1/ptt_8c_source.html https://github.com/N0NB/hamlib/blob/master/src/cm108.c#L190 + http://permalink.gmane.org/gmane.linux.hams.hamlib.devel/3420 Information about the "hidraw" device: @@ -92,12 +109,31 @@ https://github.com/signal11/hidapi/blob/master/libusb/hid.c http://stackoverflow.com/questions/899008/howto-write-to-the-gpio-pin-of-the-cm108-chip-in-linux https://www.kernel.org/doc/Documentation/hid/hidraw.txt + https://github.com/torvalds/linux/blob/master/samples/hidraw/hid-example.c + + Similar chips: SSS1621, SSS1623 + + https://irongarment.wordpress.com/2011/03/29/cm108-compatible-chips-with-gpio/ - In version 1.3, we add HAMLIB support which should be able to do this. - (Linux only & haven't verified that it actually works yet!) + Here is an attempt to add direct CM108 support. + Seems to be hardcoded for only a single USB audio adapter. + + https://github.com/donothingloop/direwolf_cm108 + + In version 1.3, we add HAMLIB support which should be able to do this in a roundabout way. + (Linux only at this point.) + + This is documented in the User Guide, section called, + "Hamlib PTT Example 2: Use GPIO of USB audio adapter. (e.g. DMK URI)" + + It's rather involved and the explanation doesn't cover the case of multiple + USB-Audio adapters. It would be nice to have a little script which lists all + of the USB-Audio adapters and the corresponding /dev/hidraw device. + ( We now have it. The included "cm108" application. ) - Might want to have CM108 GPIO support built in, someday, for simpler building & configuration. - Maybe even for Windows. ;-) + In version 1.5 we have a flexible, easy to use implementation for Linux. + Windows would be a lot of extra work because USB devices are nothing like Linux. + We'd be starting from scratch to figure out how to do it. */ @@ -130,12 +166,17 @@ typedef int HANDLE; #define INVALID_HANDLE_VALUE (-1) -#endif +#endif /* __WIN32__ */ + +#ifdef USE_CM108 +#include "cm108.h" +#endif /* USE_CM108 */ #include "textcolor.h" #include "audio.h" #include "ptt.h" #include "dlq.h" +#include "demod.h" // to mute recv audio during xmit if half duplex. #if __WIN32__ @@ -157,10 +198,6 @@ typedef int HANDLE; #endif -#if TEST -#define dw_printf printf -#endif - static struct audio_s *save_audio_config_p; /* Save config information for later use. */ @@ -321,6 +358,9 @@ static void get_access_to_gpio (const char *path) * We don't have permission. * Try a hack which requires that the user be set up to use sudo without a password. */ +// FIXME: I think this was a horrible work around for some early release that +// did not give gpio permission to the pi user. This should go. +// Provide recovery instructions when there is a permission failure. if (ptt_debug_level >= 2) { text_color_set(DW_COLOR_ERROR); // debug message but different color so it stands out. @@ -431,6 +471,9 @@ void export_gpio(int ch, int ot, int invert, int direction) exit (1); } } + /* Wait for udev to adjust permissions after enabling GPIO. */ + /* https://github.com/wb2osz/direwolf/issues/176 */ + SLEEP_MS(250); close (fd); /* @@ -455,6 +498,13 @@ void export_gpio(int ch, int ot, int invert, int direction) * matching the pattern "gpio61_*". * * We are finally implementing the third choice. + */ + +/* + * Then we have the Odroid board with GPIO numbers starting around 480. + * Can we simply use those numbers? + * Apparently, the export names look like GPIOX.17 + * https://wiki.odroid.com/odroid-c4/hardware/expansion_connectors#gpio_map_for_wiringpi_library */ struct dirent **file_list; @@ -603,10 +653,12 @@ void export_gpio(int ch, int ot, int invert, int direction) * PTT_METHOD_GPIO - general purpose I/O. * PTT_METHOD_LPT - Parallel printer port. * PTT_METHOD_HAMLIB - HAMLib rig control. + * PTT_METHOD_CM108 - GPIO pins of CM108 etc. USB Audio. * * ptt_device Name of serial port device. * e.g. COM1 or /dev/ttyS0. * HAMLIB can also use hostaddr:port. + * Like /dev/hidraw1 for CM108. * * ptt_line RTS or DTR when using serial port. * @@ -696,7 +748,7 @@ void ptt_init (struct audio_s *audio_config_p) for (ch = 0; ch < MAX_CHANS; ch++) { - if (audio_config_p->achan[ch].valid) { + if (audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { @@ -727,7 +779,7 @@ void ptt_init (struct audio_s *audio_config_p) int j, k; for (j = ch; j >= 0; j--) { - if (audio_config_p->achan[j].valid) { + if (audio_config_p->chan_medium[j] == MEDIUM_RADIO) { for (k = ((j==ch) ? (ot - 1) : (NUM_OCTYPES-1)); k >= 0; k--) { if (strcmp(audio_config_p->achan[ch].octrl[ot].ptt_device,audio_config_p->achan[j].octrl[k].ptt_device) == 0) { fd = ptt_fd[j][k]; @@ -810,7 +862,7 @@ void ptt_init (struct audio_s *audio_config_p) using_gpio = 0; for (ch=0; chachan[ch].valid) { + if (save_audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { if (audio_config_p->achan[ch].octrl[ot].ptt_method == PTT_METHOD_GPIO) { @@ -835,7 +887,7 @@ void ptt_init (struct audio_s *audio_config_p) */ for (ch = 0; ch < MAX_CHANS; ch++) { - if (save_audio_config_p->achan[ch].valid) { + if (save_audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { int ot; // output control type, PTT, DCD, CON, ... int it; // input control type @@ -867,13 +919,13 @@ void ptt_init (struct audio_s *audio_config_p) #if ( defined(__i386__) || defined(__x86_64__) ) && ( defined(__linux__) || defined(__unix__) ) for (ch = 0; ch < MAX_CHANS; ch++) { - if (save_audio_config_p->achan[ch].valid) { + if (save_audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { if (audio_config_p->achan[ch].octrl[ot].ptt_method == PTT_METHOD_LPT) { /* Can't open the same device more than once so we */ - /* need more logic to look for the case of mutiple radio */ + /* need more logic to look for the case of multiple radio */ /* channels using different pins of the LPT port. */ /* Did some earlier channel use the same ptt device name? */ @@ -882,7 +934,7 @@ void ptt_init (struct audio_s *audio_config_p) int j, k; for (j = ch; j >= 0; j--) { - if (audio_config_p->achan[j].valid) { + if (audio_config_p->chan_medium[j] == MEDIUM_RADIO) { for (k = ((j==ch) ? (ot - 1) : (NUM_OCTYPES-1)); k >= 0; k--) { if (strcmp(audio_config_p->achan[ch].octrl[ot].ptt_device,audio_config_p->achan[j].octrl[k].ptt_device) == 0) { fd = ptt_fd[j][k]; @@ -934,25 +986,37 @@ void ptt_init (struct audio_s *audio_config_p) #ifdef USE_HAMLIB for (ch = 0; ch < MAX_CHANS; ch++) { - if (save_audio_config_p->achan[ch].valid) { + if (save_audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { if (audio_config_p->achan[ch].octrl[ot].ptt_method == PTT_METHOD_HAMLIB) { if (ot == OCTYPE_PTT) { + int err = -1; + int tries = 0; /* For "AUTO" model, try to guess what is out there. */ if (audio_config_p->achan[ch].octrl[ot].ptt_model == -1) { - hamlib_port_t hport; + hamlib_port_t hport; // http://hamlib.sourceforge.net/manuals/1.2.15/structhamlib__port__t.html memset (&hport, 0, sizeof(hport)); strlcpy (hport.pathname, audio_config_p->achan[ch].octrl[ot].ptt_device, sizeof(hport.pathname)); + + if (audio_config_p->achan[ch].octrl[ot].ptt_rate > 0) { + // Override the default serial port data rate. + hport.parm.serial.rate = audio_config_p->achan[ch].octrl[ot].ptt_rate; + hport.parm.serial.data_bits = 8; + hport.parm.serial.stop_bits = 1; + hport.parm.serial.parity = RIG_PARITY_NONE; + hport.parm.serial.handshake = RIG_HANDSHAKE_NONE; + } + rig_load_all_backends(); audio_config_p->achan[ch].octrl[ot].ptt_model = rig_probe(&hport); if (audio_config_p->achan[ch].octrl[ot].ptt_model == RIG_MODEL_NONE) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Couldn't guess rig model number for AUTO option. Run \"rigctl --list\" for a list of model numbers.\n"); + dw_printf ("Hamlib Error: Couldn't guess rig model number for AUTO option. Run \"rigctl --list\" for a list of model numbers.\n"); continue; } @@ -964,19 +1028,53 @@ void ptt_init (struct audio_s *audio_config_p) rig[ch][ot] = rig_init(audio_config_p->achan[ch].octrl[ot].ptt_model); if (rig[ch][ot] == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Unknown rig model %d for hamlib. Run \"rigctl --list\" for a list of model numbers.\n", + dw_printf ("Hamlib error: Unknown rig model %d. Run \"rigctl --list\" for a list of model numbers.\n", audio_config_p->achan[ch].octrl[ot].ptt_model); continue; } strlcpy (rig[ch][ot]->state.rigport.pathname, audio_config_p->achan[ch].octrl[ot].ptt_device, sizeof(rig[ch][ot]->state.rigport.pathname)); - int err = rig_open(rig[ch][ot]); + + // Issue 290. + // We had a case where hamlib defaulted to 9600 baud for a particular + // radio model but 38400 was needed. Add an option for the configuration + // file to override the hamlib default speed. + + text_color_set(DW_COLOR_INFO); + if (audio_config_p->achan[ch].octrl[ot].ptt_model != 2) { // 2 is network, not serial port. + dw_printf ("Hamlib determined CAT control serial port rate of %d.\n", rig[ch][ot]->state.rigport.parm.serial.rate); + } + + // Config file can optionally override the rate that hamlib came up with. + + if (audio_config_p->achan[ch].octrl[ot].ptt_rate > 0) { + dw_printf ("User configuration overriding hamlib CAT control speed to %d.\n", audio_config_p->achan[ch].octrl[ot].ptt_rate); + rig[ch][ot]->state.rigport.parm.serial.rate = audio_config_p->achan[ch].octrl[ot].ptt_rate; + + // Do we want to explicitly set all of these or let it default? + rig[ch][ot]->state.rigport.parm.serial.data_bits = 8; + rig[ch][ot]->state.rigport.parm.serial.stop_bits = 1; + rig[ch][ot]->state.rigport.parm.serial.parity = RIG_PARITY_NONE; + rig[ch][ot]->state.rigport.parm.serial.handshake = RIG_HANDSHAKE_NONE; + } + tries = 0; + do { + // Try up to 5 times, Hamlib can take a moment to finish init + err = rig_open(rig[ch][ot]); + if (++tries > 5) { + break; + } else if (err != RIG_OK) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Retrying Hamlib Rig open...\n"); + sleep (5); + } + } while (err != RIG_OK); if (err != RIG_OK) { text_color_set(DW_COLOR_ERROR); dw_printf ("Hamlib Rig open error %d: %s\n", err, rigerror(err)); rig_cleanup (rig[ch][ot]); rig[ch][ot] = NULL; - continue; + exit (1); } /* Successful. Later code should check for rig[ch][ot] not NULL. */ @@ -992,11 +1090,37 @@ void ptt_init (struct audio_s *audio_config_p) #endif +/* + * Confirm what is going on with CM108 GPIO output. + * Could use some error checking for overlap. + */ + +#if USE_CM108 + + for (ch = 0; ch < MAX_CHANS; ch++) { + + if (audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { + int ot; + for (ot = 0; ot < NUM_OCTYPES; ot++) { + if (audio_config_p->achan[ch].octrl[ot].ptt_method == PTT_METHOD_CM108) { + text_color_set(DW_COLOR_INFO); + dw_printf ("Using %s GPIO %d for channel %d %s control.\n", + audio_config_p->achan[ch].octrl[ot].ptt_device, + audio_config_p->achan[ch].octrl[ot].out_gpio_num, + ch, + otnames[ot]); + } + } + } + } + +#endif + /* Why doesn't it transmit? Probably forgot to specify PTT option. */ for (ch=0; chachan[ch].valid) { + if (audio_config_p->chan_medium[ch] == MEDIUM_RADIO) { if(audio_config_p->achan[ch].octrl[OCTYPE_PTT].ptt_method == PTT_METHOD_NONE) { text_color_set(DW_COLOR_INFO); dw_printf ("Note: PTT not configured for channel %d. (Ignore this if using VOX.)\n", ch); @@ -1031,6 +1155,8 @@ void ptt_init (struct audio_s *audio_config_p) * *--------------------------------------------------------------------*/ +// JWL - save status and new get_ptt function. + void ptt_set (int ot, int chan, int ptt_signal) { @@ -1048,18 +1174,33 @@ void ptt_set (int ot, int chan, int ptt_signal) assert (chan >= 0 && chan < MAX_CHANS); - if ( ! save_audio_config_p->achan[chan].valid) { + if ( save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Internal error, ptt_set ( %s, %d, %d ), did not expect invalid channel.\n", otnames[ot], chan, ptt); return; } +// New in 1.7. +// A few people have a really bad audio cross talk situation where they receive their own transmissions. +// It usually doesn't cause a problem but it is confusing to look at. +// "half duplex" setting applied only to the transmit logic. i.e. wait for clear channel before sending. +// Receiving was still active. +// I think the simplest solution is to mute/unmute the audio input at this point if not full duplex. + +#ifndef TEST + if ( ot == OCTYPE_PTT && ! save_audio_config_p->achan[chan].fulldup) { + demod_mute_input (chan, ptt_signal); + } +#endif + /* * The data link state machine has an interest in activity on the radio channel. * This is a very convenient place to get that information. */ +#ifndef TEST dlq_channel_busy (chan, ot, ptt_signal); +#endif /* * Inverted output? @@ -1210,17 +1351,32 @@ void ptt_set (int ot, int chan, int ptt_signal) if (retcode != RIG_OK) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Error sending rig_set_ptt command for channel %d %s\n", chan, otnames[ot]); + dw_printf ("Hamlib Error: rig_set_ptt command for channel %d %s\n", chan, otnames[ot]); dw_printf ("%s\n", rigerror(retcode)); } } else { text_color_set(DW_COLOR_ERROR); - dw_printf ("Can't use rig_set_ptt for channel %d %s because rig_open failed.\n", chan, otnames[ot]); + dw_printf ("Hamlib: Can't use rig_set_ptt for channel %d %s because rig_open failed.\n", chan, otnames[ot]); } } #endif +/* + * Using CM108 USB Audio adapter GPIO? + */ + +#ifdef USE_CM108 + + if (save_audio_config_p->achan[chan].octrl[ot].ptt_method == PTT_METHOD_CM108) { + + if (cm108_set_gpio_pin (save_audio_config_p->achan[chan].octrl[ot].ptt_device, + save_audio_config_p->achan[chan].octrl[ot].out_gpio_num, ptt) != 0) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("ERROR: %s for channel %d has failed. See User Guide for troubleshooting tips.\n", otnames[ot], chan); + } + } +#endif } /* end ptt_set */ @@ -1242,7 +1398,7 @@ int get_input (int it, int chan) assert (it >= 0 && it < NUM_ICTYPES); assert (chan >= 0 && chan < MAX_CHANS); - if ( ! save_audio_config_p->achan[chan].valid) { + if ( save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { text_color_set(DW_COLOR_ERROR); dw_printf ("Internal error, get_input ( %d, %d ), did not expect invalid channel.\n", it, chan); return -1; @@ -1306,7 +1462,7 @@ void ptt_term (void) int n; for (n = 0; n < MAX_CHANS; n++) { - if (save_audio_config_p->achan[n].valid) { + if (save_audio_config_p->chan_medium[n] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { ptt_set (ot, n, 0); @@ -1315,7 +1471,7 @@ void ptt_term (void) } for (n = 0; n < MAX_CHANS; n++) { - if (save_audio_config_p->achan[n].valid) { + if (save_audio_config_p->chan_medium[n] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { if (ptt_fd[n][ot] != INVALID_HANDLE_VALUE) { @@ -1333,7 +1489,7 @@ void ptt_term (void) #ifdef USE_HAMLIB for (n = 0; n < MAX_CHANS; n++) { - if (save_audio_config_p->achan[n].valid) { + if (save_audio_config_p->chan_medium[n] == MEDIUM_RADIO) { int ot; for (ot = 0; ot < NUM_OCTYPES; ot++) { if (rig[n][ot] != NULL) { @@ -1354,18 +1510,15 @@ void ptt_term (void) /* * Quick stand-alone test for above. * - * gcc -DTEST -o ptest ptt.c ; ./ptest + * gcc -DTEST -o ptest ptt.c textcolor.o misc.a ; ./ptest * + * TODO: Retest this, add CM108 GPIO to test. */ #if TEST -void text_color_set (dw_color_t c) { } - -#define dw_printf printf - -main () +int main () { struct audio_s my_audio_config; int n; @@ -1375,17 +1528,18 @@ main () my_audio_config.adev[0].num_channels = 2; - my_audio_config.valid[0] = 1; - my_audio_config.adev[0].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_SERIAL; - //strlcpy (my_audio_config.ptt_device, "COM1", sizeof(my_audio_config.ptt_device)); - strlcpy (my_audio_config.ptt_device, "/dev/ttyUSB0", sizeof(my_audio_config.ptt_device)); - my_audio_config.adev[0].octrl[OCTYPE_PTT].ptt_line = PTT_LINE_RTS; + my_audio_config.chan_medium[0] = MEDIUM_RADIO; + my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_SERIAL; +// TODO: device should be command line argument. + strlcpy (my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_device, "COM3", sizeof(my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_device)); + //strlcpy (my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_device, "/dev/ttyUSB0", sizeof(my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_device)); + my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_line = PTT_LINE_RTS; - my_audio_config.valid[1] = 1; - my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_SERIAL; - //strlcpy (my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_device, "COM1", sizeof(my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_device)); - strlcpy (my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_device, "/dev/ttyUSB0", sizeof(my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_device)); - my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_line = PTT_LINE_DTR; + my_audio_config.chan_medium[1] = MEDIUM_RADIO; + my_audio_config.achan[1].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_SERIAL; + strlcpy (my_audio_config.achan[1].octrl[OCTYPE_PTT].ptt_device, "COM3", sizeof(my_audio_config.achan[1].octrl[OCTYPE_PTT].ptt_device)); + //strlcpy (my_audio_config.achan[1].octrl[OCTYPE_PTT].ptt_device, "/dev/ttyUSB0", sizeof(my_audio_config.achan[1].octrl[OCTYPE_PTT].ptt_device)); + my_audio_config.achan[1].octrl[OCTYPE_PTT].ptt_line = PTT_LINE_DTR; /* initialize - both off */ @@ -1420,7 +1574,7 @@ main () /* Same thing again but invert RTS. */ - my_audio_config.adev[0].octrl[OCTYPE_PTT].ptt_invert = 1; + my_audio_config.achan[0].octrl[OCTYPE_PTT].ptt_invert = 1; ptt_init (&my_audio_config); @@ -1455,7 +1609,7 @@ main () memset (&my_audio_config, 0, sizeof(my_audio_config)); my_audio_config.adev[0].num_channels = 1; - my_audio_config.valid[0] = 1; + my_audio_config.chan_medium[0] = MEDIUM_RADIO; my_audio_config.adev[0].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_GPIO; my_audio_config.adev[0].octrl[OCTYPE_PTT].out_gpio_num = 25; @@ -1476,12 +1630,20 @@ main () #endif + +/* Parallel printer port. */ + +#if ( defined(__i386__) || defined(__x86_64__) ) && ( defined(__linux__) || defined(__unix__) ) + + // TODO + +#if 0 memset (&my_audio_config, 0, sizeof(my_audio_config)); my_audio_config.num_channels = 2; - my_audio_config.valid[0] = 1; + my_audio_config.chan_medium[0] = MEDIUM_RADIO; my_audio_config.adev[0].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_LPT; my_audio_config.adev[0].octrl[OCTYPE_PTT].ptt_lpt_bit = 0; - my_audio_config.valid[1] = 1; + my_audio_config.chan_medium[1] = MEDIUM_RADIO; my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_method = PTT_METHOD_LPT; my_audio_config.adev[1].octrl[OCTYPE_PTT].ptt_lpt_bit = 1; @@ -1496,16 +1658,11 @@ main () } ptt_term (); - -/* Parallel printer port. */ - -#if ( defined(__i386__) || defined(__x86_64__) ) && ( defined(__linux__) || defined(__unix__) ) - - // TODO - #endif +#endif + return(0); } #endif /* TEST */ diff --git a/ptt.h b/src/ptt.h similarity index 100% rename from ptt.h rename to src/ptt.h diff --git a/recv.c b/src/recv.c similarity index 92% rename from recv.c rename to src/recv.c index 8d7826a1..49040e55 100644 --- a/recv.c +++ b/src/recv.c @@ -42,7 +42,7 @@ * multi_modem_process_sample(s) * * - * When a packet is succesfully decoded, somebody calls + * When a packet is successfully decoded, somebody calls * app_process_rec_frame, also in direwolf.c * * @@ -88,6 +88,7 @@ #include #include #include +#include #include //#include //#include @@ -106,7 +107,6 @@ #include "recv.h" #include "dtmf.h" #include "aprs_tt.h" -#include "dtime_now.h" #include "ax25_link.h" @@ -160,7 +160,7 @@ void recv_init (struct audio_s *pa) #endif #if __WIN32__ - xmit_th[a] = (HANDLE)_beginthreadex (NULL, 0, recv_adev_thread, (void*)(long)a, 0, NULL); + xmit_th[a] = (HANDLE)_beginthreadex (NULL, 0, recv_adev_thread, (void*)(ptrdiff_t)a, 0, NULL); if (xmit_th[a] == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("FATAL: Could not create audio receive thread for device %d.\n", a); @@ -168,7 +168,7 @@ void recv_init (struct audio_s *pa) } #else int e; - e = pthread_create (&xmit_tid[a], NULL, recv_adev_thread, (void *)(long)a); + e = pthread_create (&xmit_tid[a], NULL, recv_adev_thread, (void *)(ptrdiff_t)a); if (e != 0) { text_color_set(DW_COLOR_ERROR); @@ -203,11 +203,11 @@ static unsigned __stdcall recv_adev_thread (void *arg) static void * recv_adev_thread (void *arg) #endif { - int a = (int)(long)arg; // audio device number. + int a = (int)(ptrdiff_t)arg; // audio device number. int eof; /* This audio device can have one (mono) or two (stereo) channels. */ - /* Find number of the first channel. */ + /* Find number of the first channel and number of channels. */ int first_chan = ADEVFIRSTCHAN(a); int num_chan = save_pa->adev[a].num_channels; @@ -234,6 +234,8 @@ static void * recv_adev_thread (void *arg) if (audio_sample >= 256 * 256) eof = 1; + // Future? provide more flexible mapping. + // i.e. for each valid channel where audio_source[] is first_chan+c. multi_modem_process_sample(first_chan + c, audio_sample); @@ -262,14 +264,14 @@ static void * recv_adev_thread (void *arg) aprs_tt_button (first_chan + c, tt); } } - } + } // for c is just 0 or 0 then 1 /* When a complete frame is accumulated, */ /* dlq_rec_frame, is called. */ /* recv_process, below, drains the queue. */ - } + } // while !eof on audio stream // What should we do now? // Seimply terminate the application? @@ -337,7 +339,7 @@ void recv_process (void) * - Digipeater. */ - app_process_rec_packet (pitem->chan, pitem->subchan, pitem->slice, pitem->pp, pitem->alevel, pitem->retries, pitem->spectrum); + app_process_rec_packet (pitem->chan, pitem->subchan, pitem->slice, pitem->pp, pitem->alevel, pitem->fec_type, pitem->retries, pitem->spectrum); /* @@ -373,11 +375,21 @@ void recv_process (void) dl_unregister_callsign (pitem); break; + case DLQ_OUTSTANDING_FRAMES_REQUEST: + + dl_outstanding_frames_request (pitem); + break; + case DLQ_CHANNEL_BUSY: lm_channel_busy (pitem); break; + case DLQ_SEIZE_CONFIRM: + + lm_seize_confirm (pitem); + break; + case DLQ_CLIENT_CLEANUP: dl_client_cleanup (pitem); diff --git a/recv.h b/src/recv.h similarity index 100% rename from recv.h rename to src/recv.h diff --git a/redecode.h b/src/redecode.h similarity index 100% rename from redecode.h rename to src/redecode.h diff --git a/rpack.h b/src/rpack.h similarity index 100% rename from rpack.h rename to src/rpack.h diff --git a/rrbb.c b/src/rrbb.c similarity index 90% rename from rrbb.c rename to src/rrbb.c index 82d8aaea..e787dae5 100644 --- a/rrbb.c +++ b/src/rrbb.c @@ -51,8 +51,8 @@ #define MAGIC2 0x56788765 -static int new_count = 0; -static int delete_count = 0; +volatile static int new_count = 0; +volatile static int delete_count = 0; /*********************************************************************************** @@ -88,7 +88,11 @@ rrbb_t rrbb_new (int chan, int subchan, int slice, int is_scrambled, int descram assert (slice >= 0 && slice < MAX_SLICERS); result = malloc(sizeof(struct rrbb_s)); - + if (result == NULL) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("FATAL ERROR: Out of memory.\n"); + exit (EXIT_FAILURE); + } result->magic1 = MAGIC1; result->chan = chan; result->subchan = subchan; @@ -421,6 +425,50 @@ alevel_t rrbb_get_audio_level (rrbb_t b) +/*********************************************************************************** + * + * Name: rrbb_set_speed_error + * + * Purpose: Set speed error of the received frame. + * + * Inputs: b Handle for bit array. + * speed_error In percentage. + * + ***********************************************************************************/ + +void rrbb_set_speed_error (rrbb_t b, float speed_error) +{ + assert (b != NULL); + assert (b->magic1 == MAGIC1); + assert (b->magic2 == MAGIC2); + + b->speed_error = speed_error; +} + + +/*********************************************************************************** + * + * Name: rrbb_get_speed_error + * + * Purpose: Get speed error of the received frame. + * + * Inputs: b Handle for bit array. + * + * Returns: speed error in percentage. + * + ***********************************************************************************/ + +float rrbb_get_speed_error (rrbb_t b) +{ + assert (b != NULL); + assert (b->magic1 == MAGIC1); + assert (b->magic2 == MAGIC2); + + return (b->speed_error); +} + + + /*********************************************************************************** * * Name: rrbb_get_is_scrambled @@ -484,6 +532,7 @@ int rrbb_get_prev_descram (rrbb_t b) } + /* end rrbb.c */ diff --git a/rrbb.h b/src/rrbb.h similarity index 93% rename from rrbb.h rename to src/rrbb.h index 4b283726..894a448f 100644 --- a/rrbb.h +++ b/src/rrbb.h @@ -33,6 +33,7 @@ typedef struct rrbb_s { int slice; /* Which slicer. */ alevel_t alevel; /* Received audio level at time of frame capture. */ + float speed_error; /* Received data speed error as percentage. */ unsigned int len; /* Current number of samples in array. */ int is_scrambled; /* Is data scrambled G3RUH / K9NG style? */ @@ -84,6 +85,9 @@ int rrbb_get_slice (rrbb_t b); void rrbb_set_audio_level (rrbb_t b, alevel_t alevel); alevel_t rrbb_get_audio_level (rrbb_t b); +void rrbb_set_speed_error (rrbb_t b, float speed_error); +float rrbb_get_speed_error (rrbb_t b); + int rrbb_get_is_scrambled (rrbb_t b); int rrbb_get_descram_state (rrbb_t b); int rrbb_get_prev_descram (rrbb_t b); diff --git a/serial_port.c b/src/serial_port.c similarity index 89% rename from serial_port.c rename to src/serial_port.c index 3ccf43c8..c57ee202 100644 --- a/serial_port.c +++ b/src/serial_port.c @@ -1,13 +1,7 @@ - -// TODO: Needs more clean up and testing of error conditions. - -// TODO: use this in place of other similar code. - - // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2014, 2015 John Langner, WB2OSZ +// Copyright (C) 2014, 2015, 2017 John Langner, WB2OSZ // // 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 @@ -50,7 +44,7 @@ #include #include #include -#include +#include #endif @@ -73,8 +67,10 @@ * Inputs: devicename - For Windows, usually like COM5. * For Linux, usually /dev/tty... * "COMn" also allowed and converted to /dev/ttyS(n-1) + * Could be /dev/rfcomm0 for Bluetooth. * - * baud - Speed. 4800, 9600, etc. + * baud - Speed. 1200, 4800, 9600 bps, etc. + * If 0, leave it alone. * * Returns Handle for serial port or MYFDERROR for error. * @@ -143,6 +139,7 @@ MYFDTYPE serial_port_open (char *devicename, int baud) switch (baud) { + case 0: /* Leave it alone. */ break; case 1200: dcb.BaudRate = CBR_1200; break; case 2400: dcb.BaudRate = CBR_2400; break; case 4800: dcb.BaudRate = CBR_4800; break; @@ -233,15 +230,21 @@ MYFDTYPE serial_port_open (char *devicename, int baud) switch (baud) { + case 0: /* Leave it alone. */ break; case 1200: cfsetispeed (&ts, B1200); cfsetospeed (&ts, B1200); break; case 2400: cfsetispeed (&ts, B2400); cfsetospeed (&ts, B2400); break; case 4800: cfsetispeed (&ts, B4800); cfsetospeed (&ts, B4800); break; case 9600: cfsetispeed (&ts, B9600); cfsetospeed (&ts, B9600); break; case 19200: cfsetispeed (&ts, B19200); cfsetospeed (&ts, B19200); break; case 38400: cfsetispeed (&ts, B38400); cfsetospeed (&ts, B38400); break; +// This does not seem to be a problem anymore. +// Leaving traces behind, as clue, in case failure is encountered in some older version. +//#ifndef __APPLE__ + // Not defined for Mac OSX. + // https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/messages/2072 case 57600: cfsetispeed (&ts, B57600); cfsetospeed (&ts, B57600); break; case 115200: cfsetispeed (&ts, B115200); cfsetospeed (&ts, B115200); break; - +//#endif default: text_color_set(DW_COLOR_ERROR); dw_printf ("serial_port_open: Unsupported speed %d. Using 4800.\n", baud); cfsetispeed (&ts, B4800); cfsetospeed (&ts, B4800); @@ -301,15 +304,12 @@ int serial_port_write (MYFDTYPE fd, char *str, int len) { text_color_set(DW_COLOR_ERROR); dw_printf ("Error writing to serial port. Error %d.\n\n", err); + return (-1); } } - else if ((int)nwritten != len) - { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Error writing to serial port. Only %d of %d written.\n\n", (int)nwritten, len); - } - return (nwritten); + // nwritten is 0 for asynchronous write, at this point, so just return the requested len. + return (len); #else int written; @@ -317,8 +317,10 @@ int serial_port_write (MYFDTYPE fd, char *str, int len) written = write (fd, str, (size_t)len); if (written != len) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Error writing to serial port. err=%d\n\n", written); + // Do we want this message here? + // Or rely on caller to check and provide something more meaningful for the usage? + //text_color_set(DW_COLOR_ERROR); + //dw_printf ("Error writing to serial port. err=%d\n\n", written); return (-1); } @@ -402,8 +404,9 @@ int serial_port_get1 (MYFDTYPE fd) CloseHandle(ov_rd.hEvent); if (n != 1) { - text_color_set(DW_COLOR_ERROR); - dw_printf ("Serial port failed to get one byte. n=%d.\n\n", (int)n); + //text_color_set(DW_COLOR_ERROR); + //dw_printf ("Serial port failed to get one byte. n=%d.\n\n", (int)n); + return (-1); } @@ -414,8 +417,8 @@ int serial_port_get1 (MYFDTYPE fd) n = read(fd, &ch, (size_t)1); if (n != 1) { - text_color_set(DW_COLOR_DEBUG); - dw_printf ("serial_port_get1(%d) returns -1 for error.\n", fd); + //text_color_set(DW_COLOR_DEBUG); + //dw_printf ("serial_port_get1(%d) returns -1 for error.\n", fd); return (-1); } @@ -447,8 +450,14 @@ int serial_port_get1 (MYFDTYPE fd) * *--------------------------------------------------------------------*/ - -// TODO: +void serial_port_close (MYFDTYPE fd) +{ +#if __WIN32__ + CloseHandle (fd); +#else + close (fd); +#endif +} /* end serial_port.c */ diff --git a/serial_port.h b/src/serial_port.h similarity index 100% rename from serial_port.h rename to src/serial_port.h diff --git a/server.c b/src/server.c similarity index 75% rename from server.c rename to src/server.c index 2a1ef781..6b41af25 100644 --- a/server.c +++ b/src/server.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2020 John Langner, WB2OSZ // // 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 @@ -45,9 +45,9 @@ * 'k' Ask to start receiving RAW AX25 frames. * * 'm' Ask to start receiving Monitor AX25 frames. + * Enables sending of U, I, S, and T messages to client app. * * 'V' Transmit UI data frame. - * Generate audio for transmission. * * 'H' Report recently heard stations. Not implemented yet. * @@ -59,6 +59,8 @@ * * 'y' Ask Outstanding frames waiting on a Port (new in 1.2) * + * 'Y' How many frames waiting for transmit for a particular station (new in 1.5) + * * 'C' Connect, Start an AX.25 Connection (new in 1.4) * * 'v' Connect VIA, Start an AX.25 circuit thru digipeaters (new in 1.4) @@ -87,11 +89,22 @@ * 'K' Received AX.25 frame in raw format. * (Enabled with 'k' command.) * - * 'U' Received AX.25 frame in monitor format. + * 'U' Received AX.25 "UI" frames in monitor format. + * (Enabled with 'm' command.) + * + * 'I' Received AX.25 "I" frames in monitor format. (new in 1.6) + * (Enabled with 'm' command.) + * + * 'S' Received AX.25 "S" and "U" (other than UI) frames in monitor format. (new in 1.6) + * (Enabled with 'm' command.) + * + * 'T' Own Transmitted AX.25 frames in monitor format. (new in 1.6) * (Enabled with 'm' command.) * * 'y' Outstanding frames waiting on a Port (new in 1.2) * + * 'Y' How many frames waiting for transmit for a particular station (new in 1.5) + * * 'C' AX.25 Connection Received (new in 1.4) * * 'D' Connected AX.25 Data (new in 1.4) @@ -103,6 +116,10 @@ * References: AGWPE TCP/IP API Tutorial * http://uz7ho.org.ua/includes/agwpeapi.htm * + * It has disappeared from the original location but you can find it here: + * https://web.archive.org/web/20130807113413/http:/uz7ho.org.ua/includes/agwpeapi.htm + * https://www.on7lds.net/42/sites/default/files/AGWPEAPI.HTM + * * Getting Started with Winsock * http://msdn.microsoft.com/en-us/library/windows/desktop/bb530742(v=vs.85).aspx * @@ -132,11 +149,7 @@ #include #include #include -#ifdef __OpenBSD__ #include -#else -#include -#endif #endif #include @@ -145,6 +158,7 @@ #include #include #include +#include #include "tq.h" #include "ax25_pad.h" @@ -162,6 +176,7 @@ * You can increase the limit by changing the line below. * A larger number consumes more resources so don't go crazy by making it larger than needed. */ +// FIXME: Put in direwolf.h rather than in .c file. Change name to reflect AGW vs KISS. Update user guide 5.7. #define MAX_NET_CLIENTS 3 @@ -387,7 +402,7 @@ static void debug_print (fromto_t fromto, int client, struct agwpe_s *pmsg, int if (msg_len < 36) { text_color_set (DW_COLOR_ERROR); - dw_printf ("AGWPE message length, %d, is shorter than minumum 36.\n", msg_len); + dw_printf ("AGWPE message length, %d, is shorter than minimum 36.\n", msg_len); } if (msg_len != netle2host(pmsg->data_len_NETLE) + 36) { text_color_set (DW_COLOR_ERROR); @@ -461,14 +476,14 @@ void server_init (struct audio_s *audio_config_p, struct misc_config_s *mc) * This waits for a client to connect and sets an available client_sock[n]. */ #if __WIN32__ - connect_listen_th = (HANDLE)_beginthreadex (NULL, 0, connect_listen_thread, (void *)(unsigned int)server_port, 0, NULL); + connect_listen_th = (HANDLE)_beginthreadex (NULL, 0, connect_listen_thread, (void *)(ptrdiff_t)server_port, 0, NULL); if (connect_listen_th == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Could not create AGW connect listening thread\n"); return; } #else - e = pthread_create (&connect_listen_tid, NULL, connect_listen_thread, (void *)(long)server_port); + e = pthread_create (&connect_listen_tid, NULL, connect_listen_thread, (void *)(ptrdiff_t)server_port); if (e != 0) { text_color_set(DW_COLOR_ERROR); perror("Could not create AGW connect listening thread"); @@ -484,17 +499,19 @@ void server_init (struct audio_s *audio_config_p, struct misc_config_s *mc) for (client = 0; client < MAX_NET_CLIENTS; client++) { #if __WIN32__ - cmd_listen_th[client] = (HANDLE)_beginthreadex (NULL, 0, cmd_listen_thread, (void*)client, 0, NULL); + cmd_listen_th[client] = (HANDLE)_beginthreadex (NULL, 0, cmd_listen_thread, (void*)(ptrdiff_t)client, 0, NULL); if (cmd_listen_th[client] == NULL) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Could not create AGW command listening thread\n"); + dw_printf ("Could not create AGW command listening thread for client %d\n", client); return; } #else - e = pthread_create (&cmd_listen_tid[client], NULL, cmd_listen_thread, (void *)(long)client); + e = pthread_create (&cmd_listen_tid[client], NULL, cmd_listen_thread, (void *)(ptrdiff_t)client); if (e != 0) { text_color_set(DW_COLOR_ERROR); - perror("Could not create AGW command listening thread"); + dw_printf ("Could not create AGW command listening thread for client %d\n", client); + // Replace add perror with better message handling. + perror(""); return; } #endif @@ -533,10 +550,10 @@ static THREAD_F connect_listen_thread (void *arg) SOCKET listen_sock; WSADATA wsadata; - snprintf (server_port_str, sizeof(server_port_str), "%d", (int)(long)arg); + snprintf (server_port_str, sizeof(server_port_str), "%d", (int)(ptrdiff_t)arg); #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("DEBUG: serverport = %d = '%s'\n", (int)(long)arg, server_port_str); + dw_printf ("DEBUG: serverport = %d = '%s'\n", (int)(ptrdiff_t)arg, server_port_str); #endif err = WSAStartup (MAKEWORD(2,2), &wsadata); if (err != 0) { @@ -637,8 +654,7 @@ static THREAD_F connect_listen_thread (void *arg) } text_color_set(DW_COLOR_INFO); -// TODO: "attached" or some other term would be better because "connected" has a different meaning for AX.25. - dw_printf("\nConnected to AGW client application %d ...\n\n", client); + dw_printf("\nAttached to AGW client application %d ...\n\n", client); /* * The command to change this is actually a toggle, not explicit on or off. @@ -655,9 +671,9 @@ static THREAD_F connect_listen_thread (void *arg) #else /* End of Windows case, now Linux */ - struct sockaddr_in sockaddr; /* Internet socket address stuct */ + struct sockaddr_in sockaddr; /* Internet socket address struct */ socklen_t sockaddr_size = sizeof(struct sockaddr_in); - int server_port = (int)(long)arg; + int server_port = (int)(ptrdiff_t)arg; int listen_sock; int bcopt = 1; @@ -729,7 +745,7 @@ static THREAD_F connect_listen_thread (void *arg) client_sock[client] = accept(listen_sock, (struct sockaddr*)(&sockaddr),&sockaddr_size); text_color_set(DW_COLOR_INFO); - dw_printf("\nConnected to AGW client application %d...\n\n", client); + dw_printf("\nAttached to AGW client application %d...\n\n", client); /* * The command to change this is actually a toggle, not explicit on or off. @@ -766,10 +782,13 @@ static THREAD_F connect_listen_thread (void *arg) * * There are two different formats: * RAW - the original received frame. - * MONITOR - just the information part. + * MONITOR - human readable monitoring format. * *--------------------------------------------------------------------*/ +static void mon_addrs (int chan, packet_t pp, char *result, int result_size); +static char mon_desc (packet_t pp, char *result, int result_size); + void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int flen) { @@ -779,15 +798,11 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl } agwpe_msg; int err; - int info_len; - unsigned char *pinfo; - int client; - /* * RAW format */ - for (client=0; client 0){ @@ -813,7 +828,7 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl } #if __WIN32__ - err = send (client_sock[client], (char*)(&agwpe_msg), sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE), 0); + err = SOCK_SEND (client_sock[client], (char*)(&agwpe_msg), sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); if (err == SOCKET_ERROR) { text_color_set(DW_COLOR_ERROR); @@ -824,7 +839,7 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl dlq_client_cleanup (client); } #else - err = write (client_sock[client], &agwpe_msg, sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); + err = SOCK_SEND (client_sock[client], &agwpe_msg, sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); if (err <= 0) { text_color_set(DW_COLOR_ERROR); @@ -837,32 +852,37 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl } } + // Application might want more human readable format. -/* MONITOR format - only for UI frames. */ + server_send_monitored (chan, pp, 0); - for (client=0; client 0 - && ax25_get_control(pp) == AX25_UI_FRAME){ +} /* end server_send_rec_packet */ - time_t clock; - struct tm *tm; - int num_digi; - clock = time(NULL); - tm = localtime(&clock); // TODO: should use localtime_r - memset (&agwpe_msg.hdr, 0, sizeof(agwpe_msg.hdr)); +void server_send_monitored (int chan, packet_t pp, int own_xmit) +{ +/* + * MONITOR format - 'I' for information frames. + * 'U' for unnumbered information. + * 'S' for supervisory and other unnumbered. + */ + struct { + struct agwpe_s hdr; + char data[128+AX25_MAX_PACKET_LEN]; // Add plenty of room for header prefix. + } agwpe_msg; - agwpe_msg.hdr.portx = chan; + int err; - agwpe_msg.hdr.datakind = 'U'; + for (int client=0; client 0) { - ax25_get_addr_with_ssid (pp, AX25_DESTINATION, agwpe_msg.hdr.call_to); + memset (&agwpe_msg.hdr, 0, sizeof(agwpe_msg.hdr)); - info_len = ax25_get_info (pp, &pinfo); + agwpe_msg.hdr.portx = chan; // datakind is added later. + ax25_get_addr_with_ssid (pp, AX25_SOURCE, agwpe_msg.hdr.call_from); + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, agwpe_msg.hdr.call_to); /* http://uz7ho.org.ua/includes/agwpeapi.htm#_Toc500723812 */ @@ -881,44 +901,50 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl // AGWPE: // [AGWE-IN] 1:Fm ZL4FOX-8 To Q7P2U2 Via WIDE3-3 [08:32:14]`I0*l V>/"98}[:Barts Tracker 3.83V X - num_digi = ax25_get_num_repeaters(pp); + // Format the channel and addresses, with leading and trailing space. - if (num_digi > 0) { + mon_addrs (chan, pp, (char*)(agwpe_msg.data), sizeof(agwpe_msg.data)); - char via[AX25_MAX_REPEATERS*(AX25_MAX_ADDR_LEN+1)]; - char stemp[AX25_MAX_ADDR_LEN+1]; - int j; + // Add the description with <... > - ax25_get_addr_with_ssid (pp, AX25_REPEATER_1, via); - for (j = 1; j < num_digi; j++) { - ax25_get_addr_with_ssid (pp, AX25_REPEATER_1 + j, stemp); - strlcat (via, ",", sizeof(via)); - strlcat (via, stemp, sizeof(via)); - } - - snprintf (agwpe_msg.data, sizeof(agwpe_msg.data), " %d:Fm %s To %s Via %s [%02d:%02d:%02d]\r%s\r\r", - chan+1, agwpe_msg.hdr.call_from, agwpe_msg.hdr.call_to, via, - ax25_get_pid(pp), info_len, - tm->tm_hour, tm->tm_min, tm->tm_sec, - pinfo); + char desc[120]; + agwpe_msg.hdr.datakind = mon_desc (pp, desc, sizeof(desc)); + if (own_xmit) { + agwpe_msg.hdr.datakind = 'T'; } - else { + strlcat ((char*)(agwpe_msg.data), desc, sizeof(agwpe_msg.data)); + + // Timestamp with [...]\r - snprintf (agwpe_msg.data, sizeof(agwpe_msg.data), " %d:Fm %s To %s [%02d:%02d:%02d]\r%s\r\r", - chan+1, agwpe_msg.hdr.call_from, agwpe_msg.hdr.call_to, - ax25_get_pid(pp), info_len, - tm->tm_hour, tm->tm_min, tm->tm_sec, - pinfo); + time_t clock = time(NULL); + struct tm *tm = localtime(&clock); // TODO: use localtime_r ? + char ts[32]; + snprintf (ts, sizeof(ts), "[%02d:%02d:%02d]\r", tm->tm_hour, tm->tm_min, tm->tm_sec); + strlcat ((char*)(agwpe_msg.data), ts, sizeof(agwpe_msg.data)); + + // Information if any with \r. + + unsigned char *pinfo = NULL; + int info_len = ax25_get_info (pp, &pinfo); + int msg_data_len = strlen((char*)(agwpe_msg.data)); // result length so far + + if (info_len > 0 && pinfo != NULL) { + // Issue 367: Use of strlcat truncated information part at any nul character. + // Use memcpy instead to preserve binary data, e.g. NET/ROM. + memcpy (agwpe_msg.data + msg_data_len, pinfo, info_len); + msg_data_len += info_len; + agwpe_msg.data[msg_data_len++] = '\r'; } - agwpe_msg.hdr.data_len_NETLE = host2netle(strlen(agwpe_msg.data) + 1) /* +1 to include terminating null */ ; + agwpe_msg.data[msg_data_len++] = '\0'; // add nul at end, included in length. + agwpe_msg.hdr.data_len_NETLE = host2netle(msg_data_len); if (debug_client) { debug_print (TO_CLIENT, client, &agwpe_msg.hdr, sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); } -#if __WIN32__ - err = send (client_sock[client], (char*)(&agwpe_msg), sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE), 0); +#if __WIN32__ + err = SOCK_SEND (client_sock[client], (char*)(&agwpe_msg), sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); if (err == SOCKET_ERROR) { text_color_set(DW_COLOR_ERROR); @@ -929,7 +955,7 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl dlq_client_cleanup (client); } #else - err = write (client_sock[client], &agwpe_msg, sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); + err = SOCK_SEND (client_sock[client], &agwpe_msg, sizeof(agwpe_msg.hdr) + netle2host(agwpe_msg.hdr.data_len_NETLE)); if (err <= 0) { text_color_set(DW_COLOR_ERROR); @@ -942,8 +968,102 @@ void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int fl } } -} /* server_send_rec_packet */ +} /* server_send_monitored */ + + +// Next two are broken out in case they can be reused elsewhere. + +// Format addresses in AGWPR monitoring format such as: +// 1:Fm ZL4FOX-8 To Q7P2U2 Via WIDE3-3 + +static void mon_addrs (int chan, packet_t pp, char *result, int result_size) +{ + char src[AX25_MAX_ADDR_LEN]; + char dst[AX25_MAX_ADDR_LEN]; + + ax25_get_addr_with_ssid (pp, AX25_SOURCE, src); + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, dst); + int num_digi = ax25_get_num_repeaters(pp); + + if (num_digi > 0) { + char via[AX25_MAX_REPEATERS*(AX25_MAX_ADDR_LEN+1)]; + char stemp[AX25_MAX_ADDR_LEN+1]; + int j; + + ax25_get_addr_with_ssid (pp, AX25_REPEATER_1, via); + for (j = 1; j < num_digi; j++) { + ax25_get_addr_with_ssid (pp, AX25_REPEATER_1 + j, stemp); + strlcat (via, ",", sizeof(via)); + strlcat (via, stemp, sizeof(via)); + } + snprintf (result, result_size, " %d:Fm %s To %s Via %s ", + chan+1, src, dst, via); + } + else { + snprintf (result, result_size, " %d:Fm %s To %s ", + chan+1, src, dst); + } +} + + +// Generate frame description in AGWPE monitoring format such as +// +// +// +// +// Returns: +// 'I' for information frame. +// 'U' for unnumbered information frame. +// 'S' for supervisory and other unnumbered frames. + +static char mon_desc (packet_t pp, char *result, int result_size) +{ + cmdres_t cr; // command/response. + char ignore[80]; // direwolf description. not used here. + int pf; // poll/final bit. + int ns; // N(S) Send sequence number. + int nr; // N(R) Received sequence number. + char pf_text[4]; // P or F depending on whether command or response. + + ax25_frame_type_t ftype = ax25_frame_type (pp, &cr, ignore, &pf, &nr, &ns); + + switch (cr) { + case cr_cmd: strcpy(pf_text, "P"); break; // P only: I, SABME, SABM, DISC + case cr_res: strcpy(pf_text, "F"); break; // F only: DM, UA, FRMR + // Either: RR, RNR, REJ, SREJ, UI, XID, TEST + + default: strcpy(pf_text, "PF"); break; // Not AX.25 version >= 2.0 + // APRS is often sloppy about this but it + // is essential for connected mode. + } + + unsigned char *pinfo = NULL; // I, UI, XID, SREJ, TEST can have information part. + int info_len = ax25_get_info (pp, &pinfo); + + switch (ftype) { + + case frame_type_I: snprintf (result, result_size, "", ns, nr, ax25_get_pid(pp), info_len, pf_text, pf); return ('I'); + + case frame_type_U_UI: snprintf (result, result_size, "", ax25_get_pid(pp), info_len, pf_text, pf); return ('U'); break; + + case frame_type_S_RR: snprintf (result, result_size, "", nr, pf_text, pf); return ('S'); break; + case frame_type_S_RNR: snprintf (result, result_size, "", nr, pf_text, pf); return ('S'); break; + case frame_type_S_REJ: snprintf (result, result_size, "", nr, pf_text, pf); return ('S'); break; + case frame_type_S_SREJ: snprintf (result, result_size, "", nr, pf_text, pf, info_len); return ('S'); break; + + case frame_type_U_SABME: snprintf (result, result_size, "", pf_text, pf); return ('S'); break; + case frame_type_U_SABM: snprintf (result, result_size, "", pf_text, pf); return ('S'); break; + case frame_type_U_DISC: snprintf (result, result_size, "", pf_text, pf); return ('S'); break; + case frame_type_U_DM: snprintf (result, result_size, "", pf_text, pf); return ('S'); break; + case frame_type_U_UA: snprintf (result, result_size, "", pf_text, pf); return ('S'); break; + case frame_type_U_FRMR: snprintf (result, result_size, "", pf_text, pf); return ('S'); break; + case frame_type_U_XID: snprintf (result, result_size, "", pf_text, pf, info_len); return ('S'); break; + case frame_type_U_TEST: snprintf (result, result_size, "", pf_text, pf, info_len); return ('S'); break; + default: + case frame_type_U: snprintf (result, result_size, ""); return ('S'); break; + } +} /*------------------------------------------------------------------- @@ -1116,6 +1236,50 @@ void server_rec_conn_data (int chan, int client, char *remote_call, char *own_ca } /* end server_rec_conn_data */ +/*------------------------------------------------------------------- + * + * Name: server_outstanding_frames_reply + * + * Purpose: Send 'Y' Outstanding frames for connected data to the application. + * + * Inputs: chan - Which radio channel. + * + * client - Which one of potentially several clients. + * + * own_call - Callsign[-ssid] of my end. + * + * remote_call - Callsign[-ssid] of remote station. + * + * count - Number of frames sent from the application but + * not yet received by the other station. + * + *--------------------------------------------------------------------*/ + +void server_outstanding_frames_reply (int chan, int client, char *own_call, char *remote_call, int count) +{ + + struct { + struct agwpe_s hdr; + int count_NETLE; + } reply; + + + memset (&reply.hdr, 0, sizeof(reply.hdr)); + + reply.hdr.portx = chan; + reply.hdr.datakind = 'Y'; + + strlcpy (reply.hdr.call_from, own_call, sizeof(reply.hdr.call_from)); + strlcpy (reply.hdr.call_to, remote_call, sizeof(reply.hdr.call_to)); + + reply.hdr.data_len_NETLE = host2netle(4); + reply.count_NETLE = host2netle(count); + + send_to_client (client, &reply); + +} /* end server_outstanding_frames_reply */ + + /*------------------------------------------------------------------- * * Name: read_from_socket @@ -1142,14 +1306,7 @@ static int read_from_socket (int fd, char *ptr, int len) while (got_bytes < len) { int n; -#if __WIN32__ - -//TODO: any flags for send/recv? - - n = recv (fd, ptr + got_bytes, len - got_bytes, 0); -#else - n = read (fd, ptr + got_bytes, len - got_bytes); -#endif + n = SOCK_RECV (fd, ptr + got_bytes, len - got_bytes); #if DEBUG text_color_set(DW_COLOR_DEBUG); @@ -1192,10 +1349,7 @@ static void send_to_client (int client, void *reply_p) { struct agwpe_s *ph; int len; -#if __WIN32__ -#else int err; -#endif ph = (struct agwpe_s *) reply_p; // Replies are often hdr + other stuff. @@ -1213,12 +1367,8 @@ static void send_to_client (int client, void *reply_p) debug_print (TO_CLIENT, client, ph, len); } -#if __WIN32__ - send (client_sock[client], (char*)(ph), len, 0); -#else - err = write (client_sock[client], ph, len); + err = SOCK_SEND (client_sock[client], (char*)(ph), len); (void)err; -#endif } @@ -1229,11 +1379,12 @@ static THREAD_F cmd_listen_thread (void *arg) struct { struct agwpe_s hdr; /* Command header. */ - char data[512]; /* Additional data used by some commands. */ + char data[AX25_MAX_PACKET_LEN]; /* Additional data used by some commands. */ /* Maximum for 'V': 1 + 8*10 + 256 */ + /* Maximum for 'D': Info part length + 1 */ } cmd; - int client = (int)(long)arg; + int client = (int)(ptrdiff_t)arg; assert (client >= 0 && client < MAX_NET_CLIENTS); @@ -1260,16 +1411,20 @@ static THREAD_F cmd_listen_thread (void *arg) } /* - * Take some precautions to guard against bad data - * which could cause problems later. + * Take some precautions to guard against bad data which could cause problems later. */ + if (cmd.hdr.portx < 0 || cmd.hdr.portx >= MAX_CHANS) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("\nInvalid port number, %d, in command '%c', from AGW client application %d.\n", + cmd.hdr.portx, cmd.hdr.datakind, client); + cmd.hdr.portx = 0; // avoid subscript out of bounds, try to keep going. + } /* - * Call to/from must not exceeed 9 characters. + * Call to/from fields are 10 bytes but contents must not exceed 9 characters. * It's not guaranteed that unused bytes will contain 0 so we * don't issue error message in this case. */ - cmd.hdr.call_from[sizeof(cmd.hdr.call_from)-1] = '\0'; cmd.hdr.call_to[sizeof(cmd.hdr.call_to)-1] = '\0'; @@ -1386,47 +1541,72 @@ static THREAD_F cmd_listen_thread (void *arg) // We can have gaps in the numbering. // I wonder what applications will think about that. -#if 1 // No other place cares about total number. count = 0; for (j=0; jachan[j].valid) { + if (save_audio_config_p->chan_medium[j] == MEDIUM_RADIO || + save_audio_config_p->chan_medium[j] == MEDIUM_IGATE || + save_audio_config_p->chan_medium[j] == MEDIUM_NETTNC) { count++; } } snprintf (reply.info, sizeof(reply.info), "%d;", count); for (j=0; jachan[j].valid) { - char stemp[100]; - int a = ACHAN2ADEV(j); - // If I was really ambitious, some description could be provided. - static const char *names[8] = { "first", "second", "third", "fourth", "fifth", "sixth", "seventh", "eighth" }; - - if (save_audio_config_p->adev[a].num_channels == 1) { - snprintf (stemp, sizeof(stemp), "Port%d %s soundcard mono;", j+1, names[a]); - strlcat (reply.info, stemp, sizeof(reply.info)); - } - else { - snprintf (stemp, sizeof(stemp), "Port%d %s soundcard %s;", j+1, names[a], j&1 ? "right" : "left"); - strlcat (reply.info, stemp, sizeof(reply.info)); - } - } - } -#else - if (num_channels == 1) { - snprintf (reply.info, sizeof(reply.info), "1;Port1 Single channel;"); - } - else { - snprintf (reply.info, sizeof(reply.info), "2;Port1 Left channel;Port2 Right Channel;"); - } -#endif + switch (save_audio_config_p->chan_medium[j]) { + + case MEDIUM_RADIO: + { + char stemp[100]; + int a = ACHAN2ADEV(j); + // If I was really ambitious, some description could be provided. + static const char *names[8] = { "first", "second", "third", "fourth", "fifth", "sixth", "seventh", "eighth" }; + + if (save_audio_config_p->adev[a].num_channels == 1) { + snprintf (stemp, sizeof(stemp), "Port%d %s soundcard mono;", j+1, names[a]); + strlcat (reply.info, stemp, sizeof(reply.info)); + } + else { + snprintf (stemp, sizeof(stemp), "Port%d %s soundcard %s;", j+1, names[a], j&1 ? "right" : "left"); + strlcat (reply.info, stemp, sizeof(reply.info)); + } + } + break; + + case MEDIUM_IGATE: + { + char stemp[100]; + snprintf (stemp, sizeof(stemp), "Port%d Internet Gateway;", j+1); + strlcat (reply.info, stemp, sizeof(reply.info)); + } + break; + + case MEDIUM_NETTNC: + { + // could elaborate with hostname, etc. + char stemp[100]; + snprintf (stemp, sizeof(stemp), "Port%d Network TNC;", j+1); + strlcat (reply.info, stemp, sizeof(reply.info)); + } + break; + + default: + { + // could elaborate with hostname, etc. + char stemp[100]; + snprintf (stemp, sizeof(stemp), "Port%d INVALID CHANNEL;", j+1); + strlcat (reply.info, stemp, sizeof(reply.info)); + } + break; + + } // switch + } // for each channel + reply.hdr.data_len_NETLE = host2netle(strlen(reply.info) + 1); send_to_client (client, &reply); - } break; @@ -1456,6 +1636,7 @@ static THREAD_F cmd_listen_thread (void *arg) // YAAC asks for this. // Fake it to keep application happy. + // TODO: Supply real values instead of just faking it. reply.on_air_baud_rate = 0; reply.traffic_level = 1; @@ -1635,6 +1816,11 @@ static THREAD_F cmd_listen_thread (void *arg) break; + case 'P': /* Application Login */ + + // Silently ignore it. + break; + case 'X': /* Register CallSign */ { @@ -1651,7 +1837,9 @@ static THREAD_F cmd_listen_thread (void *arg) int chan = cmd.hdr.portx; - if (chan >= 0 && chan < MAX_CHANS && save_audio_config_p->achan[chan].valid) { + // Connected mode can only be used with internal modems. + + if (chan >= 0 && chan < MAX_CHANS && save_audio_config_p->chan_medium[chan] == MEDIUM_RADIO) { ok = 1; dlq_register_callsign (cmd.hdr.call_from, chan, client); } @@ -1678,7 +1866,9 @@ static THREAD_F cmd_listen_thread (void *arg) int chan = cmd.hdr.portx; - if (chan >= 0 && chan < MAX_CHANS && save_audio_config_p->achan[chan].valid) { + // Connected mode can only be used with internal modems. + + if (chan >= 0 && chan < MAX_CHANS && save_audio_config_p->chan_medium[chan] == MEDIUM_RADIO) { dlq_unregister_callsign (cmd.hdr.call_from, chan, client); } else { @@ -1686,7 +1876,7 @@ static THREAD_F cmd_listen_thread (void *arg) dw_printf ("AGW protocol error. Unregister callsign for invalid channel %d.\n", chan); } } - /* No reponse is expected. */ + /* No response is expected. */ break; case 'C': /* Connect, Start an AX.25 Connection */ @@ -1697,7 +1887,17 @@ static THREAD_F cmd_listen_thread (void *arg) struct via_info { unsigned char num_digi; /* Expect to be in range 1 to 7. Why not up to 8? */ char dcall[7][10]; - } *v = (struct via_info *)cmd.data; + } +#if 1 + // October 2017. gcc ??? complained: + // warning: dereferencing pointer 'v' does break strict-aliasing rules + // Try adding this attribute to get rid of the warning. + // If this upsets your compiler, take it out. + // Let me know. Maybe we could put in a compiler version check here. + + __attribute__((__may_alias__)) +#endif + *v = (struct via_info *)cmd.data; char callsigns[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; int num_calls = 2; /* 2 plus any digipeaters. */ @@ -1742,8 +1942,10 @@ static THREAD_F cmd_listen_thread (void *arg) case 'D': /* Send Connected Data */ { - char callsigns[2][AX25_MAX_ADDR_LEN]; - const int num_calls = 2; + char callsigns[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; + memset (callsigns, 0, sizeof(callsigns)); + const int num_calls = 2; // only first 2 used. Digipeater path + // must be remembered from connect request. strlcpy (callsigns[AX25_SOURCE], cmd.hdr.call_from, sizeof(callsigns[AX25_SOURCE])); strlcpy (callsigns[AX25_DESTINATION], cmd.hdr.call_to, sizeof(callsigns[AX25_SOURCE])); @@ -1756,8 +1958,9 @@ static THREAD_F cmd_listen_thread (void *arg) case 'd': /* Disconnect, Terminate an AX.25 Connection */ { - char callsigns[2][AX25_MAX_ADDR_LEN]; - const int num_calls = 2; + char callsigns[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; + memset (callsigns, 0, sizeof(callsigns)); + const int num_calls = 2; // only first 2 used. strlcpy (callsigns[AX25_SOURCE], cmd.hdr.call_from, sizeof(callsigns[AX25_SOURCE])); strlcpy (callsigns[AX25_DESTINATION], cmd.hdr.call_to, sizeof(callsigns[AX25_SOURCE])); @@ -1843,6 +2046,7 @@ static THREAD_F cmd_listen_thread (void *arg) case 'y': /* Ask Outstanding frames waiting on a Port */ + /* Number of frames sitting in transmit queue for specified channel. */ { struct { struct agwpe_s hdr; @@ -1857,7 +2061,8 @@ static THREAD_F cmd_listen_thread (void *arg) int n = 0; if (cmd.hdr.portx >= 0 && cmd.hdr.portx < MAX_CHANS) { - n = tq_count (cmd.hdr.portx, TQ_PRIO_0_HI) + tq_count (cmd.hdr.portx, TQ_PRIO_1_LO); + // Count both normal and expedited in transmit queue for given channel. + n = tq_count (cmd.hdr.portx, -1, "", "", 0); } reply.data_NETLE = host2netle(n); @@ -1865,6 +2070,57 @@ static THREAD_F cmd_listen_thread (void *arg) } break; + case 'Y': /* How Many Outstanding frames wait for tx for a particular station */ + + // This is different than the above 'y' because this refers to a specific + // link in connected mode. + + // This would be useful for a couple different purposes. + + // When sending bulk data, we want to keep a fair amount queued up to take + // advantage of large window sizes (MAXFRAME, EMAXFRAME). On the other + // hand we don't want to get TOO far ahead when transferring a large file. + + // Before disconnecting from another station, it would be good to know + // that it actually received the last message we sent. For this reason, + // I think it would be good for this to include information frames that were + // transmitted but not yet acknowledged. + // You could say that a particular frame is still waiting to be sent even + // if was already sent because it could be sent again if lost previously. + + // The documentation is inconsistent about the address order. + // One place says "callfrom" is my callsign and "callto" is the other guy. + // That would make sense. We are asking about frames going to the other guy. + + // But another place says it depends on who initiated the connection. + // + // "If we started the connection CallFrom=US and CallTo=THEM + // If the other end started the connection CallFrom=THEM and CallTo=US" + // + // The response description says nothing about the order; it just mentions two addresses. + // If you are writing a client or server application, the order would + // be clear but right here it could be either case. + // + // Another version of the documentation mentioned the source address being optional. + // + + // The only way to get this information is from inside the data link state machine. + // We will send a request to it and the result coming out will be used to + // send the reply back to the client application. + + { + + char callsigns[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN]; + memset (callsigns, 0, sizeof(callsigns)); + const int num_calls = 2; // only first 2 used. + + strlcpy (callsigns[AX25_SOURCE], cmd.hdr.call_from, sizeof(callsigns[AX25_SOURCE])); + strlcpy (callsigns[AX25_DESTINATION], cmd.hdr.call_to, sizeof(callsigns[AX25_SOURCE])); + + dlq_outstanding_frames_request (callsigns, num_calls, cmd.hdr.portx, client); + } + break; + default: text_color_set(DW_COLOR_ERROR); diff --git a/server.h b/src/server.h similarity index 80% rename from server.h rename to src/server.h index 08db5713..4cc2ea0a 100644 --- a/server.h +++ b/src/server.h @@ -15,6 +15,8 @@ void server_init (struct audio_s *audio_config_p, struct misc_config_s *misc_con void server_send_rec_packet (int chan, packet_t pp, unsigned char *fbuf, int flen); +void server_send_monitored (int chan, packet_t pp, int own_xmit); + int server_callsign_registered_by_client (char *callsign); @@ -24,5 +26,7 @@ void server_link_terminated (int chan, int client, char *remote_call, char *own_ void server_rec_conn_data (int chan, int client, char *remote_call, char *own_call, int pid, char *data_ptr, int data_len); +void server_outstanding_frames_reply (int chan, int client, char *own_call, char *remote_call, int count); + /* end server.h */ diff --git a/symbols.c b/src/symbols.c similarity index 91% rename from symbols.c rename to src/symbols.c index ad3263c3..c9f07e6e 100644 --- a/symbols.c +++ b/src/symbols.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2013, 2014, 2015 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2013, 2014, 2015, 2022 John Langner, WB2OSZ // // 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 @@ -193,9 +193,9 @@ static const struct { /* ; 27 */ { "NS", "Park/Picnic area" }, /* < 28 */ { "NT", "ADVISORY (one WX flag)" }, /* = 29 */ { "NU", "APRStt Touchtone (DTMF users)" }, - /* > 30 */ { "NV", "OVERLAYED CAR" }, + /* > 30 */ { "NV", "OVERLAID CAR" }, /* ? 31 */ { "NW", "INFO Kiosk (Blue box with ?)" }, - /* @ 32 */ { "NX", "HURICANE/Trop-Storm" }, + /* @ 32 */ { "NX", "HURRICANE/Trop-Storm" }, /* A 33 */ { "AA", "overlayBOX DTMF & RFID & XO" }, /* B 34 */ { "AB", "Blwng Snow (& future codes)" }, /* C 35 */ { "AC", "Coast Guard" }, @@ -205,7 +205,7 @@ static const struct { /* G 39 */ { "AG", "Snow Shwr (& future ovrlys)" }, /* H 40 */ { "AH", "Haze (& Overlay Hazards)" }, /* I 41 */ { "AI", "Rain Shower" }, - /* J 42 */ { "AJ", "Lightening (& future ovrlys)" }, + /* J 42 */ { "AJ", "Lightning (& future ovrlys)" }, /* K 43 */ { "AK", "Kenwood HT (W)" }, /* L 44 */ { "AL", "Lighthouse" }, /* M 45 */ { "AM", "MARS (A=Army,N=Navy,F=AF)" }, @@ -248,12 +248,12 @@ static const struct { /* r 82 */ { "SR", "Restrooms" }, /* s 83 */ { "SS", "OVERLAY SHIP/boat (top view)" }, /* t 84 */ { "ST", "Tornado" }, - /* u 85 */ { "SU", "OVERLAYED TRUCK" }, - /* v 86 */ { "SV", "OVERLAYED Van" }, + /* u 85 */ { "SU", "OVERLAID TRUCK" }, + /* v 86 */ { "SV", "OVERLAID Van" }, /* w 87 */ { "SW", "Flooding" }, /* x 88 */ { "SX", "Wreck or Obstruction ->X<-" }, /* y 89 */ { "SY", "Skywarn" }, - /* z 90 */ { "SZ", "OVERLAYED Shelter" }, + /* z 90 */ { "SZ", "OVERLAID Shelter" }, /* { 91 */ { "Q1", "Fog (& future ovrly codes)" }, /* | 92 */ { "Q2", "TNC Stream Switch" }, /* } 93 */ { "Q3", "" }, @@ -261,13 +261,24 @@ static const struct { // Make sure the array is null terminated. +// If search order is changed, do the same in decode_aprs.c for consistency. + static const char *search_locations[] = { - (const char *) "symbols-new.txt", + (const char *) "symbols-new.txt", // CWD + (const char *) "data/symbols-new.txt", // Windows with Cmake + (const char *) "../data/symbols-new.txt", // ? #ifndef __WIN32__ - (const char *) "/usr/share/direwolf/symbols-new.txt", (const char *) "/usr/local/share/direwolf/symbols-new.txt", + (const char *) "/usr/share/direwolf/symbols-new.txt", +#endif +#if __APPLE__ + // https://groups.yahoo.com/neo/groups/direwolf_packet/conversations/messages/2458 + // Adding the /opt/local tree since macports typically installs there. Users might want their + // INSTALLDIR (see Makefile.macosx) to mirror that. If so, then we need to search the /opt/local + // path as well. + (const char *) "/opt/local/share/direwolf/symbols-new.txt", #endif - (const char *) NULL + (const char *) NULL // Important - Indicates end of list. }; @@ -287,7 +298,7 @@ static const char *search_locations[] = { * Description: The primary and alternate symbol tables are constant * so they are hardcoded. * However the "new" sysmbols, which give new meanings to - * overlayed symbols, are always evolving. + * OVERLAID symbols, are always evolving. * For maximum flexibility, we will read the * data file at run time rather than compiling it in. * @@ -308,7 +319,7 @@ static const char *search_locations[] = { typedef struct new_sym_s { char overlay; char symbol; - char description[NEW_SYM_DESC_LEN+1]; + char *description; } new_sym_t; static new_sym_t *new_sym_ptr = NULL; /* Dynamically allocated array. */ @@ -341,10 +352,22 @@ void symbols_init (void) char stuff[200]; int j; +// Feb. 2022 - Noticed that some lines have - rather than =. + +// LD = LIght Rail or Subway (new Aug 2014) +// SD = Seaport Depot (new Aug 2014) +// DIGIPEATERS +// /# - Generic digipeater +// 1# - WIDE1-1 digipeater + + #define GOOD_LINE(x) (strlen(x) > 6 && \ (x[COL1_OVERLAY] == '/' || x[COL1_OVERLAY] == '\\' || isupper(x[COL1_OVERLAY]) || isdigit(x[COL1_OVERLAY])) \ && x[COL2_SYMBOL] >= '!' && x[COL2_SYMBOL] <= '~' \ - && x[COL3_SP] == ' ' && x[COL4_EQUAL] == '=' && x[COL5_SP] == ' ' && x[COL6_DESC] != ' ') + && x[COL3_SP] == ' ' \ + && (x[COL4_EQUAL] == '=' || x[COL4_EQUAL] == '-') \ + && x[COL5_SP] == ' ' \ + && x[COL6_DESC] != ' ') if (new_sym_ptr != NULL) { return; /* was called already. */ @@ -363,7 +386,7 @@ void symbols_init (void) text_color_set(DW_COLOR_ERROR); dw_printf ("Warning: Could not open 'symbols-new.txt'.\n"); - dw_printf ("The \"new\" overlayed character information will not be available.\n"); + dw_printf ("The \"new\" OVERLAID character information will not be available.\n"); new_sym_size = 1; new_sym_ptr = calloc(new_sym_size, sizeof(new_sym_t)); /* Don't try again. */ @@ -395,7 +418,7 @@ void symbols_init (void) } new_sym_ptr[new_sym_len].overlay = stuff[COL1_OVERLAY]; new_sym_ptr[new_sym_len].symbol = stuff[COL2_SYMBOL]; - strncpy(new_sym_ptr[new_sym_len].description, stuff+COL6_DESC, NEW_SYM_DESC_LEN); + new_sym_ptr[new_sym_len].description = strdup(stuff+COL6_DESC); new_sym_len++; } } @@ -484,7 +507,7 @@ void symbols_list (void) } } dw_printf ("\n"); - + dw_printf ("More information here: http://www.aprs.org/symbols.html\n"); } /* end symbols_list */ @@ -545,6 +568,7 @@ static const char ssid_to_sym[16] = { 'v' /* 15 - Van */ }; + void symbols_from_dest_or_src (char dti, char *src, char *dest, char *symtab, char *symbol) { char *p; @@ -648,18 +672,34 @@ void symbols_from_dest_or_src (char dti, char *src, char *dest, char *symtab, ch /* * When all else fails, use source SSID. + * This is totally non-obvious and confusing, but it is in the APRS protocol spec. + * Chapter 20, "Symbol in the Source Address SSID" */ - p = strchr (src, '-'); - if (p != NULL) - { - int ssid; +// January 2022 - Every time this shows up, it confuses people terribly. +// e.g. An APRS "message" shows up with Bus or Motorcycle in the description. - ssid = atoi(p+1); - if (ssid >= 1 && ssid <= 15) { - *symtab = '/'; /* All in Primary table. */ - *symbol = ssid_to_sym[ssid]; - return; +// The position and object formats all contain a proper symbol and table. +// There doesn't seem to be much reason to have a symbol for something without +// a position because it would not show up on a map. +// This just seems to be a remnant of something used long ago and no longer needed. +// The protocol spec mentions a "MIM tracker" but I can't find any references to it. + +// If this was completely removed, no one would probably ever notice. +// The only possible useful case I can think of would be someone sending a +// NMEA string directly from a GPS receiver and wanting to keep the destination field +// for the system type. + + if (dti == '$') { + + p = strchr (src, '-'); + if (p != NULL) { + int ssid = atoi(p+1); + if (ssid >= 1 && ssid <= 15) { + *symtab = '/'; /* All in Primary table. */ + *symbol = ssid_to_sym[ssid]; + return; + } } } @@ -1050,4 +1090,4 @@ int main (int argc, char *argv[]) #endif -/* end symbols.c */ \ No newline at end of file +/* end symbols.c */ diff --git a/symbols.h b/src/symbols.h similarity index 100% rename from symbols.h rename to src/symbols.h diff --git a/telemetry.c b/src/telemetry.c similarity index 94% rename from telemetry.c rename to src/telemetry.c index 0d389d03..2a6c690c 100644 --- a/telemetry.c +++ b/src/telemetry.c @@ -69,7 +69,7 @@ #define T_NUM_ANALOG 5 /* Number of analog channels. */ -#define T_NUM_DIGITAL 8 /* Number of digital channnels. */ +#define T_NUM_DIGITAL 8 /* Number of digital channels. */ #define T_STR_LEN 16 /* Max len for labels and units. */ @@ -143,9 +143,10 @@ static struct t_metadata_s * t_get_metadata (char *station) for (p = md_list_head; p != NULL; p = p->pnext) { if (strcmp(station, p->station) == 0) { - assert (p->magic1 == MAGIC1); - assert (p->magic2 == MAGIC2); - + if (p->magic1 != MAGIC1 || p->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } return (p); } } @@ -252,7 +253,10 @@ static int t_ndp (char *str) * KB1GKN-10>APRX27,UNCAN,WIDE1*:T#491,4.9,0.3,25.0,0.0,1.0,00000000 * * Not integers. Not fixed width fields. - * We will accept these but issue a warning that others might not. + * + * Originally I printed a warning if values were not in range of 000 to 255 + * but later took it out because no one pays attention to that original + * restriction anymore. * *--------------------------------------------------------------------*/ @@ -282,8 +286,10 @@ void telemetry_data_original (char *station, char *info, int quiet, char *output pm = t_get_metadata(station); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } seq = 0; for (n = 0; n < T_NUM_ANALOG; n++) { @@ -297,13 +303,13 @@ void telemetry_data_original (char *station, char *info, int quiet, char *output if (strncmp(info, "T#", 2) != 0) { if ( ! quiet) { text_color_set(DW_COLOR_ERROR); - dw_printf("Error: Information part of telemetry packet must begin with \"#\"\n"); + dw_printf("Error: Information part of telemetry packet must begin with \"T#\"\n"); } return; } /* - * Make a copy of the input string because this will alter it. + * Make a copy of the input string (excluding T#) because this will alter it. * Remove any trailing CR/LF. */ @@ -315,6 +321,15 @@ void telemetry_data_original (char *station, char *info, int quiet, char *output next = stemp; p = strsep(&next,","); + + if (p == NULL) { + if ( ! quiet) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Nothing after \"T#\" for telemetry data.\n"); + } + return; + } + seq = atoi(p); n = 0; while ((p = strsep(&next,",")) != NULL) { @@ -470,8 +485,10 @@ void telemetry_data_base91 (char *station, char *cdata, char *output, size_t out pm = t_get_metadata(station); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } seq = 0; for (n = 0; n < T_NUM_ANALOG; n++) { @@ -576,8 +593,11 @@ void telemetry_name_message (char *station, char *msg) } pm = t_get_metadata(station); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } next = stemp; @@ -652,8 +672,11 @@ void telemetry_unit_label_message (char *station, char *msg) } pm = t_get_metadata(station); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } next = stemp; @@ -684,7 +707,7 @@ void telemetry_unit_label_message (char *station, char *msg) * * Name: telemetry_coefficents_message * - * Purpose: Interpret message with scaling coefficents for analog channels. + * Purpose: Interpret message with scaling coefficients for analog channels. * * Inputs: station - Name of station reporting telemetry. * In this case it is the destination for the message, @@ -729,8 +752,11 @@ void telemetry_coefficents_message (char *station, char *msg, int quiet) } pm = t_get_metadata(station); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } next = stemp; @@ -745,7 +771,7 @@ void telemetry_coefficents_message (char *station, char *msg, int quiet) else { if ( ! quiet) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Equation coefficent position A%d%c is empty.\n", n/3+1, n%3+'a'); + dw_printf ("Equation coefficient position A%d%c is empty.\n", n/3+1, n%3+'a'); dw_printf ("Some applications might not handle this correctly.\n"); } } @@ -756,7 +782,7 @@ void telemetry_coefficents_message (char *station, char *msg, int quiet) if (n != T_NUM_ANALOG * 3) { if ( ! quiet) { text_color_set(DW_COLOR_ERROR); - dw_printf ("Found %d equation coefficents when 15 were expected.\n", n); + dw_printf ("Found %d equation coefficients when 15 were expected.\n", n); dw_printf ("Some applications might not handle this correctly.\n"); } } @@ -781,7 +807,7 @@ void telemetry_coefficents_message (char *station, char *msg, int quiet) * * Name: telemetry_bit_sense_message * - * Purpose: Interpret message with scaling coefficents for analog channels. + * Purpose: Interpret message with scaling coefficients for analog channels. * * Inputs: station - Name of station reporting telemetry. * In this case it is the destination for the message, @@ -809,8 +835,11 @@ void telemetry_bit_sense_message (char *station, char *msg, int quiet) #endif pm = t_get_metadata(station); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } if (strlen(msg) < 8) { if ( ! quiet) { @@ -881,7 +910,7 @@ void telemetry_bit_sense_message (char *station, char *msg, int quiet) * seq - Sequence number. * araw - 5 analog raw values. * ndp - Number of decimal points for each. - * draw - 8 digial raw vales. + * draw - 8 digital raw vales. * * Outputs: output - Decoded telemetry in human readable format. * @@ -919,8 +948,11 @@ static void t_data_process (struct t_metadata_s *pm, int seq, float araw[T_NUM_A assert (pm != NULL); - assert (pm->magic1 == MAGIC1); - assert (pm->magic2 == MAGIC2); + + if (pm->magic1 != MAGIC1 || pm->magic2 != MAGIC2) { + text_color_set(DW_COLOR_ERROR); + dw_printf("Internal error: REPORT THIS! Bad magic values %s %d\n", __func__, __LINE__); + } strlcpy (output, "", outputsize); diff --git a/telemetry.h b/src/telemetry.h similarity index 100% rename from telemetry.h rename to src/telemetry.h diff --git a/src/textcolor.c b/src/textcolor.c new file mode 100644 index 00000000..dea90f09 --- /dev/null +++ b/src/textcolor.c @@ -0,0 +1,401 @@ + +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2011, 2012, 2013, 2014, 2019 John Langner, WB2OSZ +// +// 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, see . +// + + +/*------------------------------------------------------------------- + * + * Name: textcolor.c + * + * Purpose: Originally this would only set color of text + * and we used printf everywhere. + * Now we also have a printf replacement that can + * be used to redirect all output to the desired place. + * This opens the door to using ncurses, a GUI, or + * running as a daemon. + * + * Description: For Linux and Cygwin use the ANSI escape sequences. + * In earlier versions of Windows, the cmd window and ANSI.SYS + * could interpret this but it doesn't seem to be available + * anymore so we use a different interface. + * + * Reference: + * http://en.wikipedia.org/wiki/ANSI_escape_code + * + * + +>>>> READ THIS PART!!! <<<< + + * + * + * Problem: Years ago, when I started on this... + * + * The ANSI escape sequences, used for text colors, allowed 8 basic colors. + * Unfortunately, white is not one of them. We only have dark + * white, also known as light gray. To get brighter colors, + * we need to apply an attribute. On some systems, the bold + * attribute produces a brighter color rather than a bold font. + * On other systems, we need to use the blink attribute to get + * bright colors, including white. However on others, blink + * does actually produce blinking characters. + * + * Previously, the only option was to put "-t 0" on the command + * line to disable all text color. This is more readable but + * makes it harder to distinguish different types of + * information, e.g. received packets vs. error messages. + * + * A few people have suggested ncurses. + * I looked at ncurses, and it doesn't seem to be the solution. + * It always sends the same color control codes rather than + * detecting the terminal type and adjusting its behavior. + * + * Version 1.6: + * + * For a long time, there was a compile time distinction between + * ARM (e.g. Raspberry Pi) and other platforms. With the arrival + * of Raspbian Buster, we get flashing and the general Linux settings + * work better. + * + * Since there doesn't seem to be a single universal solution, + * the text color option will now be allowed to have multiple values. + * Several people have also complained that bright green is + * very hard to read against a light background so only dark green will be used. + * + *--------------------------------------------------------------------*/ + + +#include "direwolf.h" // Should be first. includes windows.h + +#include +#include +#include + + +#if __WIN32__ + +#define BACKGROUND_WHITE (BACKGROUND_RED | BACKGROUND_GREEN | BACKGROUND_BLUE | BACKGROUND_INTENSITY) + +#else /* Linux, BSD, Mac OSX */ + +// Alternative 1: + +// Using RGB colors - New in version 1.6. +// Since version 1.2, we've been using RGB to set the background to white. +// From this we can deduce that pretty much everyone recognizes RGB colors by now. +// The only known exception was PuTTY 0.70 and this has been rectified in 0.71. +// Instead of picking 1 of 8 colors, and using some attribute to get bright, just specify it directly. +// This should eliminate the need to reset the background after messing with the bright/bold/blink +// attributes to get more than 8 colors. + + +// Alternative 2: + +// Was new in version 1.2, as suggested by IW2DHW. +// Tested with gnome-terminal and xterm. +// Raspbian Buster LXTerminal also likes this. +// There was probably an issue with an earlier release because I intentionally made ARM different at one time. + +// Here we are using the RGB color format to set the background. +// PuTTY 0.70 doesn't recognize the RGB format so the background is not set. +// Instead of complaining about it, just upgrade to PuTTY 0.71. + + +// Alternative 3: + +// For some terminals we needed "blink" (5) rather than the expected bright/bold (1) +// attribute to get bright white background. +// Makes no sense but I stumbled across that somewhere. + +// In some cases, you might find background (around text but not rest of line) is set to white. +// On GNOME Terminal and LXTerminal, this produces blinking text with a gray background. + + +// Alternative 4: + +// This is using the bright/bold attribute, as you would expect from the documentation. +// Whenever a dark color is used, the background is reset and needs to be set again. +// In recent tests, background is always gray, not white like it should be. + + +#define MAX_T 4 + +static const char *t_background_white[MAX_T+1] = { "", "\e[48;2;255;255;255m", "\e[48;2;255;255;255m", "\e[5;47m", "\e[1;47m" }; + +static const char *t_black[MAX_T+1] = { "", "\e[38;2;0;0;0m", "\e[0;30m" "\e[48;2;255;255;255m", "\e[0;30m" "\e[5;47m", "\e[0;30m" "\e[1;47m" }; +static const char *t_red[MAX_T+1] = { "", "\e[38;2;255;0;0m", "\e[1;31m" "\e[48;2;255;255;255m", "\e[1;31m" "\e[5;47m", "\e[1;31m" "\e[1;47m" }; +static const char *t_green[MAX_T+1] = { "", "\e[38;2;0;255;0m", "\e[1;32m" "\e[48;2;255;255;255m", "\e[1;32m" "\e[5;47m", "\e[1;32m" "\e[1;47m" }; +static const char *t_dark_green[MAX_T+1]= { "", "\e[38;2;0;192;0m", "\e[0;32m" "\e[48;2;255;255;255m", "\e[0;32m" "\e[5;47m", "\e[0;32m" "\e[1;47m" }; +static const char *t_yellow[MAX_T+1] = { "", "\e[38;2;255;255;0m", "\e[1;33m" "\e[48;2;255;255;255m", "\e[1;33m" "\e[5;47m", "\e[1;33m" "\e[1;47m" }; +static const char *t_blue[MAX_T+1] = { "", "\e[38;2;0;0;255m", "\e[1;34m" "\e[48;2;255;255;255m", "\e[1;34m" "\e[5;47m", "\e[1;34m" "\e[1;47m" }; +static const char *t_magenta[MAX_T+1] = { "", "\e[38;2;255;0;255m", "\e[1;35m" "\e[48;2;255;255;255m", "\e[1;35m" "\e[5;47m", "\e[1;35m" "\e[1;47m" }; +static const char *t_cyan[MAX_T+1] = { "", "\e[38;2;0;255;255m", "\e[0;36m" "\e[48;2;255;255;255m", "\e[0;36m" "\e[5;47m", "\e[0;36m" "\e[1;47m" }; + + +/* Clear from cursor to end of screen. */ + +static const char clear_eos[] = "\e[0J"; + +#endif /* end Linux */ + + +#include "textcolor.h" + + +/* + * g_enable_color: + * 0 = disable text colors. + * 1 = default, should be good for LXTerminal >= 0.3.2, GNOME Terminal, xterm, PuTTY >= 0.71. + * 2 = what we had for a few earlier versions. Should be good for LXTerminal, GNOME Terminal, xterm. + * 3 = use 8 basic colors, blinking attribute to get brighter color. Best for older PuTTY. + * 4 = use 8 basic colors, bold attribute to get brighter color. + * + * others... future possibility - tell me if none of these work properly for your terminal type. + * + * 9 (more accurately any invalid value) = try all of them and exit. + */ + +static int g_enable_color = 1; + + +void text_color_init (int enable_color) +{ + + +#if __WIN32__ + + + if (g_enable_color != 0) { + + HANDLE h; + CONSOLE_SCREEN_BUFFER_INFO csbi; + WORD attr = BACKGROUND_WHITE; + DWORD length; + COORD coord; + DWORD nwritten; + + h = GetStdHandle(STD_OUTPUT_HANDLE); + if (h != NULL && h != INVALID_HANDLE_VALUE) { + + GetConsoleScreenBufferInfo (h, &csbi); + + length = csbi.dwSize.X * csbi.dwSize.Y; + coord.X = 0; + coord.Y = 0; + FillConsoleOutputAttribute (h, attr, length, coord, &nwritten); + } + } + +#else + +// Run a test if outside of acceptable range. + + if (enable_color < 0 || enable_color > MAX_T) { + int t; + for (t = 0; t <= MAX_T; t++) { + text_color_init (t); + printf ("-t %d", t); + if (t) printf (" [white background] "); + printf ("\n"); + printf ("%sBlack ", t_black[t]); + printf ("%sRed ", t_red[t]); + printf ("%sGreen ", t_green[t]); + printf ("%sDark-Green ", t_dark_green[t]); + printf ("%sYellow ", t_yellow[t]); + printf ("%sBlue ", t_blue[t]); + printf ("%sMagenta ", t_magenta[t]); + printf ("%sCyan \n", t_cyan[t]); + } + exit (EXIT_SUCCESS); + } + + g_enable_color = enable_color; + + if (g_enable_color != 0) { + int t = g_enable_color; + + if (t < 0) t = 0; + if (t > MAX_T) t = MAX_T; + + printf ("%s", t_background_white[t]); + printf ("%s", clear_eos); + printf ("%s", t_black[t]); + } +#endif +} + + +#if __WIN32__ + +/* Seems that ANSI.SYS is no longer available. */ + + +void text_color_set ( enum dw_color_e c ) +{ + WORD attr; + HANDLE h; + + if (g_enable_color == 0) { + return; + } + + switch (c) { + + default: + case DW_COLOR_INFO: + attr = BACKGROUND_WHITE; + break; + + case DW_COLOR_ERROR: + attr = FOREGROUND_RED | FOREGROUND_INTENSITY | BACKGROUND_WHITE; + break; + + case DW_COLOR_REC: + // Release 1.6. Dark green, same as for debug. + // Bright green is too hard to see with white background, + // attr = FOREGROUND_GREEN | FOREGROUND_INTENSITY | BACKGROUND_WHITE; + attr = FOREGROUND_GREEN | BACKGROUND_WHITE; + break; + + case DW_COLOR_DECODED: + attr = FOREGROUND_BLUE | FOREGROUND_INTENSITY | BACKGROUND_WHITE; + break; + + case DW_COLOR_XMIT: + attr = FOREGROUND_RED | FOREGROUND_BLUE | FOREGROUND_INTENSITY | BACKGROUND_WHITE; + break; + + case DW_COLOR_DEBUG: + attr = FOREGROUND_GREEN | BACKGROUND_WHITE; + break; + } + + h = GetStdHandle(STD_OUTPUT_HANDLE); + + if (h != NULL && h != INVALID_HANDLE_VALUE) { + SetConsoleTextAttribute (h, attr); + } +} + +#else + +void text_color_set ( enum dw_color_e c ) +{ + + if (g_enable_color == 0) { + return; + } + + int t = g_enable_color; + + if (t < 0) t = 0; + if (t > MAX_T) t = MAX_T; + + switch (c) { + + default: + case DW_COLOR_INFO: + printf ("%s", t_black[t]); + break; + + case DW_COLOR_ERROR: + printf ("%s", t_red[t]); + break; + + case DW_COLOR_REC: + // Bright green is very difficult to read against a while background. + // Let's use dark green instead. release 1.6. + //printf ("%s", t_green[t]); + printf ("%s", t_dark_green[t]); + break; + + case DW_COLOR_DECODED: + printf ("%s", t_blue[t]); + break; + + case DW_COLOR_XMIT: + printf ("%s", t_magenta[t]); + break; + + case DW_COLOR_DEBUG: + printf ("%s", t_dark_green[t]); + break; + } +} + +#endif + + +/*------------------------------------------------------------------- + * + * Name: dw_printf + * + * Purpose: printf replacement that allows us to send all text + * output to stdout or other desired destination. + * + * Inputs: fmt - C language format. + * ... - Additional arguments, just like printf. + * + * + * Returns: Number of characters in result. + * + * Bug: Fixed size buffer. + * I'd rather not do a malloc for each print. + * + *--------------------------------------------------------------------*/ + + +// TODO: replace all printf, look for stderr, perror +// TODO: $ grep printf *.c | grep -v dw_printf | grep -v fprintf | gawk '{ print $1 }' | sort -u + + +int dw_printf (const char *fmt, ...) +{ +#define BSIZE 1000 + va_list args; + char buffer[BSIZE]; + int len; + + va_start (args, fmt); + len = vsnprintf (buffer, BSIZE, fmt, args); + va_end (args); + +// TODO: other possible destinations... + + fputs (buffer, stdout); + return (len); +} + + + +#if TESTC +main () +{ + printf ("Initial condition\n"); + text_color_init (1); + printf ("After text_color_init\n"); + text_color_set(DW_COLOR_INFO); printf ("Info\n"); + text_color_set(DW_COLOR_ERROR); printf ("Error\n"); + text_color_set(DW_COLOR_REC); printf ("Rec\n"); + text_color_set(DW_COLOR_DECODED); printf ("Decoded\n"); + text_color_set(DW_COLOR_XMIT); printf ("Xmit\n"); + text_color_set(DW_COLOR_DEBUG); printf ("Debug\n"); +} +#endif + +/* end textcolor.c */ diff --git a/textcolor.h b/src/textcolor.h similarity index 100% rename from textcolor.h rename to src/textcolor.h diff --git a/src/tnctest.c b/src/tnctest.c new file mode 100644 index 00000000..0d4c26b4 --- /dev/null +++ b/src/tnctest.c @@ -0,0 +1,1284 @@ +// +// This file is part of Dire Wolf, an amateur radio packet TNC. +// +// Copyright (C) 2016 John Langner, WB2OSZ +// +// 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, see . +// + + +/*------------------------------------------------------------------ + * + * Module: tnctest.c + * + * Purpose: Test AX.25 connected mode between two TNCs. + * + * Description: The first TNC will connect to the second TNC and send a bunch of data. + * Proper transfer of data will be verified. + * + * Usage: tnctest [options] port0=name0 port1=name1 + * + * Example: tnctest localhost:8000=direwolf COM1=KPC-3+ + * + * Each port can have the following forms: + * + * * host-name:tcp-port + * * ip-addr:tcp-port + * * tcp-port + * * serial port name (e.g. COM1, /dev/ttyS0) + * + *---------------------------------------------------------------*/ + + + +/* + * Native Windows: Use the Winsock interface. + * Linux: Use the BSD socket interface. + */ + +#include "direwolf.h" // Sets _WIN32_WINNT for XP API level needed by ws2tcpip.h + +#if __WIN32__ + +#include +#include // _WIN32_WINNT must be set to 0x0501 before including this + +#else + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +//#include +#include +#endif + +#include +#include +#include +#include +#include +#include + + +//#include "ax25_pad.h" +#include "textcolor.h" +#include "dtime_now.h" +#include "serial_port.h" + + +/* We don't deal with big-endian processors here. */ +/* TODO: Use agwlib (which did not exist when this was written) */ +/* rather than duplicating the effort here. */ + +struct agwpe_s { + +#if 1 + + unsigned char portx; /* 0 for first, 1 for second, etc. */ + unsigned char reserved1; + unsigned char reserved2; + unsigned char reserved3; + + unsigned char datakind; /* message type, usually written as a letter. */ + unsigned char reserved4; + unsigned char pid; + unsigned char reserved5; + +#else + short portx; /* 0 for first, 1 for second, etc. */ + short port_hi_reserved; + short kind_lo; /* message type */ + short kind_hi; +#endif + char call_from[10]; + char call_to[10]; + int data_len; /* Number of data bytes following. */ + int user_reserved; +}; + + +#if __WIN32__ +static unsigned __stdcall tnc_thread_net (void *arg); +static unsigned __stdcall tnc_thread_serial (void *arg); +#else +static void * tnc_thread_net (void *arg); +static void * tnc_thread_serial (void *arg); +#endif + + +static void tnc_connect (int from, int to); +static void tnc_disconnect (int from, int to); +static void tnc_send_data (int from, int to, char * data); +static void tnc_reset (int from, int to); + +/* + * Convert Internet address to text. + * Can't use InetNtop because it is supported only on Windows Vista and later. + */ + +static char * ia_to_text (int Family, void * pAddr, char * pStringBuf, size_t StringBufSize) +{ + struct sockaddr_in *sa4; + struct sockaddr_in6 *sa6; + + switch (Family) { + case AF_INET: + sa4 = (struct sockaddr_in *)pAddr; +#if __WIN32__ + snprintf (pStringBuf, StringBufSize, "%d.%d.%d.%d", sa4->sin_addr.S_un.S_un_b.s_b1, + sa4->sin_addr.S_un.S_un_b.s_b2, + sa4->sin_addr.S_un.S_un_b.s_b3, + sa4->sin_addr.S_un.S_un_b.s_b4); +#else + inet_ntop (AF_INET, &(sa4->sin_addr), pStringBuf, StringBufSize); +#endif + break; + case AF_INET6: + sa6 = (struct sockaddr_in6 *)pAddr; +#if __WIN32__ + snprintf (pStringBuf, StringBufSize, "%x:%x:%x:%x:%x:%x:%x:%x", + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[0]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[1]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[2]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[3]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[4]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[5]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[6]), + ntohs(((unsigned short *)(&(sa6->sin6_addr)))[7])); +#else + inet_ntop (AF_INET6, &(sa6->sin6_addr), pStringBuf, StringBufSize); +#endif + break; + default: + snprintf (pStringBuf, StringBufSize, "Invalid address family!"); + } + assert (strlen(pStringBuf) < StringBufSize); + return pStringBuf; +} + + + +/*------------------------------------------------------------------ + * + * Name: main + * + * Purpose: Basic test for connected AX.25 data mode between TNCs. + * + * Usage: Described above. + * + *---------------------------------------------------------------*/ + +#define MAX_TNC 2 // Just 2 for now. + // Could be more later for multiple concurrent connections. + +/* Obtained from the command line. */ + +static int num_tnc; /* How many TNCs for this test? */ + /* Initially only 2 but long term we might */ + /* enhance it to allow multiple concurrent connections. */ + +static char hostname[MAX_TNC][50]; /* DNS host name or IPv4 address. */ + /* Some of the code is there for IPv6 but */ + /* needs more work. */ + /* Defaults to "localhost" if not specified. */ + +static char port[MAX_TNC][30]; /* If it begins with a digit, it is considered */ + /* a TCP port number at the hostname. */ + /* Otherwise, we treat it as a serial port name. */ + +static char description[MAX_TNC][50]; /* Name used in the output. */ + +static int using_tcp[MAX_TNC]; /* Are we using TCP or serial port for each TNC? */ + /* Use corresponding one of the next two. */ + +static int server_sock[MAX_TNC]; /* File descriptor for AGW socket interface. */ + /* Set to -1 if not used. */ + /* (Don't use SOCKET type because it is unsigned.) */ + +static MYFDTYPE serial_fd[MAX_TNC]; /* Serial port handle. */ + +static volatile int busy[MAX_TNC]; /* True when TNC busy and can't accept more data. */ + /* For serial port, this is set by XON / XOFF characters. */ + +#define XOFF 0x13 +#define XON 0x11 + +static char tnc_address[MAX_TNC][12]; /* Name of the TNC used in the frames. Originally, this */ + /* was simply TNC0 and TNC1 but that can get hard to read */ + /* and confusing. Later used DW0, DW1, for direwolf */ + /* so the direction of flow is easier to grasp. */ + +#if __WIN32__ + static HANDLE tnc_th[MAX_TNC]; +#else + static pthread_t tnc_tid[MAX_TNC]; +#endif + +#define LINE_WIDTH 80 +//#define LINE_WIDTH 120 /* If I was more ambitious I might try to get */ + /* this from the terminal properties. */ + +static int column_width; /* Line width divided by number of TNCs. */ + + +/* + * Current state for each TNC. + */ + +static int is_connected[MAX_TNC]; /* -1 = not yet available. */ + /* 0 = not connected. */ + /* 1 = not connected. */ + +static int have_cmd_prompt[MAX_TNC]; /* Set if "cmd:" was the last thing seen. */ + +static int last_rec_seq[MAX_TNC]; /* Each data packet will contain a sequence number. */ + /* This is used to verify that all have been */ + /* received in the correct order. */ + + + +/* + * Start time so we can print relative elapsed time. + */ + +static double start_dtime; + + +static int max_count; + +int main (int argc, char *argv[]) +{ + int j; + int timeout; + int send_count = 0; + int burst_size = 1; + int errors = 0; + + //max_count = 20; + max_count = 200; + //max_count = 6; + max_count = 1000; + max_count = 9999; + +#if __WIN32__ +#else + int e; + + setlinebuf (stdout); +#endif + + start_dtime = dtime_now(); + +/* + * Extract command line args. + */ + num_tnc = argc - 1; + + if (num_tnc < 2 || num_tnc > MAX_TNC) { + printf ("Specify minimum 2, maximum %d TNCs on the command line.\n", MAX_TNC); + exit (EXIT_FAILURE); + } + + column_width = LINE_WIDTH / num_tnc; + + for (j=0; j 0) { + + SLEEP_MS(100); + timeout--; + ready = 1; + for (j=0; j last0) { + last0 = last_rec_seq[0]; + no_activity = 0; + } + if (last_rec_seq[1] > last1) { + last1 = last_rec_seq[1]; + no_activity = 0; + } + } + + if (last_rec_seq[0] == max_count) { + printf ("Got last expected reply.\n"); + } + else { + printf ("ERROR: Timeout - No incoming activity for %d seconds.\n", no_activity); + errors++; + } + +/* + * Did we get all expected replies? + */ + if (last_rec_seq[0] != max_count) { + printf ("ERROR: Last received reply was %d when we were expecting %d.\n", last_rec_seq[0], max_count); + errors++; + } + +/* + * Ask for disconnect. Wait until complete. + */ + + tnc_disconnect (0, 1); + + timeout = 200; // 20 sec should be generous. + ready = 0; + while ( ! ready && timeout > 0) { + + SLEEP_MS(100); + timeout--; + ready = 1; + for (j=0; j1 for the other end which answers. + * + * data - Should look something like this: + * 9999 send data + * 9999 reply + * + * Global In/Out: last_rec_seq[my_index] + * + * Description: Look for expected format. + * Extract the sequence number. + * Verify that it is the next expected one. + * Update it. + * + *--------------------------------------------------------------------*/ + +void process_rec_data (int my_index, char *data) +{ + int n; + + if (isdigit(*data) && strncmp(data+4, " send", 5) == 0) { + if (my_index > 0) { + last_rec_seq[my_index]++; + + n = atoi(data); + if (n != last_rec_seq[my_index]) { + printf ("%*s%s: Received %d when %d was expected.\n", my_index*column_width, "", tnc_address[my_index], n, last_rec_seq[my_index]); + SLEEP_MS(10000); + printf ("TEST FAILED!\n"); + exit (EXIT_FAILURE); + } + } + } + + else if (isdigit(*data) && strncmp(data+4, " reply", 6) == 0) { + if (my_index == 0) { + last_rec_seq[my_index]++; + n = atoi(data); + if (n != last_rec_seq[my_index]) { + printf ("%*s%s: Received %d when %d was expected.\n", my_index*column_width, "", tnc_address[my_index], n, last_rec_seq[my_index]); + SLEEP_MS(10000); + printf ("TEST FAILED!\n"); + exit (EXIT_FAILURE); + } + } + } + + else if (data[0] == 'A') { + + if (strncmp(data, "ABCDEFGHIJKLMNOPQRSTUVWXYZ", strlen(data)-1) != 0) { + printf ("%*s%s: Segmentation is broken.\n", my_index*column_width, "", tnc_address[my_index]); + SLEEP_MS(10000); + printf ("TEST FAILED!\n"); + exit (EXIT_FAILURE); + } + } +} + + + +/*------------------------------------------------------------------- + * + * Name: tnc_thread_net + * + * Purpose: Establish connection with a TNC via network. + * + * Inputs: arg - My instance index, 0 thru MAX_TNC-1. + * + * Outputs: packets - Received packets are put in the corresponding column + * and sent to a common function to check that they + * all arrived in order. + * + * Global Out: is_connected - Updated when connected/disconnected notifications are received. + * + * Description: Perform any necessary configuration for the TNC then wait + * for responses and process them. + * + *--------------------------------------------------------------------*/ + +#define MAX_HOSTS 30 + +#if __WIN32__ +static unsigned __stdcall tnc_thread_net (void *arg) +#else +static void * tnc_thread_net (void *arg) +#endif +{ + int my_index; + struct addrinfo hints; + struct addrinfo *ai_head = NULL; + struct addrinfo *ai; + struct addrinfo *hosts[MAX_HOSTS]; + int num_hosts, n; + int err; + char ipaddr_str[46]; /* text form of IP address */ +#if __WIN32__ + WSADATA wsadata; +#endif + + struct agwpe_s mon_cmd; + char data[4096]; + double dnow; + + + my_index = (int)(ptrdiff_t)arg; + +#if DEBUGx + printf ("DEBUG: tnc_thread_net %d start, port = '%s'\n", my_index, port[my_index]); +#endif + +#if __WIN32__ + err = WSAStartup (MAKEWORD(2,2), &wsadata); + if (err != 0) { + printf("WSAStartup failed: %d\n", err); + return (0); + } + + if (LOBYTE(wsadata.wVersion) != 2 || HIBYTE(wsadata.wVersion) != 2) { + printf("Could not find a usable version of Winsock.dll\n"); + WSACleanup(); + //sleep (1); + return (0); + } +#endif + + memset (&hints, 0, sizeof(hints)); + + hints.ai_family = AF_UNSPEC; /* Allow either IPv4 or IPv6. */ + // hints.ai_family = AF_INET; /* IPv4 only. */ + hints.ai_socktype = SOCK_STREAM; + hints.ai_protocol = IPPROTO_TCP; + +/* + * Connect to TNC server. + */ + + ai_head = NULL; + err = getaddrinfo(hostname[my_index], port[my_index], &hints, &ai_head); + if (err != 0) { +#if __WIN32__ + printf ("Can't get address for server %s, err=%d\n", + hostname[my_index], WSAGetLastError()); +#else + printf ("Can't get address for server %s, %s\n", + hostname[my_index], gai_strerror(err)); +#endif + freeaddrinfo(ai_head); + exit (1); + } + +#if DEBUG_DNS + printf ("getaddrinfo returns:\n"); +#endif + num_hosts = 0; + for (ai = ai_head; ai != NULL; ai = ai->ai_next) { +#if DEBUG_DNS + ia_to_text (ai->ai_family, ai->ai_addr, ipaddr_str, sizeof(ipaddr_str)); + printf (" %s\n", ipaddr_str); +#endif + hosts[num_hosts] = ai; + if (num_hosts < MAX_HOSTS) num_hosts++; + } + +#if DEBUG_DNS + printf ("addresses for hostname:\n"); + for (n=0; nai_family, hosts[n]->ai_addr, ipaddr_str, sizeof(ipaddr_str)); + printf (" %s\n", ipaddr_str); + } +#endif + + // Try each address until we find one that is successful. + + for (n=0; nai_family, ai->ai_addr, ipaddr_str, sizeof(ipaddr_str)); + is = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol); +#if __WIN32__ + if (is == INVALID_SOCKET) { + printf ("Socket creation failed, err=%d", WSAGetLastError()); + WSACleanup(); + is = -1; + continue; + } +#else + if (err != 0) { + printf ("Socket creation failed, err=%s", gai_strerror(err)); + (void) close (is); + is = -1; + continue; + } +#endif + +#ifndef DEBUG_DNS + err = connect(is, ai->ai_addr, (int)ai->ai_addrlen); +#if __WIN32__ + if (err == SOCKET_ERROR) { +#if DEBUGx + printf("Connect to %s on %s (%s), port %s failed.\n", + description[my_index], hostname[my_index], ipaddr_str, port[my_index]); +#endif + closesocket (is); + is = -1; + continue; + } +#else + if (err != 0) { +#if DEBUGx + printf("Connect to %s on %s (%s), port %s failed.\n", + description[my_index], hostname[my_index], ipaddr_str, port[my_index]); +#endif + (void) close (is); + is = -1; + continue; + } + int flag = 1; + err = setsockopt (is, IPPROTO_TCP, TCP_NODELAY, (void*)(long)(&flag), (socklen_t)sizeof(flag)); + if (err < 0) { + printf("setsockopt TCP_NODELAY failed.\n"); + } +#endif + +/* Success. */ + + + server_sock[my_index] = is; +#endif + break; + } + + freeaddrinfo(ai_head); + + if (server_sock[my_index] == -1) { + + printf("TNC %d unable to connect to %s on %s (%s), port %s\n", + my_index, description[my_index], hostname[my_index], ipaddr_str, port[my_index] ); + exit (1); + } + + +#if 1 // Temp test just to get something. + +/* + * Send command to toggle reception of frames in raw format. + */ + memset (&mon_cmd, 0, sizeof(mon_cmd)); + + mon_cmd.datakind = 'k'; + + SOCK_SEND(server_sock[my_index], (char*)(&mon_cmd), sizeof(mon_cmd)); + +#endif + +/* + * Send command to register my callsign for incoming connect request. + * Not really needed when we initiate the connection. + */ + + memset (&mon_cmd, 0, sizeof(mon_cmd)); + + mon_cmd.datakind = 'X'; + strlcpy (mon_cmd.call_from, tnc_address[my_index], sizeof(mon_cmd.call_from)); + + SOCK_SEND(server_sock[my_index], (char*)(&mon_cmd), sizeof(mon_cmd)); + + + +/* + * Inform main program and observer that we are ready to go. + */ + printf("TNC %d now available. %s on %s (%s), port %s\n", + my_index, description[my_index], hostname[my_index], ipaddr_str, port[my_index] ); + is_connected[my_index] = 0; + +/* + * Print what we get from TNC. + */ + + while (1) { + int n; + + n = SOCK_RECV (server_sock[my_index], (char*)(&mon_cmd), sizeof(mon_cmd)); + + if (n != sizeof(mon_cmd)) { + printf ("Read error, TNC %d received %d command bytes.\n", my_index, n); + exit (1); + } + + +#if DEBUGx + printf ("TNC %d received '%c' data, data_len = %d\n", + my_index, mon_cmd.datakind, mon_cmd.data_len); +#endif + assert (mon_cmd.data_len >= 0 && mon_cmd.data_len < (int)(sizeof(data))); + + if (mon_cmd.data_len > 0) { + + n = SOCK_RECV (server_sock[my_index], data, mon_cmd.data_len); + + if (n != mon_cmd.data_len) { + printf ("Read error, TNC %d received %d data bytes.\n", my_index, n); + exit (1); + } + data[mon_cmd.data_len] = '\0'; + } + +/* + * What did we get? + */ + + dnow = dtime_now(); + + switch (mon_cmd.datakind) { + + case 'C': // AX.25 Connection Received + + printf("%*s[R %.3f] *** Connected to %s ***\n", my_index*column_width, "", dnow-start_dtime, mon_cmd.call_from); + is_connected[my_index] = 1; + + break; + + case 'D': // Connected AX.25 Data + + printf("%*s[R %.3f] %s\n", my_index*column_width, "", dnow-start_dtime, data); + + process_rec_data (my_index, data); + + + if (isdigit(data[0]) && isdigit(data[1]) && isdigit(data[2]) && isdigit(data[3]) && + strncmp(data+4, " send", 5) == 0) { + // Expected message. Make sure it is expected sequence and send reply. + int n = atoi(data); + char reply[80]; + snprintf (reply, sizeof(reply), "%04d reply\r", n); + tnc_send_data (my_index, 1 - my_index, reply); + + // HACK! + // It gets very confusing because N(S) and N(R) are very close. + // Send a couple dozen I frames so they will be easier to distinguish visually. + // Currently don't have the same in serial port version. + + // We change the length each time to test segmentation. + // Set PACLEN to some very small number like 5. + + if (n == 1 && max_count > 1) { + int j; + for (j = 1; j <= 26; j++) { + snprintf (reply, sizeof(reply), "%.*s\r", j, "ABCDEFGHIJKLMNOPQRSTUVWXYZ"); + tnc_send_data (my_index, 1 - my_index, reply); + } + } + } + + break; + + case 'd': // Disconnected + + printf("%*s[R %.3f] *** Disconnected from %s ***\n", my_index*column_width, "", dnow-start_dtime, mon_cmd.call_from); + is_connected[my_index] = 0; + + break; + + case 'y': // Outstanding frames waiting on a Port + + printf("%*s[R %.3f] *** Outstanding frames waiting %d ***\n", my_index*column_width, "", dnow-start_dtime, 123); // TODO + + break; + + default: + + //printf("%*s[R %.3f] --- Ignoring cmd kind '%c' ---\n", my_index*column_width, "", dnow-start_dtime, mon_cmd.datakind); + + break; + } + } + +} /* end tnc_thread_net */ + + + + + +/*------------------------------------------------------------------- + * + * Name: tnc_thread_serial + * + * Purpose: Establish connection with a TNC via serial port. + * + * Inputs: arg - My instance index, 0 thru MAX_TNC-1. + * + * Outputs: packets - Received packets are put in the corresponding column + * and sent to a common function to check that they + * all arrived in order. + * + * Global Out: is_connected - Updated when connected/disconnected notifications are received. + * + * Description: Perform any necessary configuration for the TNC then wait + * for responses and process them. + * + *--------------------------------------------------------------------*/ + + +#if __WIN32__ +typedef HANDLE MYFDTYPE; +#define MYFDERROR INVALID_HANDLE_VALUE +#else +typedef int MYFDTYPE; +#define MYFDERROR (-1) +#endif + + +#if __WIN32__ +static unsigned __stdcall tnc_thread_serial (void *arg) +#else +static void * tnc_thread_serial (void *arg) +#endif +{ + int my_index = (int)(ptrdiff_t)arg; + char cmd[80]; + + serial_fd[my_index] = serial_port_open (port[my_index], 9600); + + if (serial_fd[my_index] == MYFDERROR) { + printf("TNC %d unable to connect to %s on %s.\n", + my_index, description[my_index], port[my_index] ); + exit (1); + } + + +/* + * Make sure we are in command mode. + */ + + strcpy (cmd, "\003\rreset\r"); + serial_port_write (serial_fd[my_index], cmd, strlen(cmd)); + SLEEP_MS (3000); + + strcpy (cmd, "echo on\r"); + serial_port_write (serial_fd[my_index], cmd, strlen(cmd)); + SLEEP_MS (200); + +// do any necessary set up here. such as setting mycall + + snprintf (cmd, sizeof(cmd), "mycall %s\r", tnc_address[my_index]); + serial_port_write (serial_fd[my_index], cmd, strlen(cmd)); + SLEEP_MS (200); + +// Don't want to stop tty output when typing begins. + + strcpy (cmd, "flow off\r"); + serial_port_write (serial_fd[my_index], cmd, strlen(cmd)); + + strcpy (cmd, "echo off\r"); + serial_port_write (serial_fd[my_index], cmd, strlen(cmd)); + +/* Success. */ + + printf("TNC %d now available. %s on %s\n", + my_index, description[my_index], port[my_index] ); + is_connected[my_index] = 0; + + +/* + * Read and print. + */ + + while (1) { + int ch; + char result[500]; + int len; + int done; + + len = 0; + result[len] = '\0'; + done = 0; + + while ( ! done) { + + ch = serial_port_get1(serial_fd[my_index]); + + if (ch < 0) { + printf("TNC %d fatal read error.\n", my_index); + exit (1); + } + + if (ch == '\r' || ch == '\n') { + done = 1; + } + else if (ch == XOFF) { + double dnow = dtime_now(); + printf("%*s[R %.3f] \n", my_index*column_width, "", dnow-start_dtime); + busy[my_index] = 1; + } + else if (ch == XON) { + double dnow = dtime_now(); + printf("%*s[R %.3f] \n", my_index*column_width, "", dnow-start_dtime); + busy[my_index] = 0; + } + else if (isprint(ch)) { + result[len] = ch; + len++; + result[len] = '\0'; + } + else { + char hex[12]; + + snprintf (hex, sizeof(hex), "", ch); + strlcat (result, hex, sizeof(result)); + len = strlen(result); + } + if (strcmp(result, "cmd:") == 0) { + done = 1; + have_cmd_prompt[my_index] = 1; + } + else { + have_cmd_prompt[my_index] = 0; + } + } + + if (len > 0) { + + double dnow = dtime_now(); + + printf("%*s[R %.3f] %s\n", my_index*column_width, "", dnow-start_dtime, result); + + if (strncmp(result, "*** CONNECTED", 13) == 0) { + is_connected[my_index] = 1; + } + + if (strncmp(result, "*** DISCONNECTED", 16) == 0) { + is_connected[my_index] = 0; + } + + if (strncmp(result, "Not while connected", 19) == 0) { + // Not expecting this. + // What to do? + } + + process_rec_data (my_index, result); + + if (isdigit(result[0]) && isdigit(result[1]) && isdigit(result[2]) && isdigit(result[3]) && + strncmp(result+4, " send", 5) == 0) { + // Expected message. Make sure it is expected sequence and send reply. + int n = atoi(result); + char reply[80]; + snprintf (reply, sizeof(reply), "%04d reply\r", n); + tnc_send_data (my_index, 1 - my_index, reply); + } + + } + } + +} /* end tnc_thread_serial */ + + + + +static void tnc_connect (int from, int to) +{ + + double dnow = dtime_now(); + + printf("%*s[T %.3f] *** Send connect request ***\n", from*column_width, "", dnow-start_dtime); + + if (using_tcp[from]) { + +//struct agwpe_s { +// short portx; /* 0 for first, 1 for second, etc. */ +// short port_hi_reserved; +// short datakind; /* message type */ +// short kind_hi; +// char call_from[10]; +// char call_to[10]; +// int data_len; /* Number of data bytes following. */ +// int user_reserved; +//}; + struct agwpe_s cmd; + + memset (&cmd, 0, sizeof(cmd)); + + cmd.datakind = 'C'; + strlcpy (cmd.call_from, tnc_address[from], sizeof(cmd.call_from)); + strlcpy (cmd.call_to, tnc_address[to], sizeof(cmd.call_to)); + + SOCK_SEND(server_sock[from], (char*)(&cmd), sizeof(cmd)); + } + else { + + char cmd[80]; + + if (! have_cmd_prompt[from]) { + + SLEEP_MS (1500); + strcpy (cmd, "\003\003\003"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + SLEEP_MS (1500); + + strcpy (cmd, "\r"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + SLEEP_MS (200); + } + + snprintf (cmd, sizeof(cmd), "connect %s\r", tnc_address[to]); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + } + +} /* end tnc_connect */ + + +static void tnc_disconnect (int from, int to) +{ + double dnow = dtime_now(); + + printf("%*s[T %.3f] *** Send disconnect request ***\n", from*column_width, "", dnow-start_dtime); + + if (using_tcp[from]) { + + struct agwpe_s cmd; + + memset (&cmd, 0, sizeof(cmd)); + + cmd.datakind = 'd'; + strlcpy (cmd.call_from, tnc_address[from], sizeof(cmd.call_from)); + strlcpy (cmd.call_to, tnc_address[to], sizeof(cmd.call_to)); + + SOCK_SEND(server_sock[from], (char*)(&cmd), sizeof(cmd)); + } + else { + + char cmd[80]; + + if (! have_cmd_prompt[from]) { + + SLEEP_MS (1500); + strcpy (cmd, "\003\003\003"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + SLEEP_MS (1500); + + strcpy (cmd, "\r"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + SLEEP_MS (200); + } + + strcpy (cmd, "disconnect\r"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + } + +} /* end tnc_disconnect */ + + +static void tnc_reset (int from, int to) +{ + double dnow = dtime_now(); + + printf("%*s[T %.3f] *** Send reset ***\n", from*column_width, "", dnow-start_dtime); + + if (using_tcp[from]) { + + + } + else { + + char cmd[80]; + + SLEEP_MS (1500); + strcpy (cmd, "\003\003\003"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + SLEEP_MS (1500); + + strcpy (cmd, "\r"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + SLEEP_MS (200); + + strcpy (cmd, "reset\r"); + serial_port_write (serial_fd[from], cmd, strlen(cmd)); + } + +} /* end tnc_disconnect */ + + + +static void tnc_send_data (int from, int to, char * data) +{ + double dnow = dtime_now(); + + printf("%*s[T %.3f] %s\n", from*column_width, "", dnow-start_dtime, data); + + if (using_tcp[from]) { + + struct { + struct agwpe_s hdr; + char data[256]; + } cmd; + + memset (&cmd.hdr, 0, sizeof(cmd.hdr)); + + cmd.hdr.datakind = 'D'; + cmd.hdr.pid = 0xf0; + snprintf (cmd.hdr.call_from, sizeof(cmd.hdr.call_from), "%s", tnc_address[from]); + snprintf (cmd.hdr.call_to, sizeof(cmd.hdr.call_to), "%s", tnc_address[to]); + cmd.hdr.data_len = strlen(data); + strlcpy (cmd.data, data, sizeof(cmd.data)); + + SOCK_SEND(server_sock[from], (char*)(&cmd), sizeof(cmd.hdr) + strlen(data)); + } + else { + + // The assumption is that we are in CONVERS mode. + // The data should be terminated by carriage return. + + int timeout = 600; // 60 sec. I've seen it take more than 20. + while (timeout > 0 && busy[from]) { + SLEEP_MS(100); + timeout--; + } + if (timeout == 0) { + printf ("ERROR: Gave up waiting while TNC busy.\n"); + tnc_disconnect (0,1); + SLEEP_MS(5000); + printf ("TEST FAILED!\n"); + exit (EXIT_FAILURE); + } + else { + serial_port_write (serial_fd[from], data, strlen(data)); + } + } + +} /* end tnc_disconnect */ + + + + +/* end tnctest.c */ diff --git a/tq.c b/src/tq.c similarity index 81% rename from tq.c rename to src/tq.c index fa787c73..c656af54 100644 --- a/tq.c +++ b/src/tq.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2012, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2012, 2014, 2015, 2016, 2023 John Langner, WB2OSZ // // 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 @@ -50,7 +50,8 @@ #include "audio.h" #include "tq.h" #include "dedupe.h" - +#include "igate.h" +#include "dtime_now.h" @@ -148,7 +149,7 @@ void tq_init (struct audio_s *audio_config_p) for (c = 0; c < MAX_CHANS; c++) { - if (audio_config_p->achan[c].valid) { + if (audio_config_p->chan_medium[c] == MEDIUM_RADIO) { wake_up_event[c] = CreateEvent (NULL, 0, 0, NULL); @@ -167,7 +168,7 @@ void tq_init (struct audio_s *audio_config_p) xmit_thread_is_waiting[c] = 0; - if (audio_config_p->achan[c].valid) { + if (audio_config_p->chan_medium[c] == MEDIUM_RADIO) { err = pthread_cond_init (&(wake_up_cond[c]), NULL); if (err != 0) { text_color_set(DW_COLOR_ERROR); @@ -195,6 +196,9 @@ void tq_init (struct audio_s *audio_config_p) * * Inputs: chan - Channel, 0 is first. * + * New in 1.7: + * Channel can be assigned to IGate rather than a radio. + * * prio - Priority, use TQ_PRIO_0_HI for digipeated or * TQ_PRIO_1_LO for normal. * @@ -247,11 +251,52 @@ void tq_append (int chan, int prio, packet_t pp) } #endif - if (chan < 0 || chan >= MAX_CHANS || ! save_audio_config_p->achan[chan].valid) { +// New in 1.7 - A channel can be assigned to the IGate rather than a radio. + +#ifndef DIGITEST // avoid dtest link error + + if (save_audio_config_p->chan_medium[chan] == MEDIUM_IGATE) { + + char ts[100]; // optional time stamp. + + if (strlen(save_audio_config_p->timestamp_format) > 0) { + char tstmp[100]; + timestamp_user_format (tstmp, sizeof(tstmp), save_audio_config_p->timestamp_format); + strlcpy (ts, " ", sizeof(ts)); // space after channel. + strlcat (ts, tstmp, sizeof(ts)); + } + else { + strlcpy (ts, "", sizeof(ts)); + } + + char stemp[256]; // Formated addresses. + ax25_format_addrs (pp, stemp); + unsigned char *pinfo; + int info_len = ax25_get_info (pp, &pinfo); + text_color_set(DW_COLOR_XMIT); + dw_printf ("[%d>is%s] ", chan, ts); + dw_printf ("%s", stemp); /* stations followed by : */ + ax25_safe_print ((char *)pinfo, info_len, ! ax25_is_aprs(pp)); + dw_printf ("\n"); + + igate_send_rec_packet (chan, pp); + ax25_delete(pp); + return; + } +#endif + +// Normal case - put in queue for radio transmission. +// Error if trying to transmit to a radio channel which was not configured. + + if (chan < 0 || chan >= MAX_CHANS || save_audio_config_p->chan_medium[chan] == MEDIUM_NONE) { text_color_set(DW_COLOR_ERROR); dw_printf ("ERROR - Request to transmit on invalid radio channel %d.\n", chan); dw_printf ("This is probably a client application error, not a problem with direwolf.\n"); - dw_printf ("AX.25 for Linux is known to transmit on channels 2 & 8 sometimes when it shouldn't.\n"); + dw_printf ("Are you using AX.25 for Linux? It might be trying to use a modified\n"); + dw_printf ("version of KISS which uses the port field differently than the\n"); + dw_printf ("original KISS protocol specification. The solution might be to use\n"); + dw_printf ("a command like \"kissparms -c 1 -p radio\" to set CRC none mode.\n"); + dw_printf ("\n"); ax25_delete(pp); return; } @@ -277,11 +322,9 @@ void tq_append (int chan, int prio, packet_t pp) * The check would allow an unlimited number of other types. * * Limit was 20. Changed to 100 in version 1.2 as a workaround. - * - * Implementing the 6PACK protocol is probably the proper solution. */ - if (ax25_is_aprs(pp) && tq_count(chan,prio) > 100) { + if (ax25_is_aprs(pp) && tq_count(chan,prio,"","",0) > 100) { text_color_set(DW_COLOR_ERROR); dw_printf ("Transmit packet queue for channel %d is too long. Discarding packet.\n", chan); dw_printf ("Perhaps the channel is so busy there is no opportunity to send.\n"); @@ -447,9 +490,13 @@ void lm_data_request (int chan, int prio, packet_t pp) } #endif - if (chan < 0 || chan >= MAX_CHANS || ! save_audio_config_p->achan[chan].valid) { + if (chan < 0 || chan >= MAX_CHANS || save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { + // Connected mode is allowed only with internal modems. text_color_set(DW_COLOR_ERROR); dw_printf ("ERROR - Request to transmit on invalid radio channel %d.\n", chan); + dw_printf ("Connected packet mode is allowed only with internal modems.\n"); + dw_printf ("Why aren't external KISS modems allowed? See\n"); + dw_printf ("Why-is-9600-only-twice-as-fast-as-1200.pdf for explanation.\n"); ax25_delete(pp); return; } @@ -458,7 +505,7 @@ void lm_data_request (int chan, int prio, packet_t pp) * Is transmit queue out of control? */ - if (tq_count(chan,prio) > 250) { + if (tq_count(chan,prio,"","",0) > 250) { text_color_set(DW_COLOR_ERROR); dw_printf ("Warning: Transmit packet queue for channel %d is extremely long.\n", chan); dw_printf ("Perhaps the channel is so busy there is no opportunity to send.\n"); @@ -600,9 +647,14 @@ void lm_seize_request (int chan) dw_printf ("lm_seize_request (chan=%d)\n", chan); #endif - if (chan < 0 || chan >= MAX_CHANS || ! save_audio_config_p->achan[chan].valid) { + + if (chan < 0 || chan >= MAX_CHANS || save_audio_config_p->chan_medium[chan] != MEDIUM_RADIO) { + // Connected mode is allowed only with internal modems. text_color_set(DW_COLOR_ERROR); dw_printf ("ERROR - Request to transmit on invalid radio channel %d.\n", chan); + dw_printf ("Connected packet mode is allowed only with internal modems.\n"); + dw_printf ("Why aren't external KISS modems allowed? See\n"); + dw_printf ("Why-is-9600-only-twice-as-fast-as-1200.pdf for explanation.\n"); return; } @@ -912,36 +964,108 @@ static int tq_is_empty (int chan) * * Name: tq_count * - * Purpose: Return count of the number of packets in the specified transmit queue. + * Purpose: Return count of the number of packets (or bytes) in the specified transmit queue. + * This is used only for queries from KISS or AWG client applications. * * Inputs: chan - Channel, 0 is first. * * prio - Priority, use TQ_PRIO_0_HI or TQ_PRIO_1_LO. + * Specify -1 for total of both. + * + * source - If specified, count only those with this source address. + * + * dest - If specified, count only those with this destination address. + * + * bytes - If true, return number of bytes rather than packets. * * Returns: Number of items in specified queue. * *--------------------------------------------------------------------*/ -int tq_count (int chan, int prio) +//#define DEBUG2 1 + + +int tq_count (int chan, int prio, char *source, char *dest, int bytes) { - packet_t p; - int n; +#if DEBUG2 + text_color_set(DW_COLOR_DEBUG); + dw_printf ("tq_count(chan=%d, prio=%d, source=\"%s\", dest=\"%s\", bytes=%d)\n", chan, prio, source, dest, bytes); +#endif + + if (prio == -1) { + return (tq_count(chan, TQ_PRIO_0_HI, source, dest, bytes) + + tq_count(chan, TQ_PRIO_1_LO, source, dest, bytes)); + } + + // Array bounds check. FIXME: TODO: should have internal error instead of dying. -/* Don't bother with critical section. */ -/* Only used for debugging a problem. */ + if (chan < 0 || chan >= MAX_CHANS || prio < 0 || prio >= TQ_NUM_PRIO) { + text_color_set(DW_COLOR_DEBUG); + dw_printf ("INTERNAL ERROR - tq_count(%d, %d, \"%s\", \"%s\", %d)\n", chan, prio, source, dest, bytes); + return (0); + } - n = 0; - p = queue_head[chan][prio]; - while (p != NULL) { - n++; - p = ax25_get_nextp(p); + if (queue_head[chan][prio] == 0) { +#if DEBUG2 + text_color_set(DW_COLOR_DEBUG); + dw_printf ("tq_count: queue chan %d, prio %d is empty, returning 0.\n", chan, prio); +#endif + return (0); } -#if DEBUG + // Don't want lists being rearranged while we are traversing them. + + dw_mutex_lock (&tq_mutex); + + int n = 0; // Result. Number of bytes or packets. + packet_t pp = queue_head[chan][prio];; + + while (pp != NULL) { + if (ax25_get_num_addr(pp) >= AX25_MIN_ADDRS) { + // Consider only real packets. + + int count_it = 1; + + if (source != NULL && *source != '\0') { + char frame_source[AX25_MAX_ADDR_LEN]; + ax25_get_addr_with_ssid (pp, AX25_SOURCE, frame_source); +#if DEBUG2 + // I'm cringing at the thought of printing while in a critical region. But it's only for temp debug. :-( + text_color_set(DW_COLOR_DEBUG); + dw_printf ("tq_count: compare to frame source %s\n", frame_source); +#endif + if (strcmp(source,frame_source) != 0) count_it = 0; + } + if (count_it && dest != NULL && *dest != '\0') { + char frame_dest[AX25_MAX_ADDR_LEN]; + ax25_get_addr_with_ssid (pp, AX25_DESTINATION, frame_dest); +#if DEBUG2 + // I'm cringing at the thought of printing while in a critical region. But it's only for debug debug. :-( + text_color_set(DW_COLOR_DEBUG); + dw_printf ("tq_count: compare to frame destination %s\n", frame_dest); +#endif + if (strcmp(dest,frame_dest) != 0) count_it = 0; + } + + if (count_it) { + if (bytes) { + n += ax25_get_frame_len(pp); + } + else { + n++; + } + } + } + pp = ax25_get_nextp(pp); + } + + dw_mutex_unlock (&tq_mutex); + +#if DEBUG2 text_color_set(DW_COLOR_DEBUG); - dw_printf ("tq_count(%d,%d) returns %d\n", chan, prio, n); + dw_printf ("tq_count(%d, %d, \"%s\", \"%s\", %d) returns %d\n", chan, prio, source, dest, bytes, n); #endif return (n); diff --git a/tq.h b/src/tq.h similarity index 91% rename from tq.h rename to src/tq.h index 4b5beb81..37599d59 100644 --- a/tq.h +++ b/src/tq.h @@ -34,7 +34,7 @@ packet_t tq_remove (int chan, int prio); packet_t tq_peek (int chan, int prio); -int tq_count (int chan, int prio); +int tq_count (int chan, int prio, char *source, char *dest, int bytes); #endif diff --git a/tt_text.c b/src/tt_text.c similarity index 99% rename from tt_text.c rename to src/tt_text.c index 8cd7ba80..112adfe5 100644 --- a/tt_text.c +++ b/src/tt_text.c @@ -382,7 +382,7 @@ int tt_text_to_two_key (const char *text, int quiet, char *buttons) * Outputs: buttons - Sequence of two buttons to press. * "00" for error because this is probably * being used to build up a fixed length - * string where positions are signficant. + * string where positions are significant. * Must be at least 3 bytes. * * Returns: Number of errors detected. @@ -468,9 +468,9 @@ int tt_text_to_call10 (const char *text, int quiet, char *buttons) int errors = 0; int found; char padded[8]; - char stemp[8]; - + char stemp[11]; + // FIXME: Add parameter for sizeof buttons and use strlcpy strcpy (buttons, ""); /* Quick validity check. */ @@ -540,6 +540,7 @@ int tt_text_to_call10 (const char *text, int quiet, char *buttons) /* Binary to decimal for the columns. */ snprintf (stemp, sizeof(stemp), "%04d", packed); + // FIXME: add parameter for sizeof buttons and use strlcat strcat (buttons, stemp); return (errors); @@ -1435,6 +1436,7 @@ int tt_satsq_to_text (const char *buttons, int quiet, char *text) row = buttons[0] - '0'; col = buttons[1] - '0'; + // FIXME: Add parameter for sizeof text and use strlcpy, strlcat. strcpy (text, grid[row][col]); strcat (text, buttons+2); @@ -1603,11 +1605,11 @@ int main (int argc, char *argv[]) exit (1); } - strcpy (text, argv[1]); + strlcpy (text, argv[1], sizeof(text)); for (n = 2; n < argc; n++) { - strcat (text, " "); - strcat (text, argv[n]); + strlcat (text, " ", sizeof(text)); + strlcat (text, argv[n], sizeof(text)); } dw_printf ("Push buttons for multi-press method:\n"); @@ -1670,7 +1672,7 @@ int main (int argc, char *argv[]) exit (1); } - strcpy (buttons, argv[1]); + strlcpy (buttons, argv[1], sizeof(buttons)); for (n = 2; n < argc; n++) { strlcat (buttons, argv[n], sizeof(buttons)); diff --git a/tt_text.h b/src/tt_text.h similarity index 100% rename from tt_text.h rename to src/tt_text.h diff --git a/tt_user.c b/src/tt_user.c similarity index 97% rename from tt_user.c rename to src/tt_user.c index f6ce1673..a73d6a46 100644 --- a/tt_user.c +++ b/src/tt_user.c @@ -58,8 +58,9 @@ #include "server.h" #include "kiss.h" +#include "kissserial.h" #include "kissnet.h" - +#include "kiss_frame.h" /* * Information kept about local APRStt users. @@ -229,7 +230,7 @@ void tt_user_init (struct audio_s *p_audio_config, struct tt_config_s *p_tt_conf * Inputs: callsign - full or a old style 3 DIGIT suffix abbreviation * overlay * - * Returns: Handle for refering to table position or -1 if not found. + * Returns: Handle for referring to table position or -1 if not found. * This happens to be an index into an array but * the implementation could change so the caller should * not make any assumptions. @@ -289,7 +290,7 @@ int tt_user_search (char *callsign, char overlay) * * Outputs: callsign - corresponding full callsign or empty string. * - * Returns: Handle for refering to table position (>= 0) or -1 if not found. + * Returns: Handle for referring to table position (>= 0) or -1 if not found. * This happens to be an index into an array but * the implementation could change so the caller should * not make any assumptions. @@ -350,7 +351,7 @@ static void clear_user(int i) * * Inputs: none * - * Returns: Handle for refering to table position. + * Returns: Handle for referring to table position. * * Description: If table is already full, this should delete the * least recently heard user to make room. @@ -452,7 +453,7 @@ static void digit_suffix (char *callsign, char *suffix) * * Name: tt_user_heard * - * Purpose: Record information from an APRStt trasmission. + * Purpose: Record information from an APRStt transmission. * * Inputs: callsign - full or an abbreviation * ssid @@ -880,8 +881,8 @@ static void xmit_object_report (int i, int first_time) * Any attached application(s). * IGate. * - * When transmitting over the radio, it gets sent multipe times, to help - * probablity of being heard, with increasing delays between. + * When transmitting over the radio, it gets sent multiple times, to help + * probability of being heard, with increasing delays between. * * The other methods are reliable so we only want to send it once. */ @@ -891,12 +892,14 @@ static void xmit_object_report (int i, int first_time) int flen; // TODO1.3: Put a wrapper around this so we only call one function to send by all methods. + // We see the same sequence in direwolf.c. flen = ax25_pack(pp, fbuf); server_send_rec_packet (save_tt_config_p->obj_recv_chan, pp, fbuf, flen); - kissnet_send_rec_packet (save_tt_config_p->obj_recv_chan, fbuf, flen); - kiss_send_rec_packet (save_tt_config_p->obj_recv_chan, fbuf, flen); + kissnet_send_rec_packet (save_tt_config_p->obj_recv_chan, KISS_CMD_DATA_FRAME, fbuf, flen, NULL, -1); + kissserial_send_rec_packet (save_tt_config_p->obj_recv_chan, KISS_CMD_DATA_FRAME, fbuf, flen, NULL, -1); + kisspt_send_rec_packet (save_tt_config_p->obj_recv_chan, KISS_CMD_DATA_FRAME, fbuf, flen, NULL, -1); } if (first_time && save_tt_config_p->obj_send_to_ig) { diff --git a/tt_user.h b/src/tt_user.h similarity index 100% rename from tt_user.h rename to src/tt_user.h diff --git a/ttcalc.c b/src/ttcalc.c similarity index 94% rename from ttcalc.c rename to src/ttcalc.c index ccebf893..a031ec0a 100644 --- a/ttcalc.c +++ b/src/ttcalc.c @@ -1,4 +1,3 @@ - // // This file is part of Dire Wolf, an amateur radio packet TNC. // @@ -60,7 +59,7 @@ #include #include #include -#include +#include #endif #include @@ -108,11 +107,10 @@ int main (int argc, char *argv[]) char data[1024]; char hostname[30] = "localhost"; char port[10] = "8000"; + int err; #if __WIN32__ #else - int err; - setlinebuf (stdout); #endif @@ -142,13 +140,8 @@ int main (int argc, char *argv[]) mon_cmd.kind_lo = 'k'; -#if __WIN32__ - send (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd), 0); -#else - err = write (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); + err = SOCK_SEND (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); (void)err; -#endif - /* * Print all of the monitored packets. @@ -157,11 +150,7 @@ int main (int argc, char *argv[]) while (1) { int n; -#if __WIN32__ - n = recv (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd), 0); -#else - n = read (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); -#endif + n = SOCK_RECV (server_sock, (char*)(&mon_cmd), sizeof(mon_cmd)); if (n != sizeof(mon_cmd)) { printf ("Read error, received %d command bytes.\n", n); @@ -171,14 +160,10 @@ int main (int argc, char *argv[]) assert (mon_cmd.data_len >= 0 && mon_cmd.data_len < (int)(sizeof(data))); if (mon_cmd.data_len > 0) { -#if __WIN32__ - n = recv (server_sock, data, mon_cmd.data_len, 0); -#else - n = read (server_sock, data, mon_cmd.data_len); -#endif + n = SOCK_RECV (server_sock, data, mon_cmd.data_len); if (n != mon_cmd.data_len) { - printf ("Read error, client received %d data bytes when %d expected.\n", n, mon_cmd.data_len); + printf ("Read error, client received %d data bytes when %d expected. Terminating.\n", n, mon_cmd.data_len); exit (1); } } @@ -202,7 +187,7 @@ int main (int argc, char *argv[]) ax25_format_addrs (pp, result); info_len = ax25_get_info (pp, (unsigned char **)(&pinfo)); pinfo[info_len] = '\0'; - strcat (result, pinfo); + strlcat (result, pinfo, sizeof(result)); for (p=result; *p!='\0'; p++) { if (! isprint(*p)) *p = ' '; } @@ -254,11 +239,7 @@ int main (int argc, char *argv[]) xmit_raw.hdr.kind_lo = 'K'; xmit_raw.hdr.data_len = 1 + ax25_pack (reply_pp, xmit_raw.frame); -#if __WIN32__ - send (server_sock, (char*)(&xmit_raw), sizeof(xmit_raw.hdr)+xmit_raw.hdr.data_len, 0); -#else - err = write (server_sock, (char*)(&xmit_raw), sizeof(xmit_raw.hdr)+xmit_raw.hdr.data_len); -#endif + err = SOCK_SEND (server_sock, (char*)(&xmit_raw), sizeof(xmit_raw.hdr)+xmit_raw.hdr.data_len); ax25_delete (reply_pp); } diff --git a/tune.h b/src/tune.h similarity index 100% rename from tune.h rename to src/tune.h diff --git a/utm2ll.c b/src/utm2ll.c similarity index 100% rename from utm2ll.c rename to src/utm2ll.c diff --git a/src/version.h b/src/version.h new file mode 100644 index 00000000..a09490cc --- /dev/null +++ b/src/version.h @@ -0,0 +1,21 @@ + +/* Dire Wolf version 1.6 */ + +// Put in destination field to identify the equipment used. + +#define APP_TOCALL "APDW" // Assigned by WB4APR in tocalls.txt + +// This now comes from compile command line options. + +//#define MAJOR_VERSION 1 +//#define MINOR_VERSION 6 +//#define EXTRA_VERSION "Beta Test" + + +// For user-defined data format. +// APRS protocol spec Chapter 18 and http://www.aprs.org/aprs11/expfmts.txt + +#define USER_DEF_USER_ID 'D' // user id D for direwolf + +#define USER_DEF_TYPE_AIS 'A' // data type A for AIS NMEA sentence +#define USER_DEF_TYPE_EAS 'E' // data type E for EAS broadcasts diff --git a/walk96.c b/src/walk96.c similarity index 97% rename from walk96.c rename to src/walk96.c index dfad612f..9fc791f8 100644 --- a/walk96.c +++ b/src/walk96.c @@ -86,7 +86,7 @@ int main (int argc, char *argv[]) // USB GPS happens to be COM22 memset (&config, 0, sizeof(config)); - strlcpy (config.gpsnmea_port, "COM22", sizeof(config.nmea_port)); + strlcpy (config.gpsnmea_port, "COM22", sizeof(config.gpsnmea_port)); dwgps_init (&config, debug_gps); @@ -152,7 +152,7 @@ static void walk96 (int fix, double lat, double lon, float knots, float course, info_len = encode_position (messaging, compressed, - lat, lon, (int)(DW_METERS_TO_FEET(alt)), + lat, lon, 0, (int)(DW_METERS_TO_FEET(alt)), '/', '=', G_UNKNOWN, G_UNKNOWN, G_UNKNOWN, "", // PHGd (int)roundf(course), (int)roundf(knots), diff --git a/waypoint.c b/src/waypoint.c similarity index 76% rename from waypoint.c rename to src/waypoint.c index 1ec77124..20c1cdbc 100644 --- a/waypoint.c +++ b/src/waypoint.c @@ -1,7 +1,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2014, 2015, 2016, 2020 John Langner, WB2OSZ // // 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 @@ -32,16 +32,21 @@ #include "direwolf.h" // should be first - #include #include +#include #if __WIN32__ -#include +#include +#include // _WIN32_WINNT must be set to 0x0501 before including this #else -#include #include -#include +#include +#include +#include +#include +//#include +#include // gethostbyname #endif #include @@ -57,13 +62,16 @@ #include "mgn_icon.h" /* Magellan icons */ #include "dwgpsnmea.h" #include "serial_port.h" +#include "dwsock.h" -static MYFDTYPE s_waypoint_port_fd = MYFDERROR; +static MYFDTYPE s_waypoint_serial_port_fd = MYFDERROR; +static int s_waypoint_udp_sock_fd = -1; // ideally INVALID_SOCKET for Windows. +static struct sockaddr_in s_udp_dest_addr; static int s_waypoint_formats = 0; /* which formats should we generate? */ -static int s_waypoint_debug = 0; /* Print information flowing to attached device. */ +static int s_waypoint_debug = 0; /* Print information flowing to attached device. */ @@ -86,19 +94,24 @@ void waypoint_set_debug (int n) * * Inputs: mc - Pointer to configuration options. * - * ->waypoint_port - Name of serial port. COM1, /dev/ttyS0, etc. + * ->waypoint_serial_port - Name of serial port. COM1, /dev/ttyS0, etc. * + * ->waypoint_udp_hostname - Destination host when using UDP. + * + * ->waypoint_udp_portnum - UDP port number. * * (currently none) - speed, baud. Default 4800 if not set * * * ->waypoint_formats - Set of formats enabled. - * If none set, default to generic & Kenwood. + * If none set, default to generic & Kenwood here. * - * Global output: s_waypoint_port_fd + * Global output: s_waypoint_serial_port_fd + * s_waypoint_udp_sock_fd * * Description: First to see if this is shared with GPS input. * If not, open serial port. + * In version 1.6 UDP is added. It is possible to use both. * * Restriction: MUST be done after GPS init because we might be sharing the * same serial port device. @@ -111,46 +124,79 @@ void waypoint_init (struct misc_config_s *mc) #if DEBUG text_color_set (DW_COLOR_DEBUG); - dw_printf ("waypoint_init() device=%s formats=%d\n", mc->waypoint_port, mc->waypoint_formats); + dw_printf ("waypoint_init() serial device=%s formats=%02x\n", mc->waypoint_serial_port, mc->waypoint_formats); + dw_printf ("waypoint_init() destination hostname=%s UDP port=%d\n", mc->waypoint_udp_hostname, mc->waypoint_udp_portnum); #endif - + + s_waypoint_udp_sock_fd = -1; + + if (mc->waypoint_udp_portnum > 0) { + + s_waypoint_udp_sock_fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); + if (s_waypoint_udp_sock_fd != -1) { + + // Not thread-safe. Should use getaddrinfo instead. + struct hostent *hp = gethostbyname(mc->waypoint_udp_hostname); + + if (hp != NULL) { + memset ((char *)&s_udp_dest_addr, 0, sizeof(s_udp_dest_addr)); + s_udp_dest_addr.sin_family = AF_INET; + memcpy ((char *)&s_udp_dest_addr.sin_addr, (char *)hp->h_addr, hp->h_length); + s_udp_dest_addr.sin_port = htons(mc->waypoint_udp_portnum); + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Waypoint: Couldn't get address for %s\n", mc->waypoint_udp_hostname); + close (s_waypoint_udp_sock_fd); + s_waypoint_udp_sock_fd = -1; + } + } + else { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Couldn't create socket for waypoint send to %s\n", mc->waypoint_udp_hostname); + } + } + /* * TODO: * Are we sharing with GPS input? * First try to get fd if they have same device name. * If that fails, do own serial port open. */ - if (strlen(mc->waypoint_port) > 0) { + s_waypoint_serial_port_fd = MYFDERROR; - s_waypoint_port_fd = dwgpsnmea_get_fd (mc->waypoint_port, 4800); + if (strlen(mc->waypoint_serial_port) > 0) { - if (s_waypoint_port_fd == MYFDERROR) { - s_waypoint_port_fd = serial_port_open (mc->waypoint_port, 4800); + s_waypoint_serial_port_fd = dwgpsnmea_get_fd (mc->waypoint_serial_port, 4800); + + if (s_waypoint_serial_port_fd == MYFDERROR) { + s_waypoint_serial_port_fd = serial_port_open (mc->waypoint_serial_port, 4800); } else { text_color_set (DW_COLOR_INFO); dw_printf ("Note: Sharing same port for GPS input and waypoint output.\n"); } - if (s_waypoint_port_fd == MYFDERROR) { + if (s_waypoint_serial_port_fd == MYFDERROR) { text_color_set (DW_COLOR_ERROR); - dw_printf ("Unable to open %s for waypoint output.\n", mc->waypoint_port); - return; - } - - s_waypoint_formats = mc->waypoint_formats; - if (s_waypoint_formats == 0) { - s_waypoint_formats = WPT_FORMAT_NMEA_GENERIC | WPT_FORMAT_KENWOOD; - } - if (s_waypoint_formats & WPT_FORMAT_GARMIN) { - s_waypoint_formats |= WPT_FORMAT_NMEA_GENERIC; /* See explanation below. */ + dw_printf ("Unable to open serial port %s for waypoint output.\n", mc->waypoint_serial_port); } } +// Set default formats if user did not specify any. + + s_waypoint_formats = mc->waypoint_formats; + if (s_waypoint_formats == 0) { + s_waypoint_formats = WPL_FORMAT_NMEA_GENERIC | WPL_FORMAT_KENWOOD; + } + if (s_waypoint_formats & WPL_FORMAT_GARMIN) { + s_waypoint_formats |= WPL_FORMAT_NMEA_GENERIC; /* See explanation below. */ + } #if DEBUG text_color_set (DW_COLOR_DEBUG); - dw_printf ("end of waypoint_init: s_waypoint_port_fd = %d\n", s_waypoint_port_fd); + dw_printf ("end of waypoint_init: s_waypoint_serial_port_fd = %d\n", s_waypoint_serial_port_fd); + dw_printf ("end of waypoint_init: s_waypoint_udp_sock_fd = %d\n", s_waypoint_udp_sock_fd); #endif } @@ -252,6 +298,12 @@ void waypoint_send_sentence (char *name_in, double dlat, double dlong, char symt dw_printf ("waypoint_send_sentence (\"%s\", \"%c%c\")\n", name_in, symtab, symbol); #endif +// Don't waste time if no destinations specified. + + if (s_waypoint_serial_port_fd == MYFDERROR && + s_waypoint_udp_sock_fd == -1) { + return; + } /* * We need to remove any , or * from name, symbol, or comment because they are field delimiters. @@ -324,7 +376,7 @@ void waypoint_send_sentence (char *name_in, double dlat, double dlong, char symt * *99 is checksum */ - if (s_waypoint_formats & WPT_FORMAT_NMEA_GENERIC) { + if (s_waypoint_formats & WPL_FORMAT_NMEA_GENERIC) { snprintf (sentence, sizeof(sentence), "$GPWPL,%s,%s,%s,%s,%s", slat, slat_ns, slong, slong_ew, wname); append_checksum (sentence); @@ -354,7 +406,7 @@ void waypoint_send_sentence (char *name_in, double dlat, double dlong, char symt * *99 is checksum */ - if (s_waypoint_formats & WPT_FORMAT_GARMIN) { + if (s_waypoint_formats & WPL_FORMAT_GARMIN) { int i = symbol - ' '; int grm_sym; /* Garmin symbol code. */ @@ -403,7 +455,7 @@ void waypoint_send_sentence (char *name_in, double dlat, double dlong, char symt * to delete that specific waypoint. */ - if (s_waypoint_formats & WPT_FORMAT_MAGELLAN) { + if (s_waypoint_formats & WPL_FORMAT_MAGELLAN) { int i = symbol - ' '; char sicon[3]; /* Magellan icon string. Currently 1 or 2 characters. */ @@ -515,7 +567,7 @@ void waypoint_send_sentence (char *name_in, double dlat, double dlong, char symt - if (s_waypoint_formats & WPT_FORMAT_KENWOOD) { + if (s_waypoint_formats & WPL_FORMAT_KENWOOD) { time_t now; struct tm tm; @@ -583,29 +635,59 @@ void waypoint_send_sentence (char *name_in, double dlat, double dlong, char symt } /* end waypoint_send_sentence */ +/*------------------------------------------------------------------- + * + * Name: nema_send_ais + * + * Purpose: Send NMEA AIS sentence to GPS display or other mapping application. + * + * Inputs: sentence - should look something like this, with checksum, and no CR LF. + * + * !AIVDM,1,1,,A,35NO=dPOiAJriVDH@94E84AJ0000,0*4B + * + *--------------------------------------------------------------------*/ + +void waypoint_send_ais (char *sentence) +{ + if (s_waypoint_serial_port_fd == MYFDERROR && + s_waypoint_udp_sock_fd == -1) { + return; + } + + if (s_waypoint_formats & WPL_FORMAT_AIS) { + send_sentence (sentence); + } +} + + /* * Append CR LF and send it. */ static void send_sentence (char *sent) { - char final[256]; - - if (s_waypoint_port_fd == MYFDERROR) { - return; - } - if (s_waypoint_debug) { text_color_set(DW_COLOR_XMIT); - dw_printf ("%s\n", sent); + dw_printf ("waypoint send sentence: \"%s\"\n", sent); } strlcpy (final, sent, sizeof(final)); strlcat (final, "\r\n", sizeof(final)); + int final_len = strlen(final); - serial_port_write (s_waypoint_port_fd, final, strlen(final)); + if (s_waypoint_serial_port_fd != MYFDERROR) { + serial_port_write (s_waypoint_serial_port_fd, final, final_len); + } + + if (s_waypoint_udp_sock_fd != -1) { + int n = sendto(s_waypoint_udp_sock_fd, final, final_len, 0, (struct sockaddr*)(&s_udp_dest_addr), sizeof(struct sockaddr_in)); + if (n != final_len) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Failed to send waypoint via UDP, errno=%d\n", errno); + } + } } /* send_sentence */ @@ -613,10 +695,13 @@ static void send_sentence (char *sent) void waypoint_term (void) { - - if (s_waypoint_port_fd != MYFDERROR) { + if (s_waypoint_serial_port_fd != MYFDERROR) { //serial_port_close (s_waypoint_port_fd); - s_waypoint_port_fd = MYFDERROR; + s_waypoint_serial_port_fd = MYFDERROR; + } + if (s_waypoint_udp_sock_fd != -1) { + close (s_waypoint_udp_sock_fd); + s_waypoint_udp_sock_fd = -1; } } diff --git a/waypoint.h b/src/waypoint.h similarity index 91% rename from waypoint.h rename to src/waypoint.h index 0f5ef3af..3ba6f1c8 100644 --- a/waypoint.h +++ b/src/waypoint.h @@ -16,6 +16,8 @@ void waypoint_set_debug (int n); void waypoint_send_sentence (char *wname_in, double dlat, double dlong, char symtab, char symbol, float alt, float course, float speed, char *comment_in); +void waypoint_send_ais (char *sentence); + void waypoint_term (); diff --git a/xid.c b/src/xid.c similarity index 76% rename from xid.c rename to src/xid.c index 90ce4378..14e67e8d 100644 --- a/xid.c +++ b/src/xid.c @@ -2,7 +2,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2014, 2016 John Langner, WB2OSZ +// Copyright (C) 2014, 2016, 2017 John Langner, WB2OSZ // // 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 @@ -50,6 +50,7 @@ #include #include #include +#include #include "textcolor.h" #include "xid.h" @@ -66,10 +67,36 @@ #define PI_Ack_Timer 9 #define PI_Retries 10 +// Forget about the bit order at the physical layer (e.g. HDLC). +// It doesn't matter at all here. We are dealing with bytes. +// A different encoding could send the bits in the opposite order. + +// The bit numbers are confusing because this one table (Fig. 4.5) starts +// with 1 for the LSB when everywhere else refers to the LSB as bit 0. + +// The first byte must be of the form 0xx0 0001 +// The second byte must be of the form 0000 0000 +// If we process the two byte "Classes of Procedures" like +// the other multibyte numeric fields, with the more significant +// byte first, we end up with the bit masks below. +// The bit order would be 8 7 6 5 4 3 2 1 16 15 14 13 12 11 10 9 + +// (This has nothing to do with the HDLC serializing order. +// I'm talking about the way we would normally write binary numbers.) + #define PV_Classes_Procedures_Balanced_ABM 0x0100 #define PV_Classes_Procedures_Half_Duplex 0x2000 #define PV_Classes_Procedures_Full_Duplex 0x4000 + +// The first byte must be of the form 1000 0xx0 +// The second byte must be of the form 1010 xx00 +// The third byte must be of the form 0000 0010 +// If we process the three byte "HDLC Optional Parameters" like +// the other multibyte numeric fields, with the most significant +// byte first, we end up with bit masks like this. +// The bit order would be 8 7 6 5 4 3 2 1 16 15 14 13 12 11 10 9 24 23 22 21 20 19 18 17 + #define PV_HDLC_Optional_Functions_REJ_cmd_resp 0x020000 #define PV_HDLC_Optional_Functions_SREJ_cmd_resp 0x040000 #define PV_HDLC_Optional_Functions_Extended_Address 0x800000 @@ -79,6 +106,9 @@ #define PV_HDLC_Optional_Functions_TEST_cmd_resp 0x002000 #define PV_HDLC_Optional_Functions_16_bit_FCS 0x008000 +#define PV_HDLC_Optional_Functions_Multi_SREJ_cmd_resp 0x000020 +#define PV_HDLC_Optional_Functions_Segmenter 0x000040 + #define PV_HDLC_Optional_Functions_Synchronous_Tx 0x000002 @@ -127,7 +157,7 @@ int xid_parse (unsigned char *info, int info_len, struct xid_param_s *result, ch // rej and modulo are enum so we can't use G_UNKNOWN there. result->full_duplex = G_UNKNOWN; - result->rej = unknown_reject; + result->srej = srej_not_specified; result->modulo = modulo_unknown; result->i_field_length_rx = G_UNKNOWN; result->window_size_rx = G_UNKNOWN; @@ -204,21 +234,31 @@ int xid_parse (unsigned char *info, int info_len, struct xid_param_s *result, ch case PI_HDLC_Optional_Functions: - if ((pval & PV_HDLC_Optional_Functions_REJ_cmd_resp) && (pval & PV_HDLC_Optional_Functions_SREJ_cmd_resp)) { - result->rej = selective_reject_reject; /* Both bits set */ - strlcat (desc, "SREJ-REJ ", desc_size); - } - else if ((pval & PV_HDLC_Optional_Functions_REJ_cmd_resp) && ! (pval & PV_HDLC_Optional_Functions_SREJ_cmd_resp)) { - result->rej = implicit_reject; /* Only REJ is set */ + // Pick highest of those offered. + + if (pval & PV_HDLC_Optional_Functions_REJ_cmd_resp) { strlcat (desc, "REJ ", desc_size); } - else if ( ! (pval & PV_HDLC_Optional_Functions_REJ_cmd_resp) && pval & PV_HDLC_Optional_Functions_SREJ_cmd_resp) { - result->rej = selective_reject; /* Only SREJ is set */ + if (pval & PV_HDLC_Optional_Functions_SREJ_cmd_resp) { strlcat (desc, "SREJ ", desc_size); } + if (pval & PV_HDLC_Optional_Functions_Multi_SREJ_cmd_resp) { + strlcat (desc, "Multi-SREJ ", desc_size); + } + + if (pval & PV_HDLC_Optional_Functions_Multi_SREJ_cmd_resp) { + result->srej = srej_multi; + } + else if (pval & PV_HDLC_Optional_Functions_SREJ_cmd_resp) { + result->srej = srej_single; + } + else if (pval & PV_HDLC_Optional_Functions_REJ_cmd_resp) { + result->srej = srej_none; + } else { text_color_set (DW_COLOR_ERROR); - dw_printf ("XID error: Expected one or both of REJ, SREJ to be set.\n"); + dw_printf ("XID error: Expected at least one of REJ, SREJ, Multi-SREJ to be set.\n"); + result->srej = srej_none; } if ((pval & PV_HDLC_Optional_Functions_Modulo_8) && ! (pval & PV_HDLC_Optional_Functions_Modulo_128)) { @@ -331,10 +371,10 @@ int xid_parse (unsigned char *info, int info_len, struct xid_param_s *result, ch * 0 = half duplex. * 1 = full duplex. * - * rej - One of: implicit_reject, selective_reject, selective_reject_reject. - * As command, what am I capable of processing? - * As response, take minimum of - * + * srej - Level of selective reject. + * srej_none (use REJ), srej_single, srej_multi + * As command, offer a menu of what I can handle. (i.e. perhaps multiple bits set) + * As response, take minimum of what is offered and what I can handle. (one bit set) * * modulo - 8 or 128. * @@ -355,6 +395,8 @@ int xid_parse (unsigned char *info, int info_len, struct xid_param_s *result, ch * Default is 10. * Use G_UNKNOWN to omit this. * + * cr - Is it a command or response? + * * Outputs: info - Information part of XID frame. * Does not include the control byte. * Use buffer of 40 bytes just to be safe. @@ -382,10 +424,17 @@ int xid_parse (unsigned char *info, int info_len, struct xid_param_s *result, ch * Comment: I have a problem with "... occurs at any time." What if we were in the middle * of transferring a large file with k=32 then along comes XID which says switch to modulo 8? * + * Insight: Or is it Erratum? + * After reading the base standards documents, it seems that the XID command should offer + * up a menu of all the acceptable choices. e.g. REJ, SREJ, Multi-SREJ. One or more bits + * can be set. The XID response, would set a single bit which is the desired choice from + * among those offered. + * Should go back and review half/full duplex and modulo. + * *--------------------------------------------------------------------*/ -int xid_encode (struct xid_param_s *param, unsigned char *info) +int xid_encode (struct xid_param_s *param, unsigned char *info, cmdres_t cr) { unsigned char *p; int len; @@ -438,11 +487,43 @@ int xid_encode (struct xid_param_s *param, unsigned char *info) PV_HDLC_Optional_Functions_16_bit_FCS | PV_HDLC_Optional_Functions_Synchronous_Tx; - if (param->rej == implicit_reject || param->rej == selective_reject_reject || param->rej == unknown_reject) - x |= PV_HDLC_Optional_Functions_REJ_cmd_resp; + //text_color_set (DW_COLOR_ERROR); + //dw_printf ("****** XID temp hack - test no SREJ ******\n"); + // param->srej = srej_none; - if (param->rej == selective_reject || param->rej == selective_reject_reject) - x |= PV_HDLC_Optional_Functions_SREJ_cmd_resp; + if (cr == cr_cmd) { + // offer a "menu" of acceptable choices. i.e. 1, 2 or 3 bits set. + switch (param->srej) { + case srej_none: + default: + x |= PV_HDLC_Optional_Functions_REJ_cmd_resp; + break; + case srej_single: + x |= PV_HDLC_Optional_Functions_REJ_cmd_resp | + PV_HDLC_Optional_Functions_SREJ_cmd_resp; + break; + case srej_multi: + x |= PV_HDLC_Optional_Functions_REJ_cmd_resp | + PV_HDLC_Optional_Functions_SREJ_cmd_resp | + PV_HDLC_Optional_Functions_Multi_SREJ_cmd_resp; + break; + } + } + else { + // for response, set only a single bit. + switch (param->srej) { + case srej_none: + default: + x |= PV_HDLC_Optional_Functions_REJ_cmd_resp; + break; + case srej_single: + x |= PV_HDLC_Optional_Functions_SREJ_cmd_resp; + break; + case srej_multi: + x |= PV_HDLC_Optional_Functions_Multi_SREJ_cmd_resp; + break; + } + } if (param->modulo == modulo_128) x |= PV_HDLC_Optional_Functions_Modulo_128; @@ -531,7 +612,7 @@ static unsigned char example[27] = { /* GL */ 0x17, /* (2 bytes) */ /* PI */ 0x02, /* Parameter Indicator - classes of procedures */ /* PL */ 0x02, /* Parameter Length */ -#if 0 // Example in the protocol spec looks wrong. +#if 0 // Erratum: Example in the protocol spec looks wrong. /* PV */ 0x00, /* Parameter Variable - Half Duplex, Async, Balanced Mode */ /* PV */ 0x20, /* */ #else // I think it should be like this instead. @@ -545,6 +626,11 @@ static unsigned char example[27] = { /* PV */ 0x02, /* synchronous transmit */ /* PI */ 0x06, /* Parameter Indicator - Rx I field length (bits) */ /* PL */ 0x02, /* Parameter Length */ + +// Erratum: The text does not say anything about the byte order for multibyte +// numeric values. In the example, we have two cases where 16 bit numbers are +// sent with the more significant byte first. + /* PV */ 0x04, /* Parameter Variable - 1024 bits (128 octets) */ /* PV */ 0x00, /* */ /* PI */ 0x08, /* Parameter Indicator - Rx window size */ @@ -565,7 +651,7 @@ int main (int argc, char *argv[]) { struct xid_param_s param2; int n; unsigned char info[40]; // Currently max of 27 but things can change. - char desc[100]; // 80 is not adequate. + char desc[150]; // I've seen 109. /* parse example. */ @@ -573,13 +659,18 @@ int main (int argc, char *argv[]) { n = xid_parse (example, sizeof(example), ¶m, desc, sizeof(desc)); text_color_set (DW_COLOR_DEBUG); - dw_printf ("%s\n", desc); + dw_printf ("%d: %s\n", __LINE__, desc); + fflush (stdout); + SLEEP_SEC (1); text_color_set (DW_COLOR_ERROR); +#ifdef NDEBUG +#error "This won't work properly if NDEBUG is defined. It should be undefined in direwolf.h" +#endif assert (n==1); assert (param.full_duplex == 0); - assert (param.rej == selective_reject_reject); + assert (param.srej == srej_single); assert (param.modulo == modulo_128); assert (param.i_field_length_rx == 128); assert (param.window_size_rx == 2); @@ -588,7 +679,7 @@ int main (int argc, char *argv[]) { /* encode and verify it comes out the same. */ - n = xid_encode (¶m, info); + n = xid_encode (¶m, info, cr_cmd); assert (n == sizeof(example)); n = memcmp(info, example, 27); @@ -599,52 +690,56 @@ int main (int argc, char *argv[]) { assert (n == 0); -/* try a couple different values. */ +/* try a couple different values, no srej. */ param.full_duplex = 1; - param.rej = implicit_reject; + param.srej = srej_none; param.modulo = modulo_8; param.i_field_length_rx = 2048; param.window_size_rx = 3; param.ack_timer = 1234; param.retries = 12; - n = xid_encode (¶m, info); + n = xid_encode (¶m, info, cr_cmd); n = xid_parse (info, n, ¶m2, desc, sizeof(desc)); text_color_set (DW_COLOR_DEBUG); - dw_printf ("%s\n", desc); + dw_printf ("%d: %s\n", __LINE__, desc); + fflush (stdout); + SLEEP_SEC (1); text_color_set (DW_COLOR_ERROR); assert (param2.full_duplex == 1); - assert (param2.rej == implicit_reject); + assert (param2.srej == srej_none); assert (param2.modulo == modulo_8); assert (param2.i_field_length_rx == 2048); assert (param2.window_size_rx == 3); assert (param2.ack_timer == 1234); assert (param2.retries == 12); -/* The third possbility for rej. We don't use this. */ +/* Other values, single srej. */ param.full_duplex = 0; - param.rej = selective_reject; + param.srej = srej_single; param.modulo = modulo_8; param.i_field_length_rx = 61; param.window_size_rx = 4; param.ack_timer = 5555; param.retries = 9; - n = xid_encode (¶m, info); + n = xid_encode (¶m, info, cr_cmd); n = xid_parse (info, n, ¶m2, desc, sizeof(desc)); text_color_set (DW_COLOR_DEBUG); - dw_printf ("%s\n", desc); + dw_printf ("%d: %s\n", __LINE__, desc); + fflush (stdout); + SLEEP_SEC (1); text_color_set (DW_COLOR_ERROR); assert (param2.full_duplex == 0); - assert (param2.rej == selective_reject); + assert (param2.srej == srej_single); assert (param2.modulo == modulo_8); assert (param2.i_field_length_rx == 61); assert (param2.window_size_rx == 4); @@ -652,26 +747,57 @@ int main (int argc, char *argv[]) { assert (param2.retries == 9); +/* Other values, multi srej. */ + + param.full_duplex = 0; + param.srej = srej_multi; + param.modulo = modulo_128; + param.i_field_length_rx = 61; + param.window_size_rx = 4; + param.ack_timer = 5555; + param.retries = 9; + + n = xid_encode (¶m, info, cr_cmd); + n = xid_parse (info, n, ¶m2, desc, sizeof(desc)); + + text_color_set (DW_COLOR_DEBUG); + dw_printf ("%d: %s\n", __LINE__, desc); + fflush (stdout); + SLEEP_SEC (1); + + text_color_set (DW_COLOR_ERROR); + + assert (param2.full_duplex == 0); + assert (param2.srej == srej_multi); + assert (param2.modulo == modulo_128); + assert (param2.i_field_length_rx == 61); + assert (param2.window_size_rx == 4); + assert (param2.ack_timer == 5555); + assert (param2.retries == 9); + + /* Specify some and not others. */ param.full_duplex = 0; - param.rej = selective_reject; + param.srej = srej_single; param.modulo = modulo_8; param.i_field_length_rx = G_UNKNOWN; param.window_size_rx = G_UNKNOWN; param.ack_timer = 999; param.retries = G_UNKNOWN; - n = xid_encode (¶m, info); + n = xid_encode (¶m, info, cr_cmd); n = xid_parse (info, n, ¶m2, desc, sizeof(desc)); text_color_set (DW_COLOR_DEBUG); - dw_printf ("%s\n", desc); + dw_printf ("%d: %s\n", __LINE__, desc); + fflush (stdout); + SLEEP_SEC (1); text_color_set (DW_COLOR_ERROR); assert (param2.full_duplex == 0); - assert (param2.rej == selective_reject); + assert (param2.srej == srej_single); assert (param2.modulo == modulo_8); assert (param2.i_field_length_rx == G_UNKNOWN); assert (param2.window_size_rx == G_UNKNOWN); @@ -684,12 +810,14 @@ int main (int argc, char *argv[]) { n = xid_parse (info, n, ¶m2, desc, sizeof(desc)); text_color_set (DW_COLOR_DEBUG); - dw_printf ("%s\n", desc); + dw_printf ("%d: %s\n", __LINE__, desc); + fflush (stdout); + SLEEP_SEC (1); text_color_set (DW_COLOR_ERROR); assert (param2.full_duplex == G_UNKNOWN); - assert (param2.rej == unknown_reject); + assert (param2.srej == srej_not_specified); assert (param2.modulo == modulo_unknown); assert (param2.i_field_length_rx == G_UNKNOWN); assert (param2.window_size_rx == G_UNKNOWN); diff --git a/xid.h b/src/xid.h similarity index 64% rename from xid.h rename to src/xid.h index c94003e7..a221b737 100644 --- a/xid.h +++ b/src/xid.h @@ -2,6 +2,7 @@ /* xid.h */ + #include "ax25_pad.h" // for enum ax25_modulo_e @@ -9,10 +10,10 @@ struct xid_param_s { int full_duplex; - // Order is important because negotiation keeps the lower value. - // We will support only 1 & 2. + // Order is important because negotiation keeps the lower value of + // REJ (srej_none), SREJ (default without negotiation), Multi-SREJ (if both agree). - enum rej_e {unknown_reject=0, implicit_reject=1, selective_reject=2, selective_reject_reject=3 } rej; + enum srej_e { srej_none=0, srej_single=1, srej_multi=2, srej_not_specified=3 } srej; enum ax25_modulo_e modulo; @@ -28,4 +29,4 @@ struct xid_param_s { int xid_parse (unsigned char *info, int info_len, struct xid_param_s *result, char *desc, int desc_size); -int xid_encode (struct xid_param_s *param, unsigned char *info); \ No newline at end of file +int xid_encode (struct xid_param_s *param, unsigned char *info, cmdres_t cr); \ No newline at end of file diff --git a/xmit.c b/src/xmit.c similarity index 81% rename from xmit.c rename to src/xmit.c index ee82ee18..13bbaecb 100644 --- a/xmit.c +++ b/src/xmit.c @@ -2,7 +2,7 @@ // // This file is part of Dire Wolf, an amateur radio packet TNC. // -// Copyright (C) 2011, 2013, 2014, 2015, 2016 John Langner, WB2OSZ +// Copyright (C) 2011, 2013, 2014, 2015, 2016, 2017 John Langner, WB2OSZ // // 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 @@ -60,6 +60,7 @@ #include #include #include +#include #include "direwolf.h" #include "ax25_pad.h" @@ -74,7 +75,8 @@ #include "morse.h" #include "dtmf.h" #include "xid.h" - +#include "dlq.h" +#include "server.h" /* @@ -86,7 +88,7 @@ */ -static int xmit_slottime[MAX_CHANS]; /* Slot time in 10 mS units for persistance algorithm. */ +static int xmit_slottime[MAX_CHANS]; /* Slot time in 10 mS units for persistence algorithm. */ static int xmit_persist[MAX_CHANS]; /* Sets probability for transmitting after each */ /* slot time delay. Transmit if a random number */ @@ -101,9 +103,11 @@ static int xmit_txtail[MAX_CHANS]; /* Amount of time to keep transmitting after /* dropping PTT too soon and chopping off the end */ /* of the frame. Again 10 mS units. */ +static int xmit_fulldup[MAX_CHANS]; /* Full duplex if non-zero. */ + static int xmit_bits_per_sec[MAX_CHANS]; /* Data transmission rate. */ - /* Often called baud rate which is equivalent in */ - /* this case but could be different with other */ + /* Often called baud rate which is equivalent for */ + /* 1200 & 9600 cases but could be different with other */ /* modulation techniques. */ static int g_debug_xmit_packet; /* print packet in hexadecimal form for debugging. */ @@ -112,11 +116,42 @@ static int g_debug_xmit_packet; /* print packet in hexadecimal form for debuggi // TODO: When this was first written, bits/sec was same as baud. // Need to revisit this for PSK modes where they are not the same. +#if 0 // Added during 1.5 beta test + +static int BITS_TO_MS (int b, int ch) { + + int bits_per_symbol; + + switch (save_audio_config_p->achan[ch].modem_type) { + case MODEM_QPSK: bits_per_symbol = 2; break; + case MODEM_8PSK: bits_per_symbol = 3; break; + case default: bits_per_symbol = 1; break; + } + + return ( (b * 1000) / (xmit_bits_per_sec[(ch)] * bits_per_symbol) ); +} + +static int MS_TO_BITS (int ms, int ch) { + + int bits_per_symbol; + + switch (save_audio_config_p->achan[ch].modem_type) { + case MODEM_QPSK: bits_per_symbol = 2; break; + case MODEM_8PSK: bits_per_symbol = 3; break; + case default: bits_per_symbol = 1; break; + } + + return ( (ms * xmit_bits_per_sec[(ch)] * bits_per_symbol) / 1000 ); TODO... +} + +#else // OK for 1200, 9600 but wrong for PSK #define BITS_TO_MS(b,ch) (((b)*1000)/xmit_bits_per_sec[(ch)]) #define MS_TO_BITS(ms,ch) (((ms)*xmit_bits_per_sec[(ch)])/1000) +#endif + #define MAXX(a,b) (((a)>(b)) ? (a) : (b)) @@ -137,7 +172,7 @@ static dw_mutex_t audio_out_dev_mutex[MAX_ADEVS]; -static int wait_for_clear_channel (int channel, int slotttime, int persist); +static int wait_for_clear_channel (int channel, int slotttime, int persist, int fulldup); static void xmit_ax25_frames (int c, int p, packet_t pp, int max_bundle); static int send_one_frame (int c, int p, packet_t pp); static void xmit_speech (int c, packet_t pp); @@ -218,6 +253,7 @@ void xmit_init (struct audio_s *p_modem, int debug_xmit_packet) xmit_persist[j] = p_modem->achan[j].persist; xmit_txdelay[j] = p_modem->achan[j].txdelay; xmit_txtail[j] = p_modem->achan[j].txtail; + xmit_fulldup[j] = p_modem->achan[j].fulldup; } #if DEBUG @@ -242,10 +278,9 @@ void xmit_init (struct audio_s *p_modem, int debug_xmit_packet) for (j=0; jachan[j].valid) { - + if (p_modem->chan_medium[j] == MEDIUM_RADIO) { #if __WIN32__ - xmit_th[j] = (HANDLE)_beginthreadex (NULL, 0, xmit_thread, (void*)(long)j, 0, NULL); + xmit_th[j] = (HANDLE)_beginthreadex (NULL, 0, xmit_thread, (void*)(ptrdiff_t)j, 0, NULL); if (xmit_th[j] == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf ("Could not create xmit thread %d\n", j); @@ -276,10 +311,10 @@ void xmit_init (struct audio_s *p_modem, int debug_xmit_packet) perror("pthread_attr_setschedparam"); } - e = pthread_create (&(xmit_tid[j]), &attr, xmit_thread, (void *)(long)j); + e = pthread_create (&(xmit_tid[j]), &attr, xmit_thread, (void *)(ptrdiff_t)j); pthread_attr_destroy (&attr); #else - e = pthread_create (&(xmit_tid[j]), NULL, xmit_thread, (void *)(long)j); + e = pthread_create (&(xmit_tid[j]), NULL, xmit_thread, (void *)(ptrdiff_t)j); #endif if (e != 0) { text_color_set(DW_COLOR_ERROR); @@ -306,6 +341,7 @@ void xmit_init (struct audio_s *p_modem, int debug_xmit_packet) * xmit_set_persist * xmit_set_slottime * xmit_set_txtail + * xmit_set_fulldup * * * Purpose: The KISS protocol, and maybe others, can specify @@ -355,6 +391,13 @@ void xmit_set_txtail (int channel, int value) } } +void xmit_set_fulldup (int channel, int value) +{ + if (channel >= 0 && channel < MAX_CHANS) { + xmit_fulldup[channel] = value; + } +} + /*------------------------------------------------------------------- * @@ -468,7 +511,7 @@ static unsigned __stdcall xmit_thread (void *arg) static void * xmit_thread (void *arg) #endif { - int chan = (int)(long)arg; // channel number. + int chan = (int)(ptrdiff_t)arg; // channel number. packet_t pp; int prio; int ok; @@ -479,7 +522,7 @@ static void * xmit_thread (void *arg) tq_wait_while_empty (chan); #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread, channel %d: woke up\n", c); + dw_printf ("xmit_thread, channel %d: woke up\n", chan); #endif // Does this extra loop offer any benefit? @@ -490,7 +533,7 @@ static void * xmit_thread (void *arg) * If there is something in the high priority queue, begin transmitting immediately. * Otherwise, wait a random amount of time, in hopes of minimizing collisions. */ - ok = wait_for_clear_channel (chan, xmit_slottime[chan], xmit_persist[chan]); + ok = wait_for_clear_channel (chan, xmit_slottime[chan], xmit_persist[chan], xmit_fulldup[chan]); prio = TQ_PRIO_1_LO; pp = tq_remove (chan, TQ_PRIO_0_HI); @@ -554,6 +597,11 @@ static void * xmit_thread (void *arg) case FLAVOR_APRS_DIGI: xmit_ax25_frames (chan, prio, pp, 1); /* 1 means don't bundle */ + // I don't know if this in some official specification + // somewhere, but it is generally agreed that APRS digipeaters + // should send only one frame at a time rather than + // bundling multiple frames into a single transmission. + // Discussion here: http://lists.tapr.org/pipermail/aprssig_lists.tapr.org/2021-September/049034.html break; case FLAVOR_APRS_NEW: @@ -670,6 +718,8 @@ static void * xmit_thread (void *arg) * Once we have control of the channel, we might as well keep going. * [High] Priority frames will always go to head of the line, * + * Version 1.5: Add full duplex option. + * *--------------------------------------------------------------------*/ @@ -706,17 +756,38 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: Turn on PTT now for channel %d. speed = %d\n", chan, xmit_bits_per_sec[chan]); + dw_printf ("xmit_thread: t=%.3f, Turn on PTT now for channel %d. speed = %d\n", dtime_now()-time_ptt, chan, xmit_bits_per_sec[chan]); #endif ptt_set (OCTYPE_PTT, chan, 1); +// Inform data link state machine that we are now transmitting. + + dlq_seize_confirm (chan); // C4.2. "This primitive indicates, to the Data-link State + // machine, that the transmission opportunity has arrived." + pre_flags = MS_TO_BITS(xmit_txdelay[chan] * 10, chan) / 8; - num_bits = hdlc_send_flags (chan, pre_flags, 0); + num_bits = layer2_preamble_postamble (chan, pre_flags, 0, save_audio_config_p); #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: txdelay=%d [*10], pre_flags=%d, num_bits=%d\n", xmit_txdelay[chan], pre_flags, num_bits); + dw_printf ("xmit_thread: t=%.3f, txdelay=%d [*10], pre_flags=%d, num_bits=%d\n", dtime_now()-time_ptt, xmit_txdelay[chan], pre_flags, num_bits); + double presleep = dtime_now(); #endif + SLEEP_MS (10); // Give data link state machine a chance to + // to stuff more frames into the transmit queue, + // in response to dlq_seize_confirm, so + // we don't run off the end too soon. + +#if DEBUG + text_color_set(DW_COLOR_DEBUG); + // How long did sleep last? + dw_printf ("xmit_thread: t=%.3f, Should be 0.010 second after the above.\n", dtime_now()-time_ptt); + double naptime = dtime_now() - presleep; + if (naptime > 0.015) { + text_color_set(DW_COLOR_ERROR); + dw_printf ("Sleep for 10 ms actually took %.3f second!\n", naptime); + } +#endif /* * Transmit the frame. @@ -728,7 +799,7 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) if (nb > 0) numframe++; #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: flen=%d, nb=%d, num_bits=%d, numframe=%d\n", flen, nb, num_bits, numframe); + dw_printf ("xmit_thread: t=%.3f, nb=%d, num_bits=%d, numframe=%d\n", dtime_now()-time_ptt, nb, num_bits, numframe); #endif ax25_delete (pp); @@ -770,7 +841,7 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) pp = tq_remove (chan, prio); #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: tq_remove(chan=%d, prio=%d) returned %p\n", chan, prio, pp); + dw_printf ("xmit_thread: t=%.3f, tq_remove(chan=%d, prio=%d) returned %p\n", dtime_now()-time_ptt, chan, prio, pp); #endif nb = send_one_frame (chan, prio, pp); @@ -779,7 +850,7 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) if (nb > 0) numframe++; #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: flen=%d, nb=%d, num_bits=%d, numframe=%d\n", flen, nb, num_bits, numframe); + dw_printf ("xmit_thread: t=%.3f, nb=%d, num_bits=%d, numframe=%d\n", dtime_now()-time_ptt, nb, num_bits, numframe); #endif ax25_delete (pp); @@ -796,11 +867,11 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) */ post_flags = MS_TO_BITS(xmit_txtail[chan] * 10, chan) / 8; - nb = hdlc_send_flags (chan, post_flags, 1); + nb = layer2_preamble_postamble (chan, post_flags, 1, save_audio_config_p); num_bits += nb; #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: txtail=%d [*10], post_flags=%d, nb=%d, num_bits=%d\n", xmit_txtail[chan], post_flags, nb, num_bits); + dw_printf ("xmit_thread: t=%.3f, txtail=%d [*10], post_flags=%d, nb=%d, num_bits=%d\n", dtime_now()-time_ptt, xmit_txtail[chan], post_flags, nb, num_bits); #endif @@ -833,7 +904,7 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) #if DEBUG text_color_set(DW_COLOR_DEBUG); - dw_printf ("xmit_thread: xmit duration=%d, %d already elapsed since PTT, wait %d more\n", duration, already, wait_more ); + dw_printf ("xmit_thread: t=%.3f, xmit duration=%d, %d already elapsed since PTT, wait %d more\n", dtime_now()-time_ptt, duration, already, wait_more ); #endif if (wait_more > 0) { @@ -859,7 +930,7 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) #if DEBUG text_color_set(DW_COLOR_DEBUG); time_now = dtime_now(); - dw_printf ("xmit_thread: Turn off PTT now. Actual time on was %d mS, vs. %d desired\n", (int) ((time_now - time_ptt) * 1000.), duration); + dw_printf ("xmit_thread: t=%.3f, Turn off PTT now. Actual time on was %d mS, vs. %d desired\n", dtime_now()-time_ptt, (int) ((time_now - time_ptt) * 1000.), duration); #endif ptt_set (OCTYPE_PTT, chan, 0); @@ -891,8 +962,6 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle) static int send_one_frame (int c, int p, packet_t pp) { - unsigned char fbuf[AX25_MAX_PACKET_LEN+2]; - int flen; char stemp[1024]; /* max size needed? */ int info_len; unsigned char *pinfo; @@ -900,13 +969,50 @@ static int send_one_frame (int c, int p, packet_t pp) if (ax25_is_null_frame(pp)) { + + // Issue 132 - We could end up in a situation where: + // Transmitter is already on. + // Application wants to send a frame. + // dl_seize_request turns into this null frame. + // It was being ignored here so the data got stuck in the queue. + // I think the solution is to send back a seize confirm here. + // It shouldn't hurt if we send it redundantly. + // Added for 1.5 beta test 4. + + dlq_seize_confirm (c); // C4.2. "This primitive indicates, to the Data-link State + // machine, that the transmission opportunity has arrived." + + SLEEP_MS (10); // Give data link state machine a chance to + // to stuff more frames into the transmit queue, + // in response to dlq_seize_confirm, so + // we don't run off the end too soon. + return(0); } + char ts[100]; // optional time stamp. + + if (strlen(save_audio_config_p->timestamp_format) > 0) { + char tstmp[100]; + timestamp_user_format (tstmp, sizeof(tstmp), save_audio_config_p->timestamp_format); + strlcpy (ts, " ", sizeof(ts)); // space after channel. + strlcat (ts, tstmp, sizeof(ts)); + } + else { + strlcpy (ts, "", sizeof(ts)); + } + ax25_format_addrs (pp, stemp); info_len = ax25_get_info (pp, &pinfo); text_color_set(DW_COLOR_XMIT); - dw_printf ("[%d%c] ", c, p==TQ_PRIO_0_HI ? 'H' : 'L'); +#if 0 // FIXME - enable this? + dw_printf ("[%d%c%s%s] ", c, + p==TQ_PRIO_0_HI ? 'H' : 'L', + save_audio_config_p->achan[c].fx25_strength ? "F" : "", + ts); +#else + dw_printf ("[%d%c%s] ", c, p==TQ_PRIO_0_HI ? 'H' : 'L', ts); +#endif dw_printf ("%s", stemp); /* stations followed by : */ /* Demystify non-APRS. Use same format for received frames in direwolf.c. */ @@ -925,7 +1031,7 @@ static int send_one_frame (int c, int p, packet_t pp) if (ftype == frame_type_U_XID) { struct xid_param_s param; - char info2text[100]; + char info2text[150]; xid_parse (pinfo, info_len, ¶m, info2text, sizeof(info2text)); dw_printf (" %s\n", info2text); @@ -956,9 +1062,6 @@ static int send_one_frame (int c, int p, packet_t pp) /* * Transmit the frame. */ - flen = ax25_pack (pp, fbuf); - assert (flen >= 1 && flen <= (int)(sizeof(fbuf))); - int send_invalid_fcs2 = 0; if (save_audio_config_p->xmit_error_rate != 0) { @@ -971,7 +1074,12 @@ static int send_one_frame (int c, int p, packet_t pp) } } - nb = hdlc_send_frame (c, fbuf, flen, send_invalid_fcs2); + nb = layer2_send_frame (c, pp, send_invalid_fcs2, save_audio_config_p); + +// Optionally send confirmation to AGW client app if monitoring enabled. + + server_send_monitored (c, pp, 1); + return (nb); } /* end send_one_frame */ @@ -1008,11 +1116,23 @@ static void xmit_speech (int c, packet_t pp) * Print spoken packet. Prefix by channel. */ + char ts[100]; // optional time stamp. + + if (strlen(save_audio_config_p->timestamp_format) > 0) { + char tstmp[100]; + timestamp_user_format (tstmp, sizeof(tstmp), save_audio_config_p->timestamp_format); + strlcpy (ts, " ", sizeof(ts)); // space after channel. + strlcat (ts, tstmp, sizeof(ts)); + } + else { + strlcpy (ts, "", sizeof(ts)); + } + info_len = ax25_get_info (pp, &pinfo); (void)info_len; text_color_set(DW_COLOR_XMIT); - dw_printf ("[%d.speech] \"%s\"\n", c, pinfo); + dw_printf ("[%d.speech%s] \"%s\"\n", c, ts, pinfo); if (strlen(save_audio_config_p->tts_script) == 0) { @@ -1049,9 +1169,9 @@ static void xmit_speech (int c, packet_t pp) int xmit_speak_it (char *script, int c, char *orig_msg) { int err; - char cmd[2000]; - char *p; char msg[2000]; + char cmd[sizeof(msg) + 16]; + char *p; /* Remove any quotes because it will mess up command line argument parsing. */ @@ -1123,11 +1243,22 @@ static void xmit_morse (int c, packet_t pp, int wpm) int length_ms, wait_ms; double start_ptt, wait_until, now; + char ts[100]; // optional time stamp. + + if (strlen(save_audio_config_p->timestamp_format) > 0) { + char tstmp[100]; + timestamp_user_format (tstmp, sizeof(tstmp), save_audio_config_p->timestamp_format); + strlcpy (ts, " ", sizeof(ts)); // space after channel. + strlcat (ts, tstmp, sizeof(ts)); + } + else { + strlcpy (ts, "", sizeof(ts)); + } info_len = ax25_get_info (pp, &pinfo); (void)info_len; text_color_set(DW_COLOR_XMIT); - dw_printf ("[%d.morse] \"%s\"\n", c, pinfo); + dw_printf ("[%d.morse%s] \"%s\"\n", c, ts, pinfo); ptt_set (OCTYPE_PTT, c, 1); start_ptt = dtime_now(); @@ -1184,11 +1315,22 @@ static void xmit_dtmf (int c, packet_t pp, int speed) int length_ms, wait_ms; double start_ptt, wait_until, now; + char ts[100]; // optional time stamp. + + if (strlen(save_audio_config_p->timestamp_format) > 0) { + char tstmp[100]; + timestamp_user_format (tstmp, sizeof(tstmp), save_audio_config_p->timestamp_format); + strlcpy (ts, " ", sizeof(ts)); // space after channel. + strlcat (ts, tstmp, sizeof(ts)); + } + else { + strlcpy (ts, "", sizeof(ts)); + } info_len = ax25_get_info (pp, &pinfo); (void)info_len; text_color_set(DW_COLOR_XMIT); - dw_printf ("[%d.dtmf] \"%s\"\n", c, pinfo); + dw_printf ("[%d.dtmf%s] \"%s\"\n", c, ts, pinfo); ptt_set (OCTYPE_PTT, c, 1); start_ptt = dtime_now(); @@ -1231,12 +1373,19 @@ static void xmit_dtmf (int c, packet_t pp, int speed) * slottime - Amount of time to wait for each iteration * of the waiting algorithm. 10 mSec units. * - * persist - Probability of transmitting + * persist - Probability of transmitting. + * + * fulldup - Full duplex. Just start sending immediately. * * Returns: 1 for OK. 0 for timeout. * * Description: New in version 1.2: also obtain a lock on audio out device. * + * New in version 1.5: full duplex. + * Just start transmitting rather than waiting for clear channel. + * This would only be appropriate when transmit and receive are + * using different radio frequencies. e.g. VHF up, UHF down satellite. + * * Transmit delay algorithm: * * Wait for channel to be clear. @@ -1271,10 +1420,17 @@ static void xmit_dtmf (int c, packet_t pp, int speed) #define WAIT_TIMEOUT_MS (60 * 1000) #define WAIT_CHECK_EVERY_MS 10 -static int wait_for_clear_channel (int chan, int slottime, int persist) +static int wait_for_clear_channel (int chan, int slottime, int persist, int fulldup) { int n = 0; +/* + * For dull duplex we skip the channel busy check and random wait. + * We still need to wait if operating in stereo and the other audio + * half is busy. + */ + if ( ! fulldup) { + start_over_again: while (hdlc_rec_data_detect_any(chan)) { @@ -1318,6 +1474,7 @@ static int wait_for_clear_channel (int chan, int slottime, int persist) break; } } + } /* * This is to prevent two channels from transmitting at the same time diff --git a/xmit.h b/src/xmit.h similarity index 89% rename from xmit.h rename to src/xmit.h index b444f1e7..248037df 100644 --- a/xmit.h +++ b/src/xmit.h @@ -16,6 +16,8 @@ extern void xmit_set_slottime (int channel, int value); extern void xmit_set_txtail (int channel, int value); +extern void xmit_set_fulldup (int channel, int value); + extern int xmit_speak_it (char *script, int c, char *msg); diff --git a/systemd/direwolf.logrotate b/systemd/direwolf.logrotate new file mode 100644 index 00000000..143b6083 --- /dev/null +++ b/systemd/direwolf.logrotate @@ -0,0 +1,20 @@ +/var/log/direwolf/stdout /var/log/direwolf/stderr { + missingok + rotate 30 + daily + copytruncate + notifempty + compress + delaycompress + dateext + dateyesterday + } + +/var/log/direwolf/*.log { + missingok + daily + rotate 30 + minage 7 + maxage 30 + compress +} diff --git a/systemd/direwolf.service b/systemd/direwolf.service new file mode 100644 index 00000000..c3380fac --- /dev/null +++ b/systemd/direwolf.service @@ -0,0 +1,27 @@ +[Unit] +Description=Direwolf Sound Card-based AX.25 TNC +After=sound.target +After=network.target + +[Service] +EnvironmentFile=/etc/sysconfig/direwolf +User=direwolf +# You may want to set the audio levels of your radio-connected soundcard +# prior to starting direwolf. To do so, copy this file to /etc/systemd/system/ +# and edit the ExecStartPre line to point to your preferred method of +# doing so. Then run systemctl daemon-reload so systemd uses your updated +# copy of this service file. +#ExecStartPre=/some/script.sh +ExecStart=/bin/bash -ce "exec /usr/bin/direwolf $DIREWOLF_ARGS >>/var/log/direwolf/stdout 2>>/var/log/direwolf/stderr" +Restart=always +StandardOutput=null +StandardError=null +ProtectSystem=strict +ProtectHome=true +ReadWritePaths=/var/log/direwolf + +[Install] +WantedBy=multi-user.target +DefaultInstance=1 + +# alternate version: https://www.f4fxl.org/start-direwolf-at-boot-the-systemd-way/ diff --git a/systemd/direwolf.sysconfig b/systemd/direwolf.sysconfig new file mode 100644 index 00000000..748dfe77 --- /dev/null +++ b/systemd/direwolf.sysconfig @@ -0,0 +1,2 @@ +# Set direwolf command line arguments here +DIREWOLF_ARGS="-l /var/log/direwolf -c /etc/direwolf.conf" diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt new file mode 100644 index 00000000..91e06a2c --- /dev/null +++ b/test/CMakeLists.txt @@ -0,0 +1,714 @@ +# This is a trick to avoid more complication +# because configure_file() is done at configuration time + +set(CUSTOM_TEST_BINARY_DIR "${CMAKE_BINARY_DIR}/test") +set(GEN_PACKETS_BIN "${CMAKE_BINARY_DIR}/src/gen_packets${CMAKE_EXECUTABLE_SUFFIX}") +set(ATEST_BIN "${CMAKE_BINARY_DIR}/src/atest${CMAKE_EXECUTABLE_SUFFIX}") +set(FXSEND_BIN "${CMAKE_BINARY_DIR}/test/fxsend${CMAKE_EXECUTABLE_SUFFIX}") +set(FXREC_BIN "${CMAKE_BINARY_DIR}/test/fxrec${CMAKE_EXECUTABLE_SUFFIX}") +set(IL2P_TEST_BIN "${CMAKE_BINARY_DIR}/test/il2p_test${CMAKE_EXECUTABLE_SUFFIX}") + +if(WIN32) + set(CUSTOM_SCRIPT_SUFFIX ".bat") +else() + set(CUSTOM_SCRIPT_SUFFIX "") +endif() + +set(TEST_CHECK-FX25_FILE "check-fx25") +set(TEST_CHECK-IL2P_FILE "check-il2p") +set(TEST_CHECK-MODEM1200_FILE "check-modem1200") +set(TEST_CHECK-MODEM1200_IL2P_FILE "check-modem1200-i") +set(TEST_CHECK-MODEM300_FILE "check-modem300") +set(TEST_CHECK-MODEM9600_FILE "check-modem9600") +set(TEST_CHECK-MODEM9600_IL2P_FILE "check-modem9600-i") +set(TEST_CHECK-MODEM19200_FILE "check-modem19200") +set(TEST_CHECK-MODEM2400-a_FILE "check-modem2400-a") +set(TEST_CHECK-MODEM2400-b_FILE "check-modem2400-b") +set(TEST_CHECK-MODEM2400-g_FILE "check-modem2400-g") +set(TEST_CHECK-MODEM4800_FILE "check-modem4800") +set(TEST_CHECK-MODEMEAS_FILE "check-modemeas") + +# generate the scripts that run the tests + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-FX25_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-FX25_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-IL2P_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-IL2P_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM1200_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM1200_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM1200_IL2P_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM1200_IL2P_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM300_FILE}" + "${CUSTOM_TEST_BINARY_DIR}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM9600_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM9600_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM9600_IL2P_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM9600_IL2P_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM19200_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM19200_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM2400-a_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM2400-a_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM2400-b_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM2400-b_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM2400-g_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM2400-g_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEM4800_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM4800_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + +configure_file( + "${CUSTOM_TEST_SCRIPTS_DIR}/${TEST_CHECK-MODEMEAS_FILE}" + "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEMEAS_FILE}${CUSTOM_SCRIPT_SUFFIX}" + @ONLY + ) + + +# global includes +# not ideal but not so slow +# otherwise use target_include_directories +include_directories( + ${CUSTOM_SRC_DIR} + ${GPSD_INCLUDE_DIRS} + ${HAMLIB_INCLUDE_DIRS} + ${ALSA_INCLUDE_DIRS} + ${UDEV_INCLUDE_DIRS} + ${PORTAUDIO_INCLUDE_DIRS} + ${CUSTOM_GEOTRANZ_DIR} + ${CMAKE_BINARY_DIR}/src + ) + +if(WIN32 OR CYGWIN) + include_directories( + ${CUSTOM_REGEX_DIR} + ) +endif() + +if(WIN32 OR CYGWIN) + list(REMOVE_ITEM atest9_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) +endif() + + +# Unit test for inner digipeater algorithm +list(APPEND dtest_SOURCES + ${CUSTOM_SRC_DIR}/digipeater.c + ${CUSTOM_SRC_DIR}/ais.c + ${CUSTOM_SRC_DIR}/dedupe.c + ${CUSTOM_SRC_DIR}/pfilter.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/tq.c + ${CUSTOM_SRC_DIR}/textcolor.c + ${CUSTOM_SRC_DIR}/decode_aprs.c + ${CUSTOM_SRC_DIR}/dwgpsnmea.c + ${CUSTOM_SRC_DIR}/dwgps.c + ${CUSTOM_SRC_DIR}/dwgpsd.c + ${CUSTOM_SRC_DIR}/serial_port.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/telemetry.c + ${CUSTOM_SRC_DIR}/symbols.c + ${CUSTOM_SRC_DIR}/tt_text.c + ) + +if(WIN32 OR CYGWIN) + list(REMOVE_ITEM dtest_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) +endif() + +add_executable(dtest + ${dtest_SOURCES} + ) + +set_target_properties(dtest + PROPERTIES COMPILE_FLAGS "-DDIGITEST -DUSE_REGEX_STATIC" + ) + +target_link_libraries(dtest + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + ${GPSD_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(dtest ws2_32) +endif() + + +# Unit test for APRStt tone sequence parsing. +list(APPEND ttest_SOURCES + ${CUSTOM_SRC_DIR}/aprs_tt.c + ${CUSTOM_SRC_DIR}/tt_text.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(ttest + ${ttest_SOURCES} + ) + +set_target_properties(ttest + PROPERTIES COMPILE_FLAGS "-DTT_MAIN" + ) + +target_link_libraries(ttest + ${MISC_LIBRARIES} + ${GEOTRANZ_LIBRARIES} + ) + + +# Unit test for APRStt tone sequence / text conversions. +list(APPEND tttexttest_SOURCES + ${CUSTOM_SRC_DIR}/tt_text.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(tttexttest + ${tttexttest_SOURCES} + ) + +set_target_properties(tttexttest + PROPERTIES COMPILE_FLAGS "-DTTT_TEST" + ) + +target_link_libraries(tttexttest + ${MISC_LIBRARIES} + ) + + +# Unit test for Packet Filtering. +list(APPEND pftest_SOURCES + ${CUSTOM_SRC_DIR}/pfilter.c + ${CUSTOM_SRC_DIR}/ais.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/textcolor.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/decode_aprs.c + ${CUSTOM_SRC_DIR}/dwgpsnmea.c + ${CUSTOM_SRC_DIR}/dwgps.c + ${CUSTOM_SRC_DIR}/dwgpsd.c + ${CUSTOM_SRC_DIR}/serial_port.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/symbols.c + ${CUSTOM_SRC_DIR}/telemetry.c + ${CUSTOM_SRC_DIR}/tt_text.c + ) + +if(WIN32 OR CYGWIN) + list(REMOVE_ITEM pftest_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) +endif() + +add_executable(pftest + ${pftest_SOURCES} + ) + +set_target_properties(pftest + PROPERTIES COMPILE_FLAGS "-DPFTEST -DUSE_REGEX_STATIC" + ) + +target_link_libraries(pftest + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + ${GPSD_LIBRARIES} + Threads::Threads + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(pftest ws2_32) +endif() + +# Unit test for telemetry decoding. +list(APPEND tlmtest_SOURCES + ${CUSTOM_SRC_DIR}/telemetry.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +if(WIN32 OR CYGWIN) + list(REMOVE_ITEM tlmtest_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) +endif() + +add_executable(tlmtest + ${tlmtest_SOURCES} + ) + +set_target_properties(tlmtest + PROPERTIES COMPILE_FLAGS "-DTEST -DUSE_REGEX_STATIC" + ) + +target_link_libraries(tlmtest + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(tlmtest ws2_32) +endif() + + +# Unit test for location coordinate conversion. +list(APPEND lltest_SOURCES + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(lltest + ${lltest_SOURCES} + ) + +set_target_properties(lltest + PROPERTIES COMPILE_FLAGS "-DLLTEST" + ) + +target_link_libraries(lltest + ${MISC_LIBRARIES} + ) + + +# Unit test for encoding position & object report. +list(APPEND enctest_SOURCES + ${CUSTOM_SRC_DIR}/encode_aprs.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(enctest + ${enctest_SOURCES} + ) + +set_target_properties(enctest + PROPERTIES COMPILE_FLAGS "-DEN_MAIN" + ) + +target_link_libraries(enctest + ${MISC_LIBRARIES} + ) + + +# Unit test for KISS encapsulation. +list(APPEND kisstest_SOURCES + ${CUSTOM_SRC_DIR}/kiss_frame.c + ) + +add_executable(kisstest + ${kisstest_SOURCES} + ) + +set_target_properties(kisstest + PROPERTIES COMPILE_FLAGS "-DKISSTEST" + ) + + +# Unit test for constructing frames besides UI. +list(APPEND pad2test_SOURCES + ${CUSTOM_SRC_DIR}/ax25_pad2.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(pad2test + ${pad2test_SOURCES} + ) + +set_target_properties(pad2test + PROPERTIES COMPILE_FLAGS "-DPAD2TEST -DUSE_REGEX_STATIC" + ) + +target_link_libraries(pad2test + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + ) + +if(WIN32 OR CYGWIN) + target_link_libraries(pad2test ws2_32) +endif() + + +# Unit Test for XID frame encode/decode. +list(APPEND xidtest_SOURCES + ${CUSTOM_SRC_DIR}/xid.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(xidtest + ${xidtest_SOURCES} + ) + +set_target_properties(xidtest + PROPERTIES COMPILE_FLAGS "-DXIDTEST" + ) + +target_link_libraries(xidtest + ${MISC_LIBRARIES} + ) + + +# Unit Test for DTMF encode/decode. +list(APPEND dtmftest_SOURCES + ${CUSTOM_SRC_DIR}/dtmf.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(dtmftest + ${dtmftest_SOURCES} + ) + +set_target_properties(dtmftest + PROPERTIES COMPILE_FLAGS "-DDTMF_TEST" + ) + +# Unit Test FX.25 algorithm. + +list(APPEND fxsend_SOURCES + ${CUSTOM_SRC_DIR}/fx25_send.c + ${CUSTOM_SRC_DIR}/fx25_encode.c + ${CUSTOM_SRC_DIR}/fx25_init.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(fxsend + ${fxsend_SOURCES} + ) + +set_target_properties(fxsend + PROPERTIES COMPILE_FLAGS "-DFXTEST" + ) + +list(APPEND fxrec_SOURCES + ${CUSTOM_SRC_DIR}/fx25_rec.c + ${CUSTOM_SRC_DIR}/fx25_extract.c + ${CUSTOM_SRC_DIR}/fx25_init.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(fxrec + ${fxrec_SOURCES} + ) + +set_target_properties(fxrec + PROPERTIES COMPILE_FLAGS "-DFXTEST" + ) + + +# Unit Test IL2P with out modems. + +list(APPEND il2p_test_SOURCES + ${CUSTOM_SRC_DIR}/il2p_test.c + ${CUSTOM_SRC_DIR}/il2p_init.c + ${CUSTOM_SRC_DIR}/il2p_rec.c + ${CUSTOM_SRC_DIR}/il2p_send.c + ${CUSTOM_SRC_DIR}/il2p_codec.c + ${CUSTOM_SRC_DIR}/il2p_payload.c + ${CUSTOM_SRC_DIR}/il2p_header.c + ${CUSTOM_SRC_DIR}/il2p_scramble.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/ax25_pad2.c + ${CUSTOM_SRC_DIR}/fx25_encode.c + ${CUSTOM_SRC_DIR}/fx25_extract.c + ${CUSTOM_SRC_DIR}/fx25_init.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + +add_executable(il2p_test + ${il2p_test_SOURCES} + ) + +#FIXME - remove if not needed. +#set_target_properties(il2p_test +# PROPERTIES COMPILE_FLAGS "-DXXXXX" +# ) + +target_link_libraries(il2p_test + ${MISC_LIBRARIES} + ) + +# doing ctest on previous programs + +add_test(dtest dtest) +add_test(ttest ttest) +add_test(tttexttest tttexttest) +add_test(pftest pftest) +add_test(tlmtest tlmtest) +add_test(lltest lltest) +add_test(enctest enctest) +add_test(kisstest kisstest) +add_test(pad2test pad2test) +add_test(xidtest xidtest) +add_test(dtmftest dtmftest) + +add_test(check-fx25 "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-FX25_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-il2p "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-IL2P_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem1200 "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM1200_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem1200-i "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM1200_IL2P_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem300 "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM300_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem9600 "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM9600_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem9600-i "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM9600_IL2P_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem19200 "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM19200_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem2400-a "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM2400-a_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem2400-b "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM2400-b_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem2400-g "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM2400-g_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modem4800 "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEM4800_FILE}${CUSTOM_SCRIPT_SUFFIX}") +add_test(check-modemeas "${CUSTOM_TEST_BINARY_DIR}/${TEST_CHECK-MODEMEAS_FILE}${CUSTOM_SCRIPT_SUFFIX}") + + + +# ----------------------------- Manual tests and experiments --------------------------- +if(OPTIONAL_TEST) + + # Unit test for IGate + list(APPEND itest_SOURCES + ${CUSTOM_SRC_DIR}/igate.c + ${CUSTOM_SRC_DIR}/ais.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/mheard.c + ${CUSTOM_SRC_DIR}/pfilter.c + ${CUSTOM_SRC_DIR}/telemetry.c + ${CUSTOM_SRC_DIR}/decode_aprs.c + ${CUSTOM_SRC_DIR}/dwgpsnmea.c + ${CUSTOM_SRC_DIR}/dwgps.c + ${CUSTOM_SRC_DIR}/dwgpsd.c + ${CUSTOM_SRC_DIR}/serial_port.c + ${CUSTOM_SRC_DIR}/textcolor.c + ${CUSTOM_SRC_DIR}/dtime_now.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/tt_text.c + ${CUSTOM_SRC_DIR}/symbols.c + ) + + if(WIN32 OR CYGWIN) + list(REMOVE_ITEM itest_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) + endif() + + add_executable(itest + ${itest_SOURCES} + ) + + set_target_properties(itest + PROPERTIES COMPILE_FLAGS "-DITEST" + ) + + target_link_libraries(itest + ${MISC_LIBRARIES} + ${GPSD_LIBRARIES} + Threads::Threads + ) + + if(WIN32 OR CYGWIN) + target_link_libraries(itest ws2_32) + endif() + + + # For demodulator tweaking experiments. + list(APPEND testagc_SOURCES + ${CUSTOM_SRC_DIR}/atest.c + ${CUSTOM_SRC_DIR}/ais.c + ${CUSTOM_SRC_DIR}/demod.c + ${CUSTOM_SRC_DIR}/dsp.c + ${CUSTOM_SRC_DIR}/demod_afsk.c + ${CUSTOM_SRC_DIR}/demod_psk.c + ${CUSTOM_SRC_DIR}/demod_9600.c + ${CUSTOM_SRC_DIR}/hdlc_rec.c + ${CUSTOM_SRC_DIR}/hdlc_rec2.c + ${CUSTOM_SRC_DIR}/multi_modem.c + ${CUSTOM_SRC_DIR}/rrbb.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/decode_aprs.c + ${CUSTOM_SRC_DIR}/dwgpsnmea.c + ${CUSTOM_SRC_DIR}/dwgps.c + ${CUSTOM_SRC_DIR}/dwgpsd.c + ${CUSTOM_SRC_DIR}/serial_port.c + ${CUSTOM_SRC_DIR}/telemetry.c + ${CUSTOM_SRC_DIR}/dtime_now.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/tt_text.c + ${CUSTOM_SRC_DIR}/symbols.c + ${CUSTOM_SRC_DIR}/textcolor.c + ) + + if(WIN32 OR CYGWIN) + list(REMOVE_ITEM testagc_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) + endif() + + add_executable(testagc + ${testagc_SOURCES} + ) + + target_link_libraries(testagc + ${MISC_LIBRARIES} + ${GPSD_LIBRARIES} + Threads::Threads + ) + + if(WIN32 OR CYGWIN) + target_link_libraries(testagc ws2_32) + endif() + + + # Send GPS location to KISS TNC each second. + list(APPEND walk96_SOURCES + ${CUSTOM_SRC_DIR}/walk96.c + ${CUSTOM_SRC_DIR}/ais.c + ${CUSTOM_SRC_DIR}/dwgps.c + ${CUSTOM_SRC_DIR}/dwgpsnmea.c + ${CUSTOM_SRC_DIR}/dwgpsd.c + ${CUSTOM_SRC_DIR}/kiss_frame.c + ${CUSTOM_SRC_DIR}/latlong.c + ${CUSTOM_SRC_DIR}/encode_aprs.c + ${CUSTOM_SRC_DIR}/serial_port.c + ${CUSTOM_SRC_DIR}/textcolor.c + ${CUSTOM_SRC_DIR}/ax25_pad.c + ${CUSTOM_SRC_DIR}/fcs_calc.c + ${CUSTOM_SRC_DIR}/xmit.c + ${CUSTOM_SRC_DIR}/xid.c + ${CUSTOM_SRC_DIR}/hdlc_send.c + ${CUSTOM_SRC_DIR}/gen_tone.c + ${CUSTOM_SRC_DIR}/ptt.c + ${CUSTOM_SRC_DIR}/tq.c + ${CUSTOM_SRC_DIR}/hdlc_rec.c + ${CUSTOM_SRC_DIR}/hdlc_rec2.c + ${CUSTOM_SRC_DIR}/rrbb.c + ${CUSTOM_SRC_DIR}/dsp.c + ${CUSTOM_SRC_DIR}/multi_modem.c + ${CUSTOM_SRC_DIR}/demod.c + ${CUSTOM_SRC_DIR}/demod_afsk.c + ${CUSTOM_SRC_DIR}/demod_psk.c + ${CUSTOM_SRC_DIR}/demod_9600.c + ${CUSTOM_SRC_DIR}/server.c + ${CUSTOM_SRC_DIR}/morse.c + ${CUSTOM_SRC_DIR}/dtmf.c + ${CUSTOM_SRC_DIR}/audio_stats.c + ${CUSTOM_SRC_DIR}/dtime_now.c + ${CUSTOM_SRC_DIR}/dlq.c + ) + + if(LINUX) + list(APPEND walk96_SOURCES + ${CUSTOM_SRC_DIR}/audio.c + ) + if(UDEV_FOUND) + list(APPEND walk96_SOURCES + ${CUSTOM_SRC_DIR}/cm108.c + ) + endif() + elseif(WIN32 OR CYGWIN) # windows + list(APPEND walk96_SOURCES + ${CUSTOM_SRC_DIR}/audio_win.c + ) + list(REMOVE_ITEM walk96_SOURCES + ${CUSTOM_SRC_DIR}/dwgpsd.c + ) + else() # macOS freebsd openbsd + list(APPEND walk96_SOURCES + ${CUSTOM_SRC_DIR}/audio_portaudio.c + ) + endif() + + add_executable(walk96 + ${walk96_SOURCES} + ) + + set_target_properties(walk96 + PROPERTIES COMPILE_FLAGS "-DWALK96 -DUSE_REGEX_STATIC" + ) + + target_link_libraries(walk96 + ${MISC_LIBRARIES} + ${REGEX_LIBRARIES} + ${GPSD_LIBRARIES} + ${HAMLIB_LIBRARIES} + ${ALSA_LIBRARIES} + ${PORTAUDIO_LIBRARIES} + ${UDEV_LIBRARIES} + Threads::Threads + ) + + if(WIN32 OR CYGWIN) + target_link_libraries(walk96 ws2_32) + endif() + + + # TODO miss the audio file + + # testagc + # ./atest -P H+ -F 0 ../01_Track_1.wav ../02_Track_2.wav | grep "packets decoded in" >atest.out + + # testagc3 + # ./gen_packets -B 300 -n 100 -o noisy3.wav + # ./atest3 -B 300 -P D -D 3 noisy3.wav | grep "packets decoded in" >atest.out + + # testagc96 + # ./gen_packets -B 9600 -n 100 -o noisy96.wav + # ./atest96 -B 9600 ../walkabout9600c.wav noisy96.wav zzz16.wav zzz16.wav zzz16.wav zzz8.wav zzz8.wav zzz8.wav | grep "packets decoded in" >atest.out + + # testagc24 + # ./atest24 -B 2400 test2400.wav | grep "packets decoded in" >atest.out + + # testagc24mfj + # ./atest24mfj -F 1 -B 2400 ../ref-doc/MFJ-2400-PSK/2k4_short.wav + + # testagc48 + # ./atest48 -B 4800 test4800.wav | grep "packets decoded in" >atest.out +endif() # OPTIONAL_TEST diff --git a/test/scripts/check-fx25 b/test/scripts/check-fx25 new file mode 100755 index 00000000..d5003337 --- /dev/null +++ b/test/scripts/check-fx25 @@ -0,0 +1,16 @@ +@CUSTOM_SHELL_SHABANG@ + +@FXSEND_BIN@ +@FXREC_BIN@ + +@GEN_PACKETS_BIN@ -B9600 -n 100 -X 0 -o test96f0.wav +@ATEST_BIN@ -B9600 -F0 -L60 -G64 test96f0.wav + +@GEN_PACKETS_BIN@ -B9600 -n 100 -X 16 -o test96f16.wav +@ATEST_BIN@ -B9600 -F0 -L63 -G67 test96f16.wav + +@GEN_PACKETS_BIN@ -B9600 -n 100 -X 32 -o test96f32.wav +@ATEST_BIN@ -B9600 -F0 -L64 -G68 test96f32.wav + +@GEN_PACKETS_BIN@ -B9600 -n 100 -X 64 -o test96f64.wav +@ATEST_BIN@ -B9600 -F0 -L71 -G75 test96f64.wav diff --git a/test/scripts/check-il2p b/test/scripts/check-il2p new file mode 100755 index 00000000..da404720 --- /dev/null +++ b/test/scripts/check-il2p @@ -0,0 +1,4 @@ +@CUSTOM_SHELL_SHABANG@ + +@IL2P_TEST_BIN@ + diff --git a/test/scripts/check-modem1200 b/test/scripts/check-modem1200 new file mode 100755 index 00000000..d3022796 --- /dev/null +++ b/test/scripts/check-modem1200 @@ -0,0 +1,7 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -n 100 -o test12.wav +@ATEST_BIN@ -F0 -PA -D1 -L66 -G72 test12.wav +@ATEST_BIN@ -F1 -PA -D1 -L72 -G78 test12.wav +@ATEST_BIN@ -F0 -PB -D1 -L66 -G74 test12.wav +@ATEST_BIN@ -F1 -PB -D1 -L70 -G82 test12.wav diff --git a/test/scripts/check-modem1200-i b/test/scripts/check-modem1200-i new file mode 100755 index 00000000..9ef01bab --- /dev/null +++ b/test/scripts/check-modem1200-i @@ -0,0 +1,4 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -I1 -n 100 -o test12-il2p.wav +@ATEST_BIN@ -P+ -D1 -L92 -G94 test12-il2p.wav diff --git a/test/scripts/check-modem19200 b/test/scripts/check-modem19200 new file mode 100755 index 00000000..ae49e46f --- /dev/null +++ b/test/scripts/check-modem19200 @@ -0,0 +1,7 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -r 96000 -B19200 -a 170 -o test19.wav +@ATEST_BIN@ -B19200 -F0 -L4 test19.wav +@GEN_PACKETS_BIN@ -r 96000 -B19200 -n 100 -o test19.wav +@ATEST_BIN@ -B19200 -F0 -L60 -G66 test19.wav +@ATEST_BIN@ -B19200 -F1 -L64 -G69 test19.wav diff --git a/test/scripts/check-modem2400-a b/test/scripts/check-modem2400-a new file mode 100755 index 00000000..33dfebf4 --- /dev/null +++ b/test/scripts/check-modem2400-a @@ -0,0 +1,5 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B2400 -j -n 100 -o test24-a.wav +@ATEST_BIN@ -B2400 -j -F0 -L76 -G83 test24-a.wav +@ATEST_BIN@ -B2400 -j -F1 -L84 -G89 test24-a.wav diff --git a/test/scripts/check-modem2400-b b/test/scripts/check-modem2400-b new file mode 100755 index 00000000..913a6088 --- /dev/null +++ b/test/scripts/check-modem2400-b @@ -0,0 +1,5 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B2400 -J -n 100 -o test24-b.wav +@ATEST_BIN@ -B2400 -J -F0 -L81 -G88 test24-b.wav +@ATEST_BIN@ -B2400 -J -F1 -L86 -G90 test24-b.wav diff --git a/test/scripts/check-modem2400-g b/test/scripts/check-modem2400-g new file mode 100755 index 00000000..da7e233f --- /dev/null +++ b/test/scripts/check-modem2400-g @@ -0,0 +1,4 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B2400 -g -n 100 -o test24-g.wav +@ATEST_BIN@ -B2400 -g -F0 -L99 -G101 test24-g.wav diff --git a/test/scripts/check-modem300 b/test/scripts/check-modem300 new file mode 100755 index 00000000..6d6e6432 --- /dev/null +++ b/test/scripts/check-modem300 @@ -0,0 +1,7 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B300 -n 100 -o test3.wav +@ATEST_BIN@ -B300 -PA -F0 -L65 -G71 test3.wav +@ATEST_BIN@ -B300 -PA -F1 -L69 -G75 test3.wav +@ATEST_BIN@ -B300 -PB -F0 -L69 -G75 test3.wav +@ATEST_BIN@ -B300 -PB -F1 -L73 -G79 test3.wav diff --git a/test/scripts/check-modem4800 b/test/scripts/check-modem4800 new file mode 100755 index 00000000..7d511bb1 --- /dev/null +++ b/test/scripts/check-modem4800 @@ -0,0 +1,5 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B4800 -n 100 -o test48.wav +@ATEST_BIN@ -B4800 -F0 -L68 -G74 test48.wav +@ATEST_BIN@ -B4800 -F1 -L72 -G84 test48.wav diff --git a/test/scripts/check-modem9600 b/test/scripts/check-modem9600 new file mode 100755 index 00000000..fa5f6583 --- /dev/null +++ b/test/scripts/check-modem9600 @@ -0,0 +1,7 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B9600 -a 170 -o test96.wav +@ATEST_BIN@ -B9600 -F0 -L4 -G4 test96.wav +@GEN_PACKETS_BIN@ -B9600 -n 100 -o test96.wav +@ATEST_BIN@ -B9600 -F0 -L61 -G65 test96.wav +@ATEST_BIN@ -B9600 -F1 -L62 -G66 test96.wav diff --git a/test/scripts/check-modem9600-i b/test/scripts/check-modem9600-i new file mode 100755 index 00000000..0ba01bea --- /dev/null +++ b/test/scripts/check-modem9600-i @@ -0,0 +1,18 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B9600 -I1 -n 100 -o test96-il2p-I1.wav +@ATEST_BIN@ -B9600 -L72 -G76 test96-il2p-I1.wav +@ATEST_BIN@ -B9600 -P+ -L76 -G80 test96-il2p-I1.wav + +@GEN_PACKETS_BIN@ -B9600 -I0 -n 100 -o test96-il2p-I0.wav +@ATEST_BIN@ -B9600 -L64 -G68 test96-il2p-I0.wav + + +@GEN_PACKETS_BIN@ -B9600 -i1 -n 100 -o test96-il2p-i1.wav +@ATEST_BIN@ -B9600 -L70 -G74 test96-il2p-i1.wav +@ATEST_BIN@ -B9600 -P+ -L73 -G77 test96-il2p-i1.wav + +@GEN_PACKETS_BIN@ -B9600 -i0 -n 100 -o test96-il2p-i0.wav +@ATEST_BIN@ -B9600 -L67 -G71 test96-il2p-i0.wav + + diff --git a/test/scripts/check-modemeas b/test/scripts/check-modemeas new file mode 100755 index 00000000..c2a88539 --- /dev/null +++ b/test/scripts/check-modemeas @@ -0,0 +1,4 @@ +@CUSTOM_SHELL_SHABANG@ + +@GEN_PACKETS_BIN@ -B EAS -o testeas.wav +@ATEST_BIN@ -B EAS -L 6 -G 6 testeas.wav diff --git a/textcolor.c b/textcolor.c deleted file mode 100644 index e518b939..00000000 --- a/textcolor.c +++ /dev/null @@ -1,392 +0,0 @@ - -// -// This file is part of Dire Wolf, an amateur radio packet TNC. -// -// Copyright (C) 2011, 2012, 2013, 2014 John Langner, WB2OSZ -// -// 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, see . -// - - -/*------------------------------------------------------------------- - * - * Name: textcolor.c - * - * Purpose: Originally this would only set color of text - * and we used printf everywhere. - * Now we also have a printf replacement that can - * be used to redirect all output to the desired place. - * This opens the door to using ncurses, a GUI, or - * running as a daemon. - * - * Description: For Linux and Cygwin use the ANSI escape sequences. - * In earlier versions of Windows, the cmd window and ANSI.SYS - * could interpret this but it doesn't seem to be available - * anymore so we use a different interface. - * - * References: - * http://en.wikipedia.org/wiki/ANSI_escape_code - * http://academic.evergreen.edu/projects/biophysics/technotes/program/ansi_esc.htm - * - * - ->>>> READ THIS PART!!! <<<< - - * - * - * Problem: The ANSI escape sequences, used on Linux, allow 8 basic colors. - * Unfortunately, white is not one of them. We only have dark - * white, also known as light gray. To get brighter colors, - * we need to apply an attribute. On some systems, the bold - * attribute produces a brighter color rather than a bold font. - * On other systems, we need to use the blink attribute to get - * bright colors, including white. However on others, blink - * does actually produce blinking characters. - * - * Several people have also complained that bright green is - * very hard to read against a light background. The current - * implementation does not allow easy user customization of colors. - * - * Currently, the only option is to put "-t 0" on the command - * line to disable all text color. This is more readable but - * makes it harder to distinguish different types of - * information, e.g. received packets vs. error messages. - * - * A few people have suggested ncurses. This needs to - * be investigated for a future version. The foundation has - * already been put in place. All of the printf's should have been - * replaced by dw_printf, defined in this file. All of the - * text output is now being funneled thru this one function - * so it should be easy to send it to the user by some - * other means. - * - *--------------------------------------------------------------------*/ - - -#include "direwolf.h" // Should be first. includes windows.h - -#include -#include -#include - - -#if __WIN32__ - -#define BACKGROUND_WHITE (BACKGROUND_RED | BACKGROUND_GREEN | BACKGROUND_BLUE | BACKGROUND_INTENSITY) - - - -#elif __CYGWIN__ /* Cygwin */ - -/* For Cygwin we need "blink" (5) rather than the */ -/* expected bright/bold (1) to get bright white background. */ -/* Makes no sense but I stumbled across that somewhere. */ - -static const char background_white[] = "\e[5;47m"; - -/* Whenever a dark color is used, the */ -/* background is reset and needs to be set again. */ - -static const char black[] = "\e[0;30m" "\e[5;47m"; -static const char red[] = "\e[1;31m"; -static const char green[] = "\e[1;32m"; -static const char yellow[] = "\e[1;33m"; -static const char blue[] = "\e[1;34m"; -static const char magenta[] = "\e[1;35m"; -static const char cyan[] = "\e[1;36m"; -static const char dark_green[] = "\e[0;32m" "\e[5;47m"; - -/* Clear from cursor to end of screen. */ - -static const char clear_eos[] = "\e[0J"; - - -#elif __arm__ /* Linux on Raspberry Pi or similar */ - -/* We need "blink" (5) rather than the */ -/* expected bright/bold (1) to get bright white background. */ -/* Makes no sense but I stumbled across that somewhere. */ - -/* If you do get blinking, remove all references to "\e[5;47m" */ - -static const char background_white[] = "\e[5;47m"; - -/* Whenever a dark color is used, the */ -/* background is reset and needs to be set again. */ - -static const char black[] = "\e[0;30m" "\e[5;47m"; -static const char red[] = "\e[1;31m" "\e[5;47m"; -static const char green[] = "\e[1;32m" "\e[5;47m"; -//static const char yellow[] = "\e[1;33m" "\e[5;47m"; -static const char blue[] = "\e[1;34m" "\e[5;47m"; -static const char magenta[] = "\e[1;35m" "\e[5;47m"; -//static const char cyan[] = "\e[1;36m" "\e[5;47m"; -static const char dark_green[] = "\e[0;32m" "\e[5;47m"; - -/* Clear from cursor to end of screen. */ - -static const char clear_eos[] = "\e[0J"; - - -#else /* Other Linux */ - -#if 1 /* new in version 1.2, as suggested by IW2DHW */ - /* Test done using gnome-terminal and xterm */ - -static const char background_white[] = "\e[48;2;255;255;255m"; - -/* Whenever a dark color is used, the */ -/* background is reset and needs to be set again. */ - - -static const char black[] = "\e[0;30m" "\e[48;2;255;255;255m"; -static const char red[] = "\e[0;31m" "\e[48;2;255;255;255m"; -static const char green[] = "\e[0;32m" "\e[48;2;255;255;255m"; -//static const char yellow[] = "\e[0;33m" "\e[48;2;255;255;255m"; -static const char blue[] = "\e[0;34m" "\e[48;2;255;255;255m"; -static const char magenta[] = "\e[0;35m" "\e[48;2;255;255;255m"; -//static const char cyan[] = "\e[0;36m" "\e[48;2;255;255;255m"; -static const char dark_green[] = "\e[0;32m" "\e[48;2;255;255;255m"; - - -#else /* from version 1.1 */ - - -static const char background_white[] = "\e[47;1m"; - -/* Whenever a dark color is used, the */ -/* background is reset and needs to be set again. */ - -static const char black[] = "\e[0;30m" "\e[1;47m"; -static const char red[] = "\e[1;31m" "\e[1;47m"; -static const char green[] = "\e[1;32m" "\e[1;47m"; -//static const char yellow[] = "\e[1;33m" "\e[1;47m"; -static const char blue[] = "\e[1;34m" "\e[1;47m"; -static const char magenta[] = "\e[1;35m" "\e[1;47m"; -//static const char cyan[] = "\e[1;36m" "\e[1;47m"; -static const char dark_green[] = "\e[0;32m" "\e[1;47m"; - - -#endif - - -/* Clear from cursor to end of screen. */ - -static const char clear_eos[] = "\e[0J"; - -#endif /* end Linux */ - - -#include "textcolor.h" - - -/* - * g_enable_color: - * 0 = disable text colors. - * 1 = normal. - * others... future possibility. - */ - -static int g_enable_color = 1; - - -void text_color_init (int enable_color) -{ - - g_enable_color = enable_color; - - -#if __WIN32__ - - - if (g_enable_color) { - - HANDLE h; - CONSOLE_SCREEN_BUFFER_INFO csbi; - WORD attr = BACKGROUND_WHITE; - DWORD length; - COORD coord; - DWORD nwritten; - - h = GetStdHandle(STD_OUTPUT_HANDLE); - if (h != NULL && h != INVALID_HANDLE_VALUE) { - - GetConsoleScreenBufferInfo (h, &csbi); - - length = csbi.dwSize.X * csbi.dwSize.Y; - coord.X = 0; - coord.Y = 0; - FillConsoleOutputAttribute (h, attr, length, coord, &nwritten); - } - } - -#else - if (g_enable_color) { - //printf ("%s", clear_eos); - printf ("%s", background_white); - printf ("%s", clear_eos); - printf ("%s", black); - } -#endif -} - - -#if __WIN32__ - -/* Seems that ANSI.SYS is no longer available. */ - - -void text_color_set ( enum dw_color_e c ) -{ - WORD attr; - HANDLE h; - - if (g_enable_color == 0) { - return; - } - - switch (c) { - - default: - case DW_COLOR_INFO: - attr = BACKGROUND_WHITE; - break; - - case DW_COLOR_ERROR: - attr = FOREGROUND_RED | FOREGROUND_INTENSITY | BACKGROUND_WHITE; - break; - - case DW_COLOR_REC: - attr = FOREGROUND_GREEN | FOREGROUND_INTENSITY | BACKGROUND_WHITE; - break; - - case DW_COLOR_DECODED: - attr = FOREGROUND_BLUE | FOREGROUND_INTENSITY | BACKGROUND_WHITE; - break; - - case DW_COLOR_XMIT: - attr = FOREGROUND_RED | FOREGROUND_BLUE | FOREGROUND_INTENSITY | BACKGROUND_WHITE; - break; - - case DW_COLOR_DEBUG: - attr = FOREGROUND_GREEN | BACKGROUND_WHITE; - break; - } - - h = GetStdHandle(STD_OUTPUT_HANDLE); - - if (h != NULL && h != INVALID_HANDLE_VALUE) { - SetConsoleTextAttribute (h, attr); - } -} - -#else - -void text_color_set ( enum dw_color_e c ) -{ - - if (g_enable_color == 0) { - return; - } - - switch (c) { - - default: - case DW_COLOR_INFO: - printf ("%s", black); - break; - - case DW_COLOR_ERROR: - printf ("%s", red); - break; - - case DW_COLOR_REC: - printf ("%s", green); - break; - - case DW_COLOR_DECODED: - printf ("%s", blue); - break; - - case DW_COLOR_XMIT: - printf ("%s", magenta); - break; - - case DW_COLOR_DEBUG: - printf ("%s", dark_green); - break; - } -} - -#endif - - -/*------------------------------------------------------------------- - * - * Name: dw_printf - * - * Purpose: printf replacement that allows us to send all text - * output to stdout or other desired destination. - * - * Inputs: fmt - C language format. - * ... - Addtional arguments, just like printf. - * - * - * Returns: Number of characters in result. - * - * Bug: Fixed size buffer. - * I'd rather not do a malloc for each print. - * - *--------------------------------------------------------------------*/ - - -// TODO: replace all printf, look for stderr, perror -// TODO: $ grep printf *.c | grep -v dw_printf | grep -v fprintf | gawk '{ print $1 }' | sort -u - - -int dw_printf (const char *fmt, ...) -{ -#define BSIZE 1000 - va_list args; - char buffer[BSIZE]; - int len; - - va_start (args, fmt); - len = vsnprintf (buffer, BSIZE, fmt, args); - va_end (args); - -// TODO: other possible destinations... - - fputs (buffer, stdout); - return (len); -} - - - -#if TESTC -main () -{ - printf ("Initial condition\n"); - text_color_init (1); - printf ("After text_color_init\n"); - text_color_set(DW_COLOR_INFO); printf ("Info\n"); - text_color_set(DW_COLOR_ERROR); printf ("Error\n"); - text_color_set(DW_COLOR_REC); printf ("Rec\n"); - text_color_set(DW_COLOR_DECODED); printf ("Decoded\n"); - text_color_set(DW_COLOR_XMIT); printf ("Xmit\n"); - text_color_set(DW_COLOR_DEBUG); printf ("Debug\n"); -} -#endif - -/* end textcolor.c */ diff --git a/tnc-test-cd-results.png b/tnc-test-cd-results.png new file mode 100644 index 00000000..facff309 Binary files /dev/null and b/tnc-test-cd-results.png differ diff --git a/version.h b/version.h deleted file mode 100644 index b61ab1a3..00000000 --- a/version.h +++ /dev/null @@ -1,8 +0,0 @@ - -/* Dire Wolf version 1.4 */ - -#define APP_TOCALL "APDW" - -#define MAJOR_VERSION 1 -#define MINOR_VERSION 4 -//#define EXTRA_VERSION "Beta Test"