Merge remote-tracking branch 'upstream/main' into sleep-revamp
This commit is contained in:
commit
295103bb31
32
.git-blame-ignore-revs
Normal file
32
.git-blame-ignore-revs
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# tests/run-tests.py: Reformat with Black.
|
||||||
|
2a38d7103672580882fb621a5b76e8d26805d593
|
||||||
|
|
||||||
|
# all: Update Python code to conform to latest black formatting.
|
||||||
|
06659077a81b85882254cf0953c33b27614e018e
|
||||||
|
|
||||||
|
# tools/uncrustify: Enable more opts to remove space between func and '('.
|
||||||
|
77ed6f69ac35c1663a5633a8ee1d8a2446542204
|
||||||
|
|
||||||
|
# tools/codeformat.py: Include extmod/{btstack,nimble} in code formatting.
|
||||||
|
026fda605e03113d6e753290d65fed774418bc53
|
||||||
|
|
||||||
|
# all: Format code to add space after C++-style comment start.
|
||||||
|
84fa3312cfa7d2237d4b56952f2cd6e3591210c4
|
||||||
|
|
||||||
|
# tests: Format all Python code with black, except tests in basics subdir.
|
||||||
|
3dc324d3f1312e40d3a8ed87e7244966bb756f26
|
||||||
|
|
||||||
|
# all: Remove spaces inside and around parenthesis.
|
||||||
|
1a3e386c67e03a79eb768cb6e9f6777e002d6660
|
||||||
|
|
||||||
|
# all: Remove spaces between nested paren and inside function arg paren.
|
||||||
|
feb25775851ba0c04b8d1013716f442258879d9c
|
||||||
|
|
||||||
|
# all: Reformat C and Python source code with tools/codeformat.py.
|
||||||
|
69661f3343bedf86e514337cff63d96cc42f8859
|
||||||
|
|
||||||
|
# stm32/usbdev: Convert files to unix line endings.
|
||||||
|
abde0fa2267f9062b28c3c015d7662a550125cc6
|
||||||
|
|
||||||
|
# all: Remove trailing spaces, per coding conventions.
|
||||||
|
761e4c7ff62896c7d8f8c3dfc3cc98a4cc4f2f6f
|
1
.gitattributes
vendored
1
.gitattributes
vendored
@ -11,6 +11,7 @@
|
|||||||
*.bat text eol=crlf
|
*.bat text eol=crlf
|
||||||
|
|
||||||
# These are binary so should never be modified by git.
|
# These are binary so should never be modified by git.
|
||||||
|
*.a binary
|
||||||
*.png binary
|
*.png binary
|
||||||
*.jpg binary
|
*.jpg binary
|
||||||
*.dxf binary
|
*.dxf binary
|
||||||
|
48
.github/workflows/build.yml
vendored
48
.github/workflows/build.yml
vendored
@ -37,7 +37,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
sudo apt-get install -y eatmydata
|
sudo apt-get install -y eatmydata
|
||||||
sudo eatmydata apt-get install -y gettext librsvg2-bin mingw-w64 latexmk texlive-fonts-recommended texlive-latex-recommended texlive-latex-extra
|
sudo eatmydata apt-get install -y gettext librsvg2-bin mingw-w64 latexmk texlive-fonts-recommended texlive-latex-recommended texlive-latex-extra gcc-aarch64-linux-gnu
|
||||||
pip install -r requirements-dev.txt
|
pip install -r requirements-dev.txt
|
||||||
- name: Versions
|
- name: Versions
|
||||||
run: |
|
run: |
|
||||||
@ -71,28 +71,25 @@ jobs:
|
|||||||
run: make -C mpy-cross -j2
|
run: make -C mpy-cross -j2
|
||||||
- name: Build unix port
|
- name: Build unix port
|
||||||
run: |
|
run: |
|
||||||
make -C ports/unix deplibs -j2
|
make -C ports/unix VARIANT=coverage -j2
|
||||||
make -C ports/unix -j2
|
|
||||||
make -C ports/unix coverage -j2
|
|
||||||
- name: Test all
|
- name: Test all
|
||||||
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython_coverage ./run-tests -j1
|
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython-coverage ./run-tests.py -j1
|
||||||
working-directory: tests
|
working-directory: tests
|
||||||
- name: Print failure info
|
|
||||||
run: |
|
|
||||||
shopt -s nullglob;
|
|
||||||
for exp in *.exp;
|
|
||||||
do testbase=$(basename $exp .exp);
|
|
||||||
echo -e "\nFAILURE $testbase";
|
|
||||||
diff -u $testbase.exp $testbase.out;
|
|
||||||
done
|
|
||||||
working-directory: tests
|
|
||||||
if: failure()
|
|
||||||
- name: Native Tests
|
- name: Native Tests
|
||||||
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython_coverage ./run-tests -j1 --emit native
|
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython-coverage ./run-tests.py -j1 --emit native
|
||||||
working-directory: tests
|
working-directory: tests
|
||||||
- name: mpy Tests
|
- name: mpy Tests
|
||||||
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython_coverage ./run-tests -j1 --via-mpy -d basics float
|
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython-coverage ./run-tests.py -j1 --mpy-cross-flags='-mcache-lookup-bc' --via-mpy -d basics float micropython
|
||||||
working-directory: tests
|
working-directory: tests
|
||||||
|
- name: Native mpy Tests
|
||||||
|
run: MICROPY_CPYTHON3=python3.8 MICROPY_MICROPYTHON=../ports/unix/micropython-coverage ./run-tests.py -j1 --mpy-cross-flags='-mcache-lookup-bc' --via-mpy --emit native -d basics float micropython
|
||||||
|
working-directory: tests
|
||||||
|
- name: Build mpy-cross.static-aarch64
|
||||||
|
run: make -C mpy-cross -j2 -f Makefile.static-aarch64
|
||||||
|
- uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: mpy-cross.static-aarch64
|
||||||
|
path: mpy-cross/mpy-cross.static-aarch64
|
||||||
- name: Build mpy-cross.static-raspbian
|
- name: Build mpy-cross.static-raspbian
|
||||||
run: make -C mpy-cross -j2 -f Makefile.static-raspbian
|
run: make -C mpy-cross -j2 -f Makefile.static-raspbian
|
||||||
- uses: actions/upload-artifact@v2
|
- uses: actions/upload-artifact@v2
|
||||||
@ -113,6 +110,7 @@ jobs:
|
|||||||
path: mpy-cross/mpy-cross.static.exe
|
path: mpy-cross/mpy-cross.static.exe
|
||||||
- name: Upload stubs and mpy-cross builds to S3
|
- name: Upload stubs and mpy-cross builds to S3
|
||||||
run: |
|
run: |
|
||||||
|
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static-aarch64 s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-aarch64-${{ env.CP_VERSION }} --no-progress --region us-east-1
|
||||||
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static-raspbian s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-raspbian-${{ env.CP_VERSION }} --no-progress --region us-east-1
|
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static-raspbian s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-raspbian-${{ env.CP_VERSION }} --no-progress --region us-east-1
|
||||||
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-amd64-linux-${{ env.CP_VERSION }} --no-progress --region us-east-1
|
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-amd64-linux-${{ env.CP_VERSION }} --no-progress --region us-east-1
|
||||||
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static.exe s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-x64-windows-${{ env.CP_VERSION }}.exe --no-progress --region us-east-1
|
[ -z "$AWS_ACCESS_KEY_ID" ] || aws s3 cp mpy-cross/mpy-cross.static.exe s3://adafruit-circuit-python/bin/mpy-cross/mpy-cross.static-x64-windows-${{ env.CP_VERSION }}.exe --no-progress --region us-east-1
|
||||||
@ -180,10 +178,12 @@ jobs:
|
|||||||
- "adafruit_itsybitsy_rp2040"
|
- "adafruit_itsybitsy_rp2040"
|
||||||
- "adafruit_neokey_trinkey_m0"
|
- "adafruit_neokey_trinkey_m0"
|
||||||
- "adafruit_proxlight_trinkey_m0"
|
- "adafruit_proxlight_trinkey_m0"
|
||||||
|
- "adafruit_qt2040_trinkey"
|
||||||
- "adafruit_qtpy_rp2040"
|
- "adafruit_qtpy_rp2040"
|
||||||
- "adafruit_rotary_trinkey_m0"
|
- "adafruit_rotary_trinkey_m0"
|
||||||
- "adafruit_slide_trinkey_m0"
|
- "adafruit_slide_trinkey_m0"
|
||||||
- "aloriumtech_evo_m51"
|
- "aloriumtech_evo_m51"
|
||||||
|
- "aramcon2_badge"
|
||||||
- "aramcon_badge_2019"
|
- "aramcon_badge_2019"
|
||||||
- "arduino_mkr1300"
|
- "arduino_mkr1300"
|
||||||
- "arduino_mkrzero"
|
- "arduino_mkrzero"
|
||||||
@ -208,6 +208,7 @@ jobs:
|
|||||||
- "clue_nrf52840_express"
|
- "clue_nrf52840_express"
|
||||||
- "cp32-m4"
|
- "cp32-m4"
|
||||||
- "cp_sapling_m0"
|
- "cp_sapling_m0"
|
||||||
|
- "cp_sapling_m0_revb"
|
||||||
- "cp_sapling_m0_spiflash"
|
- "cp_sapling_m0_spiflash"
|
||||||
- "datalore_ip_m4"
|
- "datalore_ip_m4"
|
||||||
- "datum_distance"
|
- "datum_distance"
|
||||||
@ -236,7 +237,6 @@ jobs:
|
|||||||
- "feather_mimxrt1011"
|
- "feather_mimxrt1011"
|
||||||
- "feather_mimxrt1062"
|
- "feather_mimxrt1062"
|
||||||
- "feather_nrf52840_express"
|
- "feather_nrf52840_express"
|
||||||
- "feather_radiofruit_zigbee"
|
|
||||||
- "feather_stm32f405_express"
|
- "feather_stm32f405_express"
|
||||||
- "fluff_m0"
|
- "fluff_m0"
|
||||||
- "gemma_m0"
|
- "gemma_m0"
|
||||||
@ -289,6 +289,8 @@ jobs:
|
|||||||
- "pewpew_m4"
|
- "pewpew_m4"
|
||||||
- "picoplanet"
|
- "picoplanet"
|
||||||
- "pimoroni_keybow2040"
|
- "pimoroni_keybow2040"
|
||||||
|
- "pimoroni_picolipo_16mb"
|
||||||
|
- "pimoroni_picolipo_4mb"
|
||||||
- "pimoroni_picosystem"
|
- "pimoroni_picosystem"
|
||||||
- "pimoroni_tiny2040"
|
- "pimoroni_tiny2040"
|
||||||
- "pirkey_m0"
|
- "pirkey_m0"
|
||||||
@ -320,6 +322,7 @@ jobs:
|
|||||||
- "simmel"
|
- "simmel"
|
||||||
- "snekboard"
|
- "snekboard"
|
||||||
- "sparkfun_lumidrive"
|
- "sparkfun_lumidrive"
|
||||||
|
- "sparkfun_micromod_rp2040"
|
||||||
- "sparkfun_nrf52840_micromod"
|
- "sparkfun_nrf52840_micromod"
|
||||||
- "sparkfun_nrf52840_mini"
|
- "sparkfun_nrf52840_mini"
|
||||||
- "sparkfun_pro_micro_rp2040"
|
- "sparkfun_pro_micro_rp2040"
|
||||||
@ -379,6 +382,8 @@ jobs:
|
|||||||
python3 --version
|
python3 --version
|
||||||
- name: mpy-cross
|
- name: mpy-cross
|
||||||
run: make -C mpy-cross -j2
|
run: make -C mpy-cross -j2
|
||||||
|
- name: Setup build failure matcher
|
||||||
|
run: echo "::add-matcher::$GITHUB_WORKSPACE/.github/workflows/match-build-fail.json"
|
||||||
- name: build
|
- name: build
|
||||||
run: python3 -u build_release_files.py
|
run: python3 -u build_release_files.py
|
||||||
working-directory: tools
|
working-directory: tools
|
||||||
@ -428,6 +433,8 @@ jobs:
|
|||||||
python3 --version
|
python3 --version
|
||||||
- name: mpy-cross
|
- name: mpy-cross
|
||||||
run: make -C mpy-cross -j2
|
run: make -C mpy-cross -j2
|
||||||
|
- name: Setup build failure matcher
|
||||||
|
run: echo "::add-matcher::$GITHUB_WORKSPACE/.github/workflows/match-build-fail.json"
|
||||||
- name: build
|
- name: build
|
||||||
run: python3 -u build_release_files.py
|
run: python3 -u build_release_files.py
|
||||||
working-directory: tools
|
working-directory: tools
|
||||||
@ -457,6 +464,7 @@ jobs:
|
|||||||
- "adafruit_magtag_2.9_grayscale"
|
- "adafruit_magtag_2.9_grayscale"
|
||||||
- "adafruit_metro_esp32s2"
|
- "adafruit_metro_esp32s2"
|
||||||
- "artisense_rd00"
|
- "artisense_rd00"
|
||||||
|
- "atmegazero_esp32s2"
|
||||||
- "electroniccats_bastwifi"
|
- "electroniccats_bastwifi"
|
||||||
- "espressif_kaluga_1"
|
- "espressif_kaluga_1"
|
||||||
- "espressif_saola_1_wroom"
|
- "espressif_saola_1_wroom"
|
||||||
@ -489,7 +497,7 @@ jobs:
|
|||||||
id: idf-cache
|
id: idf-cache
|
||||||
with:
|
with:
|
||||||
path: ${{ github.workspace }}/.idf_tools
|
path: ${{ github.workspace }}/.idf_tools
|
||||||
key: ${{ runner.os }}-idf-tools-${{ hashFiles('.git/modules/ports/esp32s2/esp-idf/HEAD') }}-20210304
|
key: ${{ runner.os }}-idf-tools-${{ hashFiles('.git/modules/ports/esp32s2/esp-idf/HEAD') }}-20210506
|
||||||
- name: Clone IDF submodules
|
- name: Clone IDF submodules
|
||||||
run: |
|
run: |
|
||||||
(cd $IDF_PATH && git submodule update --init)
|
(cd $IDF_PATH && git submodule update --init)
|
||||||
@ -526,6 +534,8 @@ jobs:
|
|||||||
IDF_TOOLS_PATH: ${{ github.workspace }}/.idf_tools
|
IDF_TOOLS_PATH: ${{ github.workspace }}/.idf_tools
|
||||||
- name: mpy-cross
|
- name: mpy-cross
|
||||||
run: make -C mpy-cross -j2
|
run: make -C mpy-cross -j2
|
||||||
|
- name: Setup build failure matcher
|
||||||
|
run: echo "::add-matcher::$GITHUB_WORKSPACE/.github/workflows/match-build-fail.json"
|
||||||
- name: build
|
- name: build
|
||||||
run: |
|
run: |
|
||||||
source $IDF_PATH/export.sh
|
source $IDF_PATH/export.sh
|
||||||
|
14
.github/workflows/match-build-fail.json
vendored
Normal file
14
.github/workflows/match-build-fail.json
vendored
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
"problemMatcher": [
|
||||||
|
{
|
||||||
|
"severity": "error",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": "^(Build .+ and \\x1b\\[31mfailed\\x1b\\[0m)$",
|
||||||
|
"message": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"owner": "build-failed"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
6
.gitignore
vendored
6
.gitignore
vendored
@ -28,16 +28,16 @@ dist/
|
|||||||
######################
|
######################
|
||||||
*.swp
|
*.swp
|
||||||
|
|
||||||
# Build directory
|
# Build directories
|
||||||
######################
|
######################
|
||||||
build/
|
build/
|
||||||
bin/
|
bin/
|
||||||
circuitpython-stubs/
|
circuitpython-stubs/
|
||||||
|
build-*/
|
||||||
|
|
||||||
# Test failure outputs
|
# Test failure outputs
|
||||||
######################
|
######################
|
||||||
tests/*.exp
|
tests/results/*
|
||||||
tests/*.out
|
|
||||||
|
|
||||||
# Python cache files
|
# Python cache files
|
||||||
######################
|
######################
|
||||||
|
2
.gitmodules
vendored
2
.gitmodules
vendored
@ -113,7 +113,7 @@
|
|||||||
url = https://github.com/adafruit/Adafruit_CircuitPython_Register.git
|
url = https://github.com/adafruit/Adafruit_CircuitPython_Register.git
|
||||||
[submodule "extmod/ulab"]
|
[submodule "extmod/ulab"]
|
||||||
path = extmod/ulab
|
path = extmod/ulab
|
||||||
url = https://github.com/v923z/micropython-ulab
|
url = https://github.com/adafruit/circuitpython-ulab
|
||||||
[submodule "frozen/Adafruit_CircuitPython_ESP32SPI"]
|
[submodule "frozen/Adafruit_CircuitPython_ESP32SPI"]
|
||||||
path = frozen/Adafruit_CircuitPython_ESP32SPI
|
path = frozen/Adafruit_CircuitPython_ESP32SPI
|
||||||
url = https://github.com/adafruit/Adafruit_CircuitPython_ESP32SPI
|
url = https://github.com/adafruit/Adafruit_CircuitPython_ESP32SPI
|
||||||
|
@ -762,7 +762,6 @@ today. The names appear in order of pledging.
|
|||||||
1642 Udine
|
1642 Udine
|
||||||
1643 Simon Critchley
|
1643 Simon Critchley
|
||||||
1644 Sven Haiges, Germany
|
1644 Sven Haiges, Germany
|
||||||
1645 Yi Qing Sim
|
|
||||||
1646 "silicium" ("silicium_one", if "silicium" is busy)
|
1646 "silicium" ("silicium_one", if "silicium" is busy)
|
||||||
1648 Andy O'Malia, @andyomalia
|
1648 Andy O'Malia, @andyomalia
|
||||||
1650 RedCamelApps.com
|
1650 RedCamelApps.com
|
||||||
|
2
LICENSE
2
LICENSE
@ -1,6 +1,6 @@
|
|||||||
The MIT License (MIT)
|
The MIT License (MIT)
|
||||||
|
|
||||||
Copyright (c) 2013, 2014 Damien P. George
|
Copyright (c) 2013-2021 Damien P. George
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
6
Makefile
6
Makefile
@ -7,7 +7,7 @@
|
|||||||
|
|
||||||
# You can set these variables from the command line.
|
# You can set these variables from the command line.
|
||||||
PYTHON = python3
|
PYTHON = python3
|
||||||
SPHINXOPTS =
|
SPHINXOPTS = -W --keep-going
|
||||||
SPHINXBUILD = sphinx-build
|
SPHINXBUILD = sphinx-build
|
||||||
PAPER =
|
PAPER =
|
||||||
# path to build the generated docs
|
# path to build the generated docs
|
||||||
@ -223,7 +223,7 @@ pseudoxml:
|
|||||||
all-source:
|
all-source:
|
||||||
|
|
||||||
locale/circuitpython.pot: all-source
|
locale/circuitpython.pot: all-source
|
||||||
find $(TRANSLATE_SOURCES) -type d \( $(TRANSLATE_SOURCES_EXC) \) -prune -o -type f \( -iname "*.c" -o -iname "*.h" \) -print | (LC_ALL=C sort) | xgettext -f- -L C -s --add-location=file --keyword=translate -o - | sed -e '/"POT-Creation-Date: /d' > $@
|
find $(TRANSLATE_SOURCES) -type d \( $(TRANSLATE_SOURCES_EXC) \) -prune -o -type f \( -iname "*.c" -o -iname "*.h" \) -print | (LC_ALL=C sort) | xgettext -f- -L C -s --add-location=file --keyword=translate --keyword=MP_ERROR_TEXT -o - | sed -e '/"POT-Creation-Date: /d' > $@
|
||||||
|
|
||||||
# Historically, `make translate` updated the .pot file and ran msgmerge.
|
# Historically, `make translate` updated the .pot file and ran msgmerge.
|
||||||
# However, this was a frequent source of merge conflicts. Weblate can perform
|
# However, this was a frequent source of merge conflicts. Weblate can perform
|
||||||
@ -248,7 +248,7 @@ merge-translate:
|
|||||||
|
|
||||||
.PHONY: check-translate
|
.PHONY: check-translate
|
||||||
check-translate:
|
check-translate:
|
||||||
find $(TRANSLATE_SOURCES) -type d \( $(TRANSLATE_SOURCES_EXC) \) -prune -o -type f \( -iname "*.c" -o -iname "*.h" \) -print | (LC_ALL=C sort) | xgettext -f- -L C -s --add-location=file --keyword=translate -o circuitpython.pot.tmp -p locale
|
find $(TRANSLATE_SOURCES) -type d \( $(TRANSLATE_SOURCES_EXC) \) -prune -o -type f \( -iname "*.c" -o -iname "*.h" \) -print | (LC_ALL=C sort) | xgettext -f- -L C -s --add-location=file --keyword=translate --keyword=MP_ERROR_TEXT -o circuitpython.pot.tmp -p locale
|
||||||
$(PYTHON) tools/check_translations.py locale/circuitpython.pot.tmp locale/circuitpython.pot; status=$$?; rm -f locale/circuitpython.pot.tmp; exit $$status
|
$(PYTHON) tools/check_translations.py locale/circuitpython.pot.tmp locale/circuitpython.pot; status=$$?; rm -f locale/circuitpython.pot.tmp; exit $$status
|
||||||
|
|
||||||
stubs:
|
stubs:
|
||||||
|
@ -62,33 +62,4 @@ The tinyusb examples already include a "WebUSB serial" example.
|
|||||||
Basically, this feature was ported into CircuitPython by pulling code snippets out of the
|
Basically, this feature was ported into CircuitPython by pulling code snippets out of the
|
||||||
tinyusb example, and putting them where they best belonged in the CircuitPython codebase.
|
tinyusb example, and putting them where they best belonged in the CircuitPython codebase.
|
||||||
|
|
||||||
There was one complication:
|
### TODO: This needs to be reworked for dynamic USB descriptors.
|
||||||
|
|
||||||
tinyusb uses C preprocessor macros to define things like USB descriptors.
|
|
||||||
|
|
||||||
CircuitPython uses a Python program (tools/gen_usb_descriptor.py) to create USB descriptors (etc.)
|
|
||||||
using "helper objects" from another repo (adafruit_usb_descriptor). This means some of the example
|
|
||||||
code had to be adapted to the new programing model, and gen_usb_descriptor gained new command-line
|
|
||||||
options to control the generated code.
|
|
||||||
|
|
||||||
The generated files go into the "build" directory, look for autogen_usb_descriptor.c and
|
|
||||||
genhdr/autogen_usb_descriptor.h.
|
|
||||||
|
|
||||||
|
|
||||||
Also worth pointing out - the re-use of the CDC connect/disconnect mechanism is not actually part
|
|
||||||
of the WebUSB standard, it's more of "common idiom". We make use of it here because we need to know
|
|
||||||
when we should be paying attention to the WebUSB serial interface, and when we should ignore it..
|
|
||||||
|
|
||||||
## Possible future work areas
|
|
||||||
|
|
||||||
The current code uses the existing Python infrastructure to create the Interface descriptor, but
|
|
||||||
simply outputs the code snippets from the original tinyusb demo code to create the WEBUSB_URL,
|
|
||||||
BOS, and MS_OS_20 descriptors. I suppose additional work could be done to add these to the
|
|
||||||
adafruit_usb_descriptor project, and then gen_usb_descriptor.py could be modified to make use
|
|
||||||
of them.
|
|
||||||
|
|
||||||
Program gen_usb_descriptor.py creates objects for most interface types, regardless of whether or
|
|
||||||
not they are actually enabled. This increases the size of a generated string table. I made the
|
|
||||||
new vendor-interface-related code not do this (because some of the ARM platforms would no longer
|
|
||||||
build), but I did not go back and do this for the other interface types (CDC, MIDI, HID, etc.)
|
|
||||||
Some FLASH savings are probably possible if this is done.
|
|
||||||
|
6
conf.py
6
conf.py
@ -25,7 +25,6 @@ import sys
|
|||||||
import urllib.parse
|
import urllib.parse
|
||||||
import time
|
import time
|
||||||
|
|
||||||
import recommonmark
|
|
||||||
from sphinx.transforms import SphinxTransform
|
from sphinx.transforms import SphinxTransform
|
||||||
from docutils import nodes
|
from docutils import nodes
|
||||||
from sphinx import addnodes
|
from sphinx import addnodes
|
||||||
@ -68,8 +67,9 @@ extensions = [
|
|||||||
'sphinx.ext.intersphinx',
|
'sphinx.ext.intersphinx',
|
||||||
'sphinx.ext.todo',
|
'sphinx.ext.todo',
|
||||||
'sphinx.ext.coverage',
|
'sphinx.ext.coverage',
|
||||||
|
'sphinx_search.extension',
|
||||||
'rstjinja',
|
'rstjinja',
|
||||||
'recommonmark',
|
'myst_parser',
|
||||||
]
|
]
|
||||||
|
|
||||||
# Add any paths that contain templates here, relative to this directory.
|
# Add any paths that contain templates here, relative to this directory.
|
||||||
@ -489,6 +489,8 @@ class CoreModuleTransform(SphinxTransform):
|
|||||||
|
|
||||||
def setup(app):
|
def setup(app):
|
||||||
app.add_css_file("customstyle.css")
|
app.add_css_file("customstyle.css")
|
||||||
|
app.add_css_file("filter.css")
|
||||||
|
app.add_js_file("filter.js")
|
||||||
app.add_config_value('redirects_file', 'redirects', 'env')
|
app.add_config_value('redirects_file', 'redirects', 'env')
|
||||||
app.connect('builder-inited', generate_redirects)
|
app.connect('builder-inited', generate_redirects)
|
||||||
app.add_transform(CoreModuleTransform)
|
app.add_transform(CoreModuleTransform)
|
||||||
|
@ -886,10 +886,10 @@ uint16_t bleio_adapter_add_attribute(bleio_adapter_obj_t *adapter, mp_obj_t *att
|
|||||||
uint16_t handle = (uint16_t)adapter->attributes->len;
|
uint16_t handle = (uint16_t)adapter->attributes->len;
|
||||||
mp_obj_list_append(adapter->attributes, attribute);
|
mp_obj_list_append(adapter->attributes, attribute);
|
||||||
|
|
||||||
if (MP_OBJ_IS_TYPE(attribute, &bleio_service_type)) {
|
if (mp_obj_is_type(attribute, &bleio_service_type)) {
|
||||||
adapter->last_added_service_handle = handle;
|
adapter->last_added_service_handle = handle;
|
||||||
}
|
}
|
||||||
if (MP_OBJ_IS_TYPE(attribute, &bleio_characteristic_type)) {
|
if (mp_obj_is_type(attribute, &bleio_characteristic_type)) {
|
||||||
adapter->last_added_characteristic_handle = handle;
|
adapter->last_added_characteristic_handle = handle;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,15 +33,15 @@
|
|||||||
|
|
||||||
|
|
||||||
bleio_uuid_obj_t *bleio_attribute_get_uuid(mp_obj_t *attribute) {
|
bleio_uuid_obj_t *bleio_attribute_get_uuid(mp_obj_t *attribute) {
|
||||||
if (MP_OBJ_IS_TYPE(attribute, &bleio_characteristic_type)) {
|
if (mp_obj_is_type(attribute, &bleio_characteristic_type)) {
|
||||||
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute);
|
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute);
|
||||||
return characteristic->uuid;
|
return characteristic->uuid;
|
||||||
}
|
}
|
||||||
if (MP_OBJ_IS_TYPE(attribute, &bleio_descriptor_type)) {
|
if (mp_obj_is_type(attribute, &bleio_descriptor_type)) {
|
||||||
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute);
|
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute);
|
||||||
return descriptor->uuid;
|
return descriptor->uuid;
|
||||||
}
|
}
|
||||||
if (MP_OBJ_IS_TYPE(attribute, &bleio_service_type)) {
|
if (mp_obj_is_type(attribute, &bleio_service_type)) {
|
||||||
bleio_service_obj_t *service = MP_OBJ_TO_PTR(attribute);
|
bleio_service_obj_t *service = MP_OBJ_TO_PTR(attribute);
|
||||||
return service->uuid;
|
return service->uuid;
|
||||||
}
|
}
|
||||||
|
@ -78,6 +78,10 @@ bleio_service_obj_t *common_hal_bleio_characteristic_get_service(bleio_character
|
|||||||
return self->service;
|
return self->service;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t common_hal_bleio_characteristic_get_max_length(bleio_characteristic_obj_t *self) {
|
||||||
|
return self->max_length;
|
||||||
|
}
|
||||||
|
|
||||||
size_t common_hal_bleio_characteristic_get_value(bleio_characteristic_obj_t *self, uint8_t *buf, size_t len) {
|
size_t common_hal_bleio_characteristic_get_value(bleio_characteristic_obj_t *self, uint8_t *buf, size_t len) {
|
||||||
// Do GATT operations only if this characteristic has been added to a registered service.
|
// Do GATT operations only if this characteristic has been added to a registered service.
|
||||||
if (self->handle != BLE_GATT_HANDLE_INVALID) {
|
if (self->handle != BLE_GATT_HANDLE_INVALID) {
|
||||||
@ -209,9 +213,9 @@ bool bleio_characteristic_set_local_value(bleio_characteristic_obj_t *self, mp_b
|
|||||||
|
|
||||||
self->value = mp_obj_new_bytes(bufinfo->buf, bufinfo->len);
|
self->value = mp_obj_new_bytes(bufinfo->buf, bufinfo->len);
|
||||||
|
|
||||||
if (MP_OBJ_IS_TYPE(self->observer, &bleio_characteristic_buffer_type)) {
|
if (mp_obj_is_type(self->observer, &bleio_characteristic_buffer_type)) {
|
||||||
bleio_characteristic_buffer_update(MP_OBJ_FROM_PTR(self->observer), bufinfo);
|
bleio_characteristic_buffer_update(MP_OBJ_FROM_PTR(self->observer), bufinfo);
|
||||||
} else if (MP_OBJ_IS_TYPE(self->observer, &bleio_packet_buffer_type)) {
|
} else if (mp_obj_is_type(self->observer, &bleio_packet_buffer_type)) {
|
||||||
bleio_packet_buffer_update(MP_OBJ_FROM_PTR(self->observer), bufinfo);
|
bleio_packet_buffer_update(MP_OBJ_FROM_PTR(self->observer), bufinfo);
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
|
@ -644,7 +644,7 @@ void common_hal_bleio_connection_set_connection_interval(bleio_connection_intern
|
|||||||
// mp_obj_t iterable = mp_getiter(service_uuids_whitelist, &iter_buf);
|
// mp_obj_t iterable = mp_getiter(service_uuids_whitelist, &iter_buf);
|
||||||
// mp_obj_t uuid_obj;
|
// mp_obj_t uuid_obj;
|
||||||
// while ((uuid_obj = mp_iternext(iterable)) != MP_OBJ_STOP_ITERATION) {
|
// while ((uuid_obj = mp_iternext(iterable)) != MP_OBJ_STOP_ITERATION) {
|
||||||
// if (!MP_OBJ_IS_TYPE(uuid_obj, &bleio_uuid_type)) {
|
// if (!mp_obj_is_type(uuid_obj, &bleio_uuid_type)) {
|
||||||
// mp_raise_TypeError(translate("non-UUID found in service_uuids_whitelist"));
|
// mp_raise_TypeError(translate("non-UUID found in service_uuids_whitelist"));
|
||||||
// }
|
// }
|
||||||
// bleio_uuid_obj_t *uuid = MP_OBJ_TO_PTR(uuid_obj);
|
// bleio_uuid_obj_t *uuid = MP_OBJ_TO_PTR(uuid_obj);
|
||||||
|
@ -81,7 +81,7 @@ void bleio_packet_buffer_update(bleio_packet_buffer_obj_t *self, mp_buffer_info_
|
|||||||
|
|
||||||
void common_hal_bleio_packet_buffer_construct(
|
void common_hal_bleio_packet_buffer_construct(
|
||||||
bleio_packet_buffer_obj_t *self, bleio_characteristic_obj_t *characteristic,
|
bleio_packet_buffer_obj_t *self, bleio_characteristic_obj_t *characteristic,
|
||||||
size_t buffer_size) {
|
size_t buffer_size, size_t max_packet_size) {
|
||||||
|
|
||||||
self->characteristic = characteristic;
|
self->characteristic = characteristic;
|
||||||
self->client = self->characteristic->service->is_remote;
|
self->client = self->characteristic->service->is_remote;
|
||||||
@ -101,7 +101,7 @@ void common_hal_bleio_packet_buffer_construct(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (incoming) {
|
if (incoming) {
|
||||||
if (!ringbuf_alloc(&self->ringbuf, buffer_size * (sizeof(uint16_t) + characteristic->max_length), false)) {
|
if (!ringbuf_alloc(&self->ringbuf, buffer_size * (sizeof(uint16_t) + max_packet_size), false)) {
|
||||||
mp_raise_ValueError(translate("Buffer too large and unable to allocate"));
|
mp_raise_ValueError(translate("Buffer too large and unable to allocate"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -110,12 +110,13 @@ void common_hal_bleio_packet_buffer_construct(
|
|||||||
self->packet_queued = false;
|
self->packet_queued = false;
|
||||||
self->pending_index = 0;
|
self->pending_index = 0;
|
||||||
self->pending_size = 0;
|
self->pending_size = 0;
|
||||||
self->outgoing[0] = m_malloc(characteristic->max_length, false);
|
self->outgoing[0] = m_malloc(max_packet_size, false);
|
||||||
self->outgoing[1] = m_malloc(characteristic->max_length, false);
|
self->outgoing[1] = m_malloc(max_packet_size, false);
|
||||||
} else {
|
} else {
|
||||||
self->outgoing[0] = NULL;
|
self->outgoing[0] = NULL;
|
||||||
self->outgoing[1] = NULL;
|
self->outgoing[1] = NULL;
|
||||||
}
|
}
|
||||||
|
self->max_packet_size = max_packet_size;
|
||||||
|
|
||||||
bleio_characteristic_set_observer(self->characteristic, self);
|
bleio_characteristic_set_observer(self->characteristic, self);
|
||||||
}
|
}
|
||||||
@ -243,7 +244,8 @@ mp_int_t common_hal_bleio_packet_buffer_get_outgoing_packet_length(bleio_packet_
|
|||||||
if (self->conn_handle != BLE_CONN_HANDLE_INVALID) {
|
if (self->conn_handle != BLE_CONN_HANDLE_INVALID) {
|
||||||
bleio_connection_internal_t *connection = bleio_conn_handle_to_connection(self->conn_handle);
|
bleio_connection_internal_t *connection = bleio_conn_handle_to_connection(self->conn_handle);
|
||||||
if (connection) {
|
if (connection) {
|
||||||
return MIN(common_hal_bleio_connection_get_max_packet_length(connection),
|
return MIN(MIN(common_hal_bleio_connection_get_max_packet_length(connection),
|
||||||
|
self->max_packet_size),
|
||||||
self->characteristic->max_length);
|
self->characteristic->max_length);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -251,7 +253,7 @@ mp_int_t common_hal_bleio_packet_buffer_get_outgoing_packet_length(bleio_packet_
|
|||||||
// we can't tell what the largest outgoing packet length would be.
|
// we can't tell what the largest outgoing packet length would be.
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
return self->characteristic->max_length;
|
return MIN(self->characteristic->max_length, self->max_packet_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool common_hal_bleio_packet_buffer_deinited(bleio_packet_buffer_obj_t *self) {
|
bool common_hal_bleio_packet_buffer_deinited(bleio_packet_buffer_obj_t *self) {
|
||||||
|
@ -42,6 +42,7 @@ typedef struct {
|
|||||||
// We remember the conn_handle so we can do a NOTIFY/INDICATE to a client.
|
// We remember the conn_handle so we can do a NOTIFY/INDICATE to a client.
|
||||||
// We can find out the conn_handle on a Characteristic write or a CCCD write (but not a read).
|
// We can find out the conn_handle on a Characteristic write or a CCCD write (but not a read).
|
||||||
volatile uint16_t conn_handle;
|
volatile uint16_t conn_handle;
|
||||||
|
uint16_t max_packet_size;
|
||||||
uint8_t pending_index;
|
uint8_t pending_index;
|
||||||
uint8_t write_type;
|
uint8_t write_type;
|
||||||
bool client;
|
bool client;
|
||||||
|
@ -539,7 +539,7 @@ void att_remove_connection(uint16_t conn_handle, uint8_t reason) {
|
|||||||
.len = sizeof(zero),
|
.len = sizeof(zero),
|
||||||
};
|
};
|
||||||
|
|
||||||
if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_descriptor_type)) {
|
if (mp_obj_is_type(attribute_obj, &bleio_descriptor_type)) {
|
||||||
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
if (bleio_uuid_get_uuid16_or_unknown(descriptor->uuid) == BLE_UUID_CCCD) {
|
if (bleio_uuid_get_uuid16_or_unknown(descriptor->uuid) == BLE_UUID_CCCD) {
|
||||||
common_hal_bleio_descriptor_set_value(descriptor, &zero_cccd_value);
|
common_hal_bleio_descriptor_set_value(descriptor, &zero_cccd_value);
|
||||||
@ -800,7 +800,7 @@ STATIC void process_find_info_req(uint16_t conn_handle, uint16_t mtu, uint8_t dl
|
|||||||
// Fetch the uuid for the given attribute, which might be a characteristic or a descriptor.
|
// Fetch the uuid for the given attribute, which might be a characteristic or a descriptor.
|
||||||
bleio_uuid_obj_t *uuid;
|
bleio_uuid_obj_t *uuid;
|
||||||
|
|
||||||
if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_characteristic_type)) {
|
if (mp_obj_is_type(attribute_obj, &bleio_characteristic_type)) {
|
||||||
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
if (characteristic->handle != handle) {
|
if (characteristic->handle != handle) {
|
||||||
// If the handles don't match, this is the characteristic definition attribute.
|
// If the handles don't match, this is the characteristic definition attribute.
|
||||||
@ -971,7 +971,7 @@ void process_read_group_req(uint16_t conn_handle, uint16_t mtu, uint8_t dlen, ui
|
|||||||
}
|
}
|
||||||
|
|
||||||
mp_obj_t *attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
mp_obj_t *attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
||||||
if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_service_type)) {
|
if (mp_obj_is_type(attribute_obj, &bleio_service_type)) {
|
||||||
bleio_service_obj_t *service = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_service_obj_t *service = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
|
|
||||||
// Is this a 16-bit or a 128-bit uuid? It must match in size with any previous attribute
|
// Is this a 16-bit or a 128-bit uuid? It must match in size with any previous attribute
|
||||||
@ -1083,7 +1083,7 @@ STATIC void process_read_or_read_blob_req(uint16_t conn_handle, uint16_t mtu, ui
|
|||||||
size_t rsp_length = sizeof(rsp_t);
|
size_t rsp_length = sizeof(rsp_t);
|
||||||
|
|
||||||
mp_obj_t *attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
mp_obj_t *attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
||||||
if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_service_type)) {
|
if (mp_obj_is_type(attribute_obj, &bleio_service_type)) {
|
||||||
if (offset) {
|
if (offset) {
|
||||||
send_error(conn_handle, BT_ATT_ERR_ATTRIBUTE_NOT_LONG, handle, BT_ATT_ERR_INVALID_PDU);
|
send_error(conn_handle, BT_ATT_ERR_ATTRIBUTE_NOT_LONG, handle, BT_ATT_ERR_INVALID_PDU);
|
||||||
return;
|
return;
|
||||||
@ -1095,7 +1095,7 @@ STATIC void process_read_or_read_blob_req(uint16_t conn_handle, uint16_t mtu, ui
|
|||||||
common_hal_bleio_uuid_pack_into(service->uuid, rsp->r.value);
|
common_hal_bleio_uuid_pack_into(service->uuid, rsp->r.value);
|
||||||
rsp_length += sizeof_service_uuid;
|
rsp_length += sizeof_service_uuid;
|
||||||
|
|
||||||
} else if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_characteristic_type)) {
|
} else if (mp_obj_is_type(attribute_obj, &bleio_characteristic_type)) {
|
||||||
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
if (characteristic->decl_handle == handle) {
|
if (characteristic->decl_handle == handle) {
|
||||||
// Read characteristic declaration. Return properties, value handle, and uuid.
|
// Read characteristic declaration. Return properties, value handle, and uuid.
|
||||||
@ -1135,7 +1135,7 @@ STATIC void process_read_or_read_blob_req(uint16_t conn_handle, uint16_t mtu, ui
|
|||||||
memcpy(rsp->r.value, bufinfo.buf + offset, value_length);
|
memcpy(rsp->r.value, bufinfo.buf + offset, value_length);
|
||||||
rsp_length += value_length;
|
rsp_length += value_length;
|
||||||
}
|
}
|
||||||
} else if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_descriptor_type)) {
|
} else if (mp_obj_is_type(attribute_obj, &bleio_descriptor_type)) {
|
||||||
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
|
|
||||||
mp_buffer_info_t bufinfo;
|
mp_buffer_info_t bufinfo;
|
||||||
@ -1203,7 +1203,7 @@ STATIC void process_read_type_req(uint16_t conn_handle, uint16_t mtu, uint8_t dl
|
|||||||
mp_obj_t *attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
mp_obj_t *attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
||||||
|
|
||||||
if (type_uuid == BLE_UUID_CHARACTERISTIC &&
|
if (type_uuid == BLE_UUID_CHARACTERISTIC &&
|
||||||
MP_OBJ_IS_TYPE(attribute_obj, &bleio_characteristic_type)) {
|
mp_obj_is_type(attribute_obj, &bleio_characteristic_type)) {
|
||||||
// Request is for characteristic declarations.
|
// Request is for characteristic declarations.
|
||||||
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
|
|
||||||
@ -1250,7 +1250,7 @@ STATIC void process_read_type_req(uint16_t conn_handle, uint16_t mtu, uint8_t dl
|
|||||||
rsp_length += data_length;
|
rsp_length += data_length;
|
||||||
no_data = false;
|
no_data = false;
|
||||||
|
|
||||||
} else if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_descriptor_type)) {
|
} else if (mp_obj_is_type(attribute_obj, &bleio_descriptor_type)) {
|
||||||
// See if request is for a descriptor value with a 16-bit UUID, such as the CCCD.
|
// See if request is for a descriptor value with a 16-bit UUID, such as the CCCD.
|
||||||
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
if (bleio_uuid_get_uuid16_or_unknown(descriptor->uuid) == type_uuid) {
|
if (bleio_uuid_get_uuid16_or_unknown(descriptor->uuid) == type_uuid) {
|
||||||
@ -1271,7 +1271,7 @@ STATIC void process_read_type_req(uint16_t conn_handle, uint16_t mtu, uint8_t dl
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_characteristic_type)) {
|
} else if (mp_obj_is_type(attribute_obj, &bleio_characteristic_type)) {
|
||||||
// See if request is for a characteristic value with a 16-bit UUID.
|
// See if request is for a characteristic value with a 16-bit UUID.
|
||||||
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
if (bleio_uuid_get_uuid16_or_unknown(characteristic->uuid) == type_uuid) {
|
if (bleio_uuid_get_uuid16_or_unknown(characteristic->uuid) == type_uuid) {
|
||||||
@ -1359,7 +1359,7 @@ STATIC void process_write_req_or_cmd(uint16_t conn_handle, uint16_t mtu, uint8_t
|
|||||||
|
|
||||||
mp_obj_t attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, req->handle);
|
mp_obj_t attribute_obj = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, req->handle);
|
||||||
|
|
||||||
if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_characteristic_type)) {
|
if (mp_obj_is_type(attribute_obj, &bleio_characteristic_type)) {
|
||||||
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_characteristic_obj_t *characteristic = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
|
|
||||||
// Don't write the characteristic declaration.
|
// Don't write the characteristic declaration.
|
||||||
@ -1377,7 +1377,7 @@ STATIC void process_write_req_or_cmd(uint16_t conn_handle, uint16_t mtu, uint8_t
|
|||||||
// Just change the local value. Don't fire off notifications, etc.
|
// Just change the local value. Don't fire off notifications, etc.
|
||||||
bleio_characteristic_set_local_value(characteristic, &bufinfo);
|
bleio_characteristic_set_local_value(characteristic, &bufinfo);
|
||||||
|
|
||||||
} else if (MP_OBJ_IS_TYPE(attribute_obj, &bleio_descriptor_type)) {
|
} else if (mp_obj_is_type(attribute_obj, &bleio_descriptor_type)) {
|
||||||
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
bleio_descriptor_obj_t *descriptor = MP_OBJ_TO_PTR(attribute_obj);
|
||||||
// Only CCCD's are writable.
|
// Only CCCD's are writable.
|
||||||
if (bleio_uuid_get_uuid16_or_unknown(descriptor->uuid) != BLE_UUID_CCCD) {
|
if (bleio_uuid_get_uuid16_or_unknown(descriptor->uuid) != BLE_UUID_CCCD) {
|
||||||
@ -1427,7 +1427,7 @@ STATIC void process_prepare_write_req(uint16_t conn_handle, uint16_t mtu, uint8_
|
|||||||
|
|
||||||
mp_obj_t *attribute = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
mp_obj_t *attribute = bleio_adapter_get_attribute(&common_hal_bleio_adapter_obj, handle);
|
||||||
|
|
||||||
if (!MP_OBJ_IS_TYPE(attribute, &bleio_characteristic_type)) {
|
if (!mp_obj_is_type(attribute, &bleio_characteristic_type)) {
|
||||||
send_error(conn_handle, BT_ATT_OP_PREPARE_WRITE_REQ, handle, BT_ATT_ERR_ATTRIBUTE_NOT_LONG);
|
send_error(conn_handle, BT_ATT_OP_PREPARE_WRITE_REQ, handle, BT_ATT_ERR_ATTRIBUTE_NOT_LONG);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,7 @@ Start libraries with the cookiecutter
|
|||||||
Cookiecutter is a tool that lets you bootstrap a new repo based on another repo.
|
Cookiecutter is a tool that lets you bootstrap a new repo based on another repo.
|
||||||
We've made one `here <https://github.com/adafruit/cookiecutter-adafruit-circuitpython>`_
|
We've made one `here <https://github.com/adafruit/cookiecutter-adafruit-circuitpython>`_
|
||||||
for CircuitPython libraries that include configs for Travis CI and ReadTheDocs
|
for CircuitPython libraries that include configs for Travis CI and ReadTheDocs
|
||||||
along with a setup.py, license, code of conduct and readme.
|
along with a setup.py, license, code of conduct, readme among other files.
|
||||||
|
|
||||||
.. code-block::sh
|
.. code-block::sh
|
||||||
|
|
||||||
@ -238,11 +238,36 @@ Module description
|
|||||||
After the license comment::
|
After the license comment::
|
||||||
|
|
||||||
"""
|
"""
|
||||||
`<module name>` - <Short description>
|
`<module name>`
|
||||||
=================================================
|
=================================================
|
||||||
<Longer description.>
|
|
||||||
|
<Longer description>
|
||||||
|
|
||||||
|
* Author(s):
|
||||||
|
|
||||||
|
Implementation Notes
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
|
||||||
|
**Hardware:**
|
||||||
|
|
||||||
|
* `Adafruit Device Description
|
||||||
|
<hyperlink>`_ (Product ID: <Product Number>)
|
||||||
|
|
||||||
|
**Software and Dependencies:**
|
||||||
|
|
||||||
|
* Adafruit CircuitPython firmware for the supported boards:
|
||||||
|
https://circuitpython.org/downloads
|
||||||
|
|
||||||
|
* Adafruit's Bus Device library:
|
||||||
|
https://github.com/adafruit/Adafruit_CircuitPython_BusDevice
|
||||||
|
|
||||||
|
* Adafruit's Register library:
|
||||||
|
https://github.com/adafruit/Adafruit_CircuitPython_Register
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
Class description
|
Class description
|
||||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
@ -252,7 +277,7 @@ At the class level document what class does and how to initialize it::
|
|||||||
"""DS3231 real-time clock.
|
"""DS3231 real-time clock.
|
||||||
|
|
||||||
:param ~busio.I2C i2c_bus: The I2C bus the DS3231 is connected to.
|
:param ~busio.I2C i2c_bus: The I2C bus the DS3231 is connected to.
|
||||||
:param int address: The I2C address of the device.
|
:param int address: The I2C address of the device. Defaults to :const:`0x40`
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, i2c_bus, address=0x40):
|
def __init__(self, i2c_bus, address=0x40):
|
||||||
@ -267,7 +292,59 @@ Renders as:
|
|||||||
DS3231 real-time clock.
|
DS3231 real-time clock.
|
||||||
|
|
||||||
:param ~busio.I2C i2c_bus: The I2C bus the DS3231 is connected to.
|
:param ~busio.I2C i2c_bus: The I2C bus the DS3231 is connected to.
|
||||||
:param int address: The I2C address of the device.
|
:param int address: The I2C address of the device. Defaults to :const:`0x40`
|
||||||
|
|
||||||
|
|
||||||
|
Documenting Parameters
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
Although there are different ways to document class and functions definitions in Python,
|
||||||
|
the following is the prevalent method of documenting parameters
|
||||||
|
for CircuitPython libraries. When documenting class parameters you should use the
|
||||||
|
following structure:
|
||||||
|
|
||||||
|
.. code-block:: sh
|
||||||
|
|
||||||
|
:param param_type param_name: Parameter_description
|
||||||
|
|
||||||
|
|
||||||
|
param_type
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
The type of the parameter. This could be among other `int`, `float`, `str` `bool`, etc.
|
||||||
|
To document an object in the CircuitPython domain, you need to include a ``~`` before the
|
||||||
|
definition as shown in the following example:
|
||||||
|
|
||||||
|
.. code-block:: sh
|
||||||
|
|
||||||
|
:param ~busio.I2C i2c_bus: The I2C bus the DS3231 is connected to.
|
||||||
|
|
||||||
|
|
||||||
|
To include references to CircuitPython modules, cookiecutter creates an entry in the
|
||||||
|
intersphinx_mapping section in the ``conf.py`` file located within the ``docs`` directory.
|
||||||
|
To add different types outside CircuitPython you need to include them in the intersphinx_mapping::
|
||||||
|
|
||||||
|
|
||||||
|
intersphinx_mapping = {
|
||||||
|
"python": ("https://docs.python.org/3.4", None),
|
||||||
|
"BusDevice":("https://circuitpython.readthedocs.io/projects/busdevice/en/latest/", None,),
|
||||||
|
"CircuitPython": ("https://circuitpython.readthedocs.io/en/latest/", None),
|
||||||
|
}
|
||||||
|
|
||||||
|
The intersphinx_mapping above includes references to Python, BusDevice and CircuitPython
|
||||||
|
Documentation
|
||||||
|
|
||||||
|
|
||||||
|
param_name
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
Parameter name used in the class or method definition
|
||||||
|
|
||||||
|
|
||||||
|
Parameter_description
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
Parameter description. When the parameter defaults to a particular value, it is good
|
||||||
|
practice to include the default::
|
||||||
|
|
||||||
|
:param int pitch: Pitch value for the servo. Defaults to :const:`4500`
|
||||||
|
|
||||||
|
|
||||||
Attributes
|
Attributes
|
||||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
@ -383,6 +460,14 @@ Renders as:
|
|||||||
|
|
||||||
:param float degrees: Degrees to turn right
|
:param float degrees: Degrees to turn right
|
||||||
|
|
||||||
|
Documentation References to other Libraries
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
When you need to make references to documentation in other libraries you should refer the class using single
|
||||||
|
backticks ``:class:`~adafruit_motor.servo.Servo```. You must also add the reference in the ``conf.py`` file in the
|
||||||
|
``intersphinx_mapping section`` by adding a new entry::
|
||||||
|
|
||||||
|
"adafruit_motor": ("https://circuitpython.readthedocs.io/projects/motor/en/latest/", None,),
|
||||||
|
|
||||||
Use BusDevice
|
Use BusDevice
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
@ -442,6 +527,50 @@ SPI Example
|
|||||||
spi.readinto(self.buf)
|
spi.readinto(self.buf)
|
||||||
return self.buf[0]
|
return self.buf[0]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Class documentation example template
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
When documenting classes, you should use the following template to illustrate basic usage.
|
||||||
|
It is similar with the simpletest example, however this will display the information in the Read The Docs
|
||||||
|
documentation.
|
||||||
|
The advantage of using this template is it makes the documentation consistent across the libraries.
|
||||||
|
|
||||||
|
This is an example for a AHT20 temperature sensor. Include the following after the class parameter:
|
||||||
|
|
||||||
|
|
||||||
|
.. code-block:: python
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
**Quickstart: Importing and using the AHT10/AHT20 temperature sensor**
|
||||||
|
|
||||||
|
Here is an example of using the :class:`AHTx0` class.
|
||||||
|
First you will need to import the libraries to use the sensor
|
||||||
|
|
||||||
|
.. code-block:: python
|
||||||
|
|
||||||
|
import board
|
||||||
|
import adafruit_ahtx0
|
||||||
|
|
||||||
|
Once this is done you can define your `board.I2C` object and define your sensor object
|
||||||
|
|
||||||
|
.. code-block:: python
|
||||||
|
|
||||||
|
i2c = board.I2C() # uses board.SCL and board.SDA
|
||||||
|
aht = adafruit_ahtx0.AHTx0(i2c)
|
||||||
|
|
||||||
|
Now you have access to the temperature and humidity using
|
||||||
|
the :attr:`temperature` and :attr:`relative_humidity` attributes
|
||||||
|
|
||||||
|
.. code-block:: python
|
||||||
|
|
||||||
|
temperature = aht.temperature
|
||||||
|
relative_humidity = aht.relative_humidity
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
Use composition
|
Use composition
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
@ -458,10 +587,10 @@ object instead of the pins themselves. This allows the calling code to provide
|
|||||||
any object with the appropriate methods such as an I2C expansion board.
|
any object with the appropriate methods such as an I2C expansion board.
|
||||||
|
|
||||||
Another example is to expect a :py:class:`~digitalio.DigitalInOut` for a pin to
|
Another example is to expect a :py:class:`~digitalio.DigitalInOut` for a pin to
|
||||||
toggle instead of a :py:class:`~microcontroller.Pin` from `board`. Taking in the
|
toggle instead of a :py:class:`~microcontroller.Pin` from :py:mod:`board`.
|
||||||
:py:class:`~microcontroller.Pin` object alone would limit the driver to pins on
|
Taking in the :py:class:`~microcontroller.Pin` object alone would limit the
|
||||||
the actual microcontroller instead of pins provided by another driver such as an
|
driver to pins on the actual microcontroller instead of pins provided by another
|
||||||
IO expander.
|
driver such as an IO expander.
|
||||||
|
|
||||||
Lots of small modules
|
Lots of small modules
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
@ -516,6 +645,23 @@ when using ``const()``, keep in mind these general guide lines:
|
|||||||
- If user will not need access to variable, prefix name with a leading
|
- If user will not need access to variable, prefix name with a leading
|
||||||
underscore, ex: ``_SOME_CONST``.
|
underscore, ex: ``_SOME_CONST``.
|
||||||
|
|
||||||
|
Libraries Examples
|
||||||
|
------------------
|
||||||
|
When adding examples, cookiecutter will add a ``<name>_simpletest.py`` file in the examples directory for you.
|
||||||
|
Be sure to include code with the library minimal functionalities to work on a device.
|
||||||
|
You could other examples if needed featuring different
|
||||||
|
functionalities of the library.
|
||||||
|
If you add additional examples, be sure to include them in the ``examples.rst``. Naming of the examples
|
||||||
|
files should use the name of the library followed by a description, using underscore to separate them.
|
||||||
|
When using print statements you should use the ``" ".format()`` format, as there are particular boards
|
||||||
|
that are not capable to use f-strings.
|
||||||
|
|
||||||
|
.. code-block:: python
|
||||||
|
|
||||||
|
text_to_display = "World!"
|
||||||
|
|
||||||
|
print("Hello {}".format(text_to_display))
|
||||||
|
|
||||||
Sensor properties and units
|
Sensor properties and units
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
@ -536,7 +682,7 @@ properties.
|
|||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``gyro`` | (float, float, float) | x, y, z radians per second |
|
| ``gyro`` | (float, float, float) | x, y, z radians per second |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``temperature`` | float | degrees centigrade |
|
| ``temperature`` | float | degrees Celsius |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``CO2`` | float | measured CO2 in ppm |
|
| ``CO2`` | float | measured CO2 in ppm |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
@ -544,9 +690,9 @@ properties.
|
|||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``TVOC`` | float | Total Volatile Organic Compounds in ppb |
|
| ``TVOC`` | float | Total Volatile Organic Compounds in ppb |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``distance`` | float | centimeters |
|
| ``distance`` | float | centimeters (cm) |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``proximity`` | int | non-unit-specifc proximity values (monotonic but not actual distance) |
|
| ``proximity`` | int | non-unit-specific proximity values (monotonic but not actual distance) |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``light`` | float | non-unit-specific light levels (should be monotonic but is not lux) |
|
| ``light`` | float | non-unit-specific light levels (should be monotonic but is not lux) |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
@ -568,7 +714,7 @@ properties.
|
|||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``duty_cycle`` | int | 16-bit PWM duty cycle (regardless of output resolution) |
|
| ``duty_cycle`` | int | 16-bit PWM duty cycle (regardless of output resolution) |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``frequency`` | int | Hertz |
|
| ``frequency`` | int | Hertz (Hz) |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
| ``value`` | bool | Digital logic |
|
| ``value`` | bool | Digital logic |
|
||||||
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
+-----------------------+-----------------------+-------------------------------------------------------------------------+
|
||||||
@ -592,9 +738,8 @@ mimic the structure in ``shared-bindings``.
|
|||||||
To test your native modules or core enhancements, follow these Adafruit Learning Guides
|
To test your native modules or core enhancements, follow these Adafruit Learning Guides
|
||||||
for building local firmware to flash onto your device(s):
|
for building local firmware to flash onto your device(s):
|
||||||
|
|
||||||
`SAMD21 - Build Firmware Learning Guide <https://learn.adafruit.com/micropython-for-samd21/build-firmware>`_
|
`Build CircuitPython <https://learn.adafruit.com/building-circuitpython>`_
|
||||||
|
|
||||||
`ESP8266 - Build Firmware Learning Guide <https://learn.adafruit.com/building-and-running-micropython-on-the-esp8266/overview>`_
|
|
||||||
|
|
||||||
MicroPython compatibility
|
MicroPython compatibility
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
@ -36,6 +36,7 @@ Full Table of Contents
|
|||||||
:caption: MicroPython specific
|
:caption: MicroPython specific
|
||||||
|
|
||||||
library/index.rst
|
library/index.rst
|
||||||
|
reference/glossary.rst
|
||||||
|
|
||||||
.. toctree::
|
.. toctree::
|
||||||
:maxdepth: 1
|
:maxdepth: 1
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
:mod:`array` -- arrays of numeric data
|
:mod:`array` -- arrays of numeric data
|
||||||
======================================
|
=======================================
|
||||||
|
|
||||||
.. module:: array
|
.. module:: array
|
||||||
:synopsis: efficient arrays of numeric data
|
:synopsis: efficient arrays of numeric data
|
||||||
@ -13,7 +13,7 @@ floating-point support).
|
|||||||
Classes
|
Classes
|
||||||
-------
|
-------
|
||||||
|
|
||||||
.. class:: array.array(typecode, [iterable])
|
.. class:: array(typecode, [iterable])
|
||||||
|
|
||||||
Create array with elements of given type. Initial contents of the
|
Create array with elements of given type. Initial contents of the
|
||||||
array are given by an `iterable`. If it is not provided, an empty
|
array are given by an `iterable`. If it is not provided, an empty
|
||||||
|
@ -14,13 +14,11 @@ Functions
|
|||||||
|
|
||||||
.. function:: hexlify(data, [sep])
|
.. function:: hexlify(data, [sep])
|
||||||
|
|
||||||
Convert binary data to hexadecimal representation. Returns bytes string.
|
Convert the bytes in the *data* object to a hexadecimal representation.
|
||||||
|
Returns a bytes object.
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
If the additional argument *sep* is supplied it is used as a separator
|
||||||
:class: attention
|
between hexadecimal values.
|
||||||
|
|
||||||
If additional argument, *sep* is supplied, it is used as a separator
|
|
||||||
between hexadecimal values.
|
|
||||||
|
|
||||||
.. function:: unhexlify(data)
|
.. function:: unhexlify(data)
|
||||||
|
|
||||||
|
@ -78,7 +78,7 @@ Example::
|
|||||||
Functions
|
Functions
|
||||||
---------
|
---------
|
||||||
|
|
||||||
.. function:: open(stream, \*, flags=0, pagesize=0, cachesize=0, minkeypage=0)
|
.. function:: open(stream, *, flags=0, pagesize=0, cachesize=0, minkeypage=0)
|
||||||
|
|
||||||
Open a database from a random-access ``stream`` (like an open file). All
|
Open a database from a random-access ``stream`` (like an open file). All
|
||||||
other parameters are optional and keyword-only, and allow to tweak advanced
|
other parameters are optional and keyword-only, and allow to tweak advanced
|
||||||
@ -118,9 +118,9 @@ Methods
|
|||||||
Flush any data in cache to the underlying stream.
|
Flush any data in cache to the underlying stream.
|
||||||
|
|
||||||
.. method:: btree.__getitem__(key)
|
.. method:: btree.__getitem__(key)
|
||||||
btree.get(key, default=None)
|
btree.get(key, default=None, /)
|
||||||
btree.__setitem__(key, val)
|
btree.__setitem__(key, val)
|
||||||
btree.__detitem__(key)
|
btree.__delitem__(key)
|
||||||
btree.__contains__(key)
|
btree.__contains__(key)
|
||||||
|
|
||||||
Standard dictionary methods.
|
Standard dictionary methods.
|
||||||
|
@ -14,6 +14,33 @@ hold/accumulate various objects.
|
|||||||
Classes
|
Classes
|
||||||
-------
|
-------
|
||||||
|
|
||||||
|
.. function:: deque(iterable, maxlen[, flags])
|
||||||
|
|
||||||
|
Deques (double-ended queues) are a list-like container that support O(1)
|
||||||
|
appends and pops from either side of the deque. New deques are created
|
||||||
|
using the following arguments:
|
||||||
|
|
||||||
|
- *iterable* must be the empty tuple, and the new deque is created empty.
|
||||||
|
|
||||||
|
- *maxlen* must be specified and the deque will be bounded to this
|
||||||
|
maximum length. Once the deque is full, any new items added will
|
||||||
|
discard items from the opposite end.
|
||||||
|
|
||||||
|
- The optional *flags* can be 1 to check for overflow when adding items.
|
||||||
|
|
||||||
|
As well as supporting `bool` and `len`, deque objects have the following
|
||||||
|
methods:
|
||||||
|
|
||||||
|
.. method:: deque.append(x)
|
||||||
|
|
||||||
|
Add *x* to the right side of the deque.
|
||||||
|
Raises IndexError if overflow checking is enabled and there is no more room left.
|
||||||
|
|
||||||
|
.. method:: deque.popleft()
|
||||||
|
|
||||||
|
Remove and return an item from the left side of the deque.
|
||||||
|
Raises IndexError if no items are present.
|
||||||
|
|
||||||
.. function:: namedtuple(name, fields)
|
.. function:: namedtuple(name, fields)
|
||||||
|
|
||||||
This is factory function to create a new namedtuple type with a specific
|
This is factory function to create a new namedtuple type with a specific
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
|see_cpython_module| :mod:`cpython:errno`.
|
|see_cpython_module| :mod:`cpython:errno`.
|
||||||
|
|
||||||
This module provides access to symbolic error codes for `OSError` exception.
|
This module provides access to symbolic error codes for `OSError` exception.
|
||||||
|
A particular inventory of codes depends on :term:`MicroPython port`.
|
||||||
|
|
||||||
Constants
|
Constants
|
||||||
---------
|
---------
|
||||||
@ -14,7 +15,8 @@ Constants
|
|||||||
.. data:: EEXIST, EAGAIN, etc.
|
.. data:: EEXIST, EAGAIN, etc.
|
||||||
|
|
||||||
Error codes, based on ANSI C/POSIX standard. All error codes start with
|
Error codes, based on ANSI C/POSIX standard. All error codes start with
|
||||||
"E". Errors are usually accessible as ``exc.args[0]``
|
"E". As mentioned above, inventory of the codes depends on
|
||||||
|
:term:`MicroPython port`. Errors are usually accessible as ``exc.args[0]``
|
||||||
where ``exc`` is an instance of `OSError`. Usage example::
|
where ``exc`` is an instance of `OSError`. Usage example::
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -28,5 +30,5 @@ Constants
|
|||||||
Dictionary mapping numeric error codes to strings with symbolic error
|
Dictionary mapping numeric error codes to strings with symbolic error
|
||||||
code (see above)::
|
code (see above)::
|
||||||
|
|
||||||
>>> print(errno.errorcode[uerrno.EEXIST])
|
>>> print(errno.errorcode[errno.EEXIST])
|
||||||
EEXIST
|
EEXIST
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
:mod:`framebuf` --- Frame buffer manipulation
|
:mod:`framebuf` --- frame buffer manipulation
|
||||||
=============================================
|
=============================================
|
||||||
|
|
||||||
.. include:: ../templates/unsupported_in_circuitpython.inc
|
.. include:: ../templates/unsupported_in_circuitpython.inc
|
||||||
@ -21,7 +21,7 @@ For example::
|
|||||||
import framebuf
|
import framebuf
|
||||||
|
|
||||||
# FrameBuffer needs 2 bytes for every RGB565 pixel
|
# FrameBuffer needs 2 bytes for every RGB565 pixel
|
||||||
fbuf = FrameBuffer(bytearray(10 * 100 * 2), 10, 100, framebuf.RGB565)
|
fbuf = framebuf.FrameBuffer(bytearray(10 * 100 * 2), 10, 100, framebuf.RGB565)
|
||||||
|
|
||||||
fbuf.fill(0)
|
fbuf.fill(0)
|
||||||
fbuf.text('MicroPython!', 0, 0, 0xffff)
|
fbuf.text('MicroPython!', 0, 0, 0xffff)
|
||||||
@ -30,7 +30,7 @@ For example::
|
|||||||
Constructors
|
Constructors
|
||||||
------------
|
------------
|
||||||
|
|
||||||
.. class:: FrameBuffer(buffer, width, height, format, stride=width)
|
.. class:: FrameBuffer(buffer, width, height, format, stride=width, /)
|
||||||
|
|
||||||
Construct a FrameBuffer object. The parameters are:
|
Construct a FrameBuffer object. The parameters are:
|
||||||
|
|
||||||
@ -132,7 +132,7 @@ Constants
|
|||||||
|
|
||||||
Monochrome (1-bit) color format
|
Monochrome (1-bit) color format
|
||||||
This defines a mapping where the bits in a byte are horizontally mapped.
|
This defines a mapping where the bits in a byte are horizontally mapped.
|
||||||
Each byte occupies 8 horizontal pixels with bit 0 being the leftmost.
|
Each byte occupies 8 horizontal pixels with bit 7 being the leftmost.
|
||||||
Subsequent bytes appear at successive horizontal locations until the
|
Subsequent bytes appear at successive horizontal locations until the
|
||||||
rightmost edge is reached. Further bytes are rendered on the next row, one
|
rightmost edge is reached. Further bytes are rendered on the next row, one
|
||||||
pixel lower.
|
pixel lower.
|
||||||
@ -141,7 +141,7 @@ Constants
|
|||||||
|
|
||||||
Monochrome (1-bit) color format
|
Monochrome (1-bit) color format
|
||||||
This defines a mapping where the bits in a byte are horizontally mapped.
|
This defines a mapping where the bits in a byte are horizontally mapped.
|
||||||
Each byte occupies 8 horizontal pixels with bit 7 being the leftmost.
|
Each byte occupies 8 horizontal pixels with bit 0 being the leftmost.
|
||||||
Subsequent bytes appear at successive horizontal locations until the
|
Subsequent bytes appear at successive horizontal locations until the
|
||||||
rightmost edge is reached. Further bytes are rendered on the next row, one
|
rightmost edge is reached. Further bytes are rendered on the next row, one
|
||||||
pixel lower.
|
pixel lower.
|
||||||
|
@ -37,10 +37,9 @@ with the ``u`` prefix dropped:
|
|||||||
json.rst
|
json.rst
|
||||||
re.rst
|
re.rst
|
||||||
sys.rst
|
sys.rst
|
||||||
|
uasyncio.rst
|
||||||
uctypes.rst
|
uctypes.rst
|
||||||
uselect.rst
|
uselect.rst
|
||||||
usocket.rst
|
|
||||||
ussl.rst
|
|
||||||
uzlib.rst
|
uzlib.rst
|
||||||
|
|
||||||
Omitted functions in the ``string`` library
|
Omitted functions in the ``string`` library
|
||||||
|
@ -73,14 +73,26 @@ Functions
|
|||||||
|
|
||||||
.. function:: heap_lock()
|
.. function:: heap_lock()
|
||||||
.. function:: heap_unlock()
|
.. function:: heap_unlock()
|
||||||
|
.. function:: heap_locked()
|
||||||
|
|
||||||
Lock or unlock the heap. When locked no memory allocation can occur and a
|
Lock or unlock the heap. When locked no memory allocation can occur and a
|
||||||
`MemoryError` will be raised if any heap allocation is attempted.
|
`MemoryError` will be raised if any heap allocation is attempted.
|
||||||
|
`heap_locked()` returns a true value if the heap is currently locked.
|
||||||
|
|
||||||
These functions can be nested, ie `heap_lock()` can be called multiple times
|
These functions can be nested, ie `heap_lock()` can be called multiple times
|
||||||
in a row and the lock-depth will increase, and then `heap_unlock()` must be
|
in a row and the lock-depth will increase, and then `heap_unlock()` must be
|
||||||
called the same number of times to make the heap available again.
|
called the same number of times to make the heap available again.
|
||||||
|
|
||||||
|
Both `heap_unlock()` and `heap_locked()` return the current lock depth
|
||||||
|
(after unlocking for the former) as a non-negative integer, with 0 meaning
|
||||||
|
the heap is not locked.
|
||||||
|
|
||||||
|
If the REPL becomes active with the heap locked then it will be forcefully
|
||||||
|
unlocked.
|
||||||
|
|
||||||
|
Note: `heap_locked()` is not enabled on most ports by default,
|
||||||
|
requires ``MICROPY_PY_MICROPYTHON_HEAP_LOCKED``.
|
||||||
|
|
||||||
.. function:: kbd_intr(chr)
|
.. function:: kbd_intr(chr)
|
||||||
|
|
||||||
Set the character that will raise a `KeyboardInterrupt` exception. By
|
Set the character that will raise a `KeyboardInterrupt` exception. By
|
||||||
@ -91,3 +103,36 @@ Functions
|
|||||||
This function can be used to prevent the capturing of Ctrl-C on the
|
This function can be used to prevent the capturing of Ctrl-C on the
|
||||||
incoming stream of characters that is usually used for the REPL, in case
|
incoming stream of characters that is usually used for the REPL, in case
|
||||||
that stream is used for other purposes.
|
that stream is used for other purposes.
|
||||||
|
|
||||||
|
.. function:: schedule(func, arg)
|
||||||
|
|
||||||
|
Schedule the function *func* to be executed "very soon". The function
|
||||||
|
is passed the value *arg* as its single argument. "Very soon" means that
|
||||||
|
the MicroPython runtime will do its best to execute the function at the
|
||||||
|
earliest possible time, given that it is also trying to be efficient, and
|
||||||
|
that the following conditions hold:
|
||||||
|
|
||||||
|
- A scheduled function will never preempt another scheduled function.
|
||||||
|
- Scheduled functions are always executed "between opcodes" which means
|
||||||
|
that all fundamental Python operations (such as appending to a list)
|
||||||
|
are guaranteed to be atomic.
|
||||||
|
- A given port may define "critical regions" within which scheduled
|
||||||
|
functions will never be executed. Functions may be scheduled within
|
||||||
|
a critical region but they will not be executed until that region
|
||||||
|
is exited. An example of a critical region is a preempting interrupt
|
||||||
|
handler (an IRQ).
|
||||||
|
|
||||||
|
A use for this function is to schedule a callback from a preempting IRQ.
|
||||||
|
Such an IRQ puts restrictions on the code that runs in the IRQ (for example
|
||||||
|
the heap may be locked) and scheduling a function to call later will lift
|
||||||
|
those restrictions.
|
||||||
|
|
||||||
|
Note: If `schedule()` is called from a preempting IRQ, when memory
|
||||||
|
allocation is not allowed and the callback to be passed to `schedule()` is
|
||||||
|
a bound method, passing this directly will fail. This is because creating a
|
||||||
|
reference to a bound method causes memory allocation. A solution is to
|
||||||
|
create a reference to the method in the class constructor and to pass that
|
||||||
|
reference to `schedule()`.
|
||||||
|
|
||||||
|
There is a finite queue to hold the scheduled functions and `schedule()`
|
||||||
|
will raise a `RuntimeError` if the queue is full.
|
||||||
|
@ -10,41 +10,101 @@ This module implements regular expression operations. Regular expression
|
|||||||
syntax supported is a subset of CPython ``re`` module (and actually is
|
syntax supported is a subset of CPython ``re`` module (and actually is
|
||||||
a subset of POSIX extended regular expressions).
|
a subset of POSIX extended regular expressions).
|
||||||
|
|
||||||
Supported operators are:
|
Supported operators and special sequences are:
|
||||||
|
|
||||||
``'.'``
|
``.``
|
||||||
Match any character.
|
Match any character.
|
||||||
|
|
||||||
``'[...]'``
|
``[...]``
|
||||||
Match set of characters. Individual characters and ranges are supported,
|
Match set of characters. Individual characters and ranges are supported,
|
||||||
including negated sets (e.g. ``[^a-c]``).
|
including negated sets (e.g. ``[^a-c]``).
|
||||||
|
|
||||||
``'^'``
|
``^``
|
||||||
|
Match the start of the string.
|
||||||
|
|
||||||
``'$'``
|
``$``
|
||||||
|
Match the end of the string.
|
||||||
|
|
||||||
``'?'``
|
``?``
|
||||||
|
Match zero or one of the previous sub-pattern.
|
||||||
|
|
||||||
``'*'``
|
``*``
|
||||||
|
Match zero or more of the previous sub-pattern.
|
||||||
|
|
||||||
``'+'``
|
``+``
|
||||||
|
Match one or more of the previous sub-pattern.
|
||||||
|
|
||||||
``'??'``
|
``??``
|
||||||
|
Non-greedy version of ``?``, match zero or one, with the preference
|
||||||
|
for zero.
|
||||||
|
|
||||||
``'*?'``
|
``*?``
|
||||||
|
Non-greedy version of ``*``, match zero or more, with the preference
|
||||||
|
for the shortest match.
|
||||||
|
|
||||||
``'+?'``
|
``+?``
|
||||||
|
Non-greedy version of ``+``, match one or more, with the preference
|
||||||
|
for the shortest match.
|
||||||
|
|
||||||
``'|'``
|
``|``
|
||||||
|
Match either the left-hand side or the right-hand side sub-patterns of
|
||||||
|
this operator.
|
||||||
|
|
||||||
``'(...)'``
|
``(...)``
|
||||||
Grouping. Each group is capturing (a substring it captures can be accessed
|
Grouping. Each group is capturing (a substring it captures can be accessed
|
||||||
with `match.group()` method).
|
with `match.group()` method).
|
||||||
|
|
||||||
**NOT SUPPORTED**: Counted repetitions (``{m,n}``), more advanced assertions
|
``\d``
|
||||||
(``\b``, ``\B``), named groups (``(?P<name>...)``), non-capturing groups
|
Matches digit. Equivalent to ``[0-9]``.
|
||||||
(``(?:...)``), etc.
|
|
||||||
|
|
||||||
|
``\D``
|
||||||
|
Matches non-digit. Equivalent to ``[^0-9]``.
|
||||||
|
|
||||||
|
``\s``
|
||||||
|
Matches whitespace. Equivalent to ``[ \t-\r]``.
|
||||||
|
|
||||||
|
``\S``
|
||||||
|
Matches non-whitespace. Equivalent to ``[^ \t-\r]``.
|
||||||
|
|
||||||
|
``\w``
|
||||||
|
Matches "word characters" (ASCII only). Equivalent to ``[A-Za-z0-9_]``.
|
||||||
|
|
||||||
|
``\W``
|
||||||
|
Matches non "word characters" (ASCII only). Equivalent to ``[^A-Za-z0-9_]``.
|
||||||
|
|
||||||
|
``\``
|
||||||
|
Escape character. Any other character following the backslash, except
|
||||||
|
for those listed above, is taken literally. For example, ``\*`` is
|
||||||
|
equivalent to literal ``*`` (not treated as the ``*`` operator).
|
||||||
|
Note that ``\r``, ``\n``, etc. are not handled specially, and will be
|
||||||
|
equivalent to literal letters ``r``, ``n``, etc. Due to this, it's
|
||||||
|
not recommended to use raw Python strings (``r""``) for regular
|
||||||
|
expressions. For example, ``r"\r\n"`` when used as the regular
|
||||||
|
expression is equivalent to ``"rn"``. To match CR character followed
|
||||||
|
by LF, use ``"\r\n"``.
|
||||||
|
|
||||||
|
**NOT SUPPORTED**:
|
||||||
|
|
||||||
|
* counted repetitions (``{m,n}``)
|
||||||
|
* named groups (``(?P<name>...)``)
|
||||||
|
* non-capturing groups (``(?:...)``)
|
||||||
|
* more advanced assertions (``\b``, ``\B``)
|
||||||
|
* special character escapes like ``\r``, ``\n`` - use Python's own escaping
|
||||||
|
instead
|
||||||
|
* etc.
|
||||||
|
|
||||||
|
Example::
|
||||||
|
|
||||||
|
import ure
|
||||||
|
|
||||||
|
# As ure doesn't support escapes itself, use of r"" strings is not
|
||||||
|
# recommended.
|
||||||
|
regex = ure.compile("[\r\n]")
|
||||||
|
|
||||||
|
regex.split("line1\rline2\nline3\r\n")
|
||||||
|
|
||||||
|
# Result:
|
||||||
|
# ['line1', 'line2', 'line3', '', '']
|
||||||
|
|
||||||
Functions
|
Functions
|
||||||
---------
|
---------
|
||||||
@ -64,9 +124,26 @@ Functions
|
|||||||
string for first position which matches regex (which still may be
|
string for first position which matches regex (which still may be
|
||||||
0 if regex is anchored).
|
0 if regex is anchored).
|
||||||
|
|
||||||
|
.. function:: sub(regex_str, replace, string, count=0, flags=0, /)
|
||||||
|
|
||||||
|
Compile *regex_str* and search for it in *string*, replacing all matches
|
||||||
|
with *replace*, and returning the new string.
|
||||||
|
|
||||||
|
*replace* can be a string or a function. If it is a string then escape
|
||||||
|
sequences of the form ``\<number>`` and ``\g<number>`` can be used to
|
||||||
|
expand to the corresponding group (or an empty string for unmatched groups).
|
||||||
|
If *replace* is a function then it must take a single argument (the match)
|
||||||
|
and should return a replacement string.
|
||||||
|
|
||||||
|
If *count* is specified and non-zero then substitution will stop after
|
||||||
|
this many substitutions are made. The *flags* argument is ignored.
|
||||||
|
|
||||||
|
Note: availability of this function depends on :term:`MicroPython port`.
|
||||||
|
|
||||||
.. data:: DEBUG
|
.. data:: DEBUG
|
||||||
|
|
||||||
Flag value, display debug information about compiled expression.
|
Flag value, display debug information about compiled expression.
|
||||||
|
(Availability depends on :term:`MicroPython port`.)
|
||||||
|
|
||||||
|
|
||||||
.. _regex:
|
.. _regex:
|
||||||
@ -79,12 +156,14 @@ Compiled regular expression. Instances of this class are created using
|
|||||||
|
|
||||||
.. method:: regex.match(string)
|
.. method:: regex.match(string)
|
||||||
regex.search(string)
|
regex.search(string)
|
||||||
|
regex.sub(replace, string, count=0, flags=0, /)
|
||||||
|
|
||||||
Similar to the module-level functions :meth:`match` and :meth:`search`.
|
Similar to the module-level functions :meth:`match`, :meth:`search`
|
||||||
|
and :meth:`sub`.
|
||||||
Using methods is (much) more efficient if the same regex is applied to
|
Using methods is (much) more efficient if the same regex is applied to
|
||||||
multiple strings.
|
multiple strings.
|
||||||
|
|
||||||
.. method:: regex.split(string, max_split=-1)
|
.. method:: regex.split(string, max_split=-1, /)
|
||||||
|
|
||||||
Split a *string* using regex. If *max_split* is given, it specifies
|
Split a *string* using regex. If *max_split* is given, it specifies
|
||||||
maximum number of splits to perform. Returns list of strings (there
|
maximum number of splits to perform. Returns list of strings (there
|
||||||
@ -93,9 +172,31 @@ Compiled regular expression. Instances of this class are created using
|
|||||||
Match objects
|
Match objects
|
||||||
-------------
|
-------------
|
||||||
|
|
||||||
Match objects as returned by `match()` and `search()` methods.
|
Match objects as returned by `match()` and `search()` methods, and passed
|
||||||
|
to the replacement function in `sub()`.
|
||||||
|
|
||||||
.. method:: match.group([index])
|
.. method:: match.group(index)
|
||||||
|
|
||||||
Return matching (sub)string. *index* is 0 for entire match,
|
Return matching (sub)string. *index* is 0 for entire match,
|
||||||
1 and above for each capturing group. Only numeric groups are supported.
|
1 and above for each capturing group. Only numeric groups are supported.
|
||||||
|
|
||||||
|
.. method:: match.groups()
|
||||||
|
|
||||||
|
Return a tuple containing all the substrings of the groups of the match.
|
||||||
|
|
||||||
|
Note: availability of this method depends on :term:`MicroPython port`.
|
||||||
|
|
||||||
|
.. method:: match.start([index])
|
||||||
|
match.end([index])
|
||||||
|
|
||||||
|
Return the index in the original string of the start or end of the
|
||||||
|
substring group that was matched. *index* defaults to the entire
|
||||||
|
group, otherwise it will select a group.
|
||||||
|
|
||||||
|
Note: availability of these methods depends on :term:`MicroPython port`.
|
||||||
|
|
||||||
|
.. method:: match.span([index])
|
||||||
|
|
||||||
|
Returns the 2-tuple ``(match.start(index), match.end(index))``.
|
||||||
|
|
||||||
|
Note: availability of this method depends on :term:`MicroPython port`.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
:mod:`sys` -- system specific functions
|
:mod:`sys` -- system specific functions
|
||||||
=======================================
|
========================================
|
||||||
|
|
||||||
.. include:: ../templates/unsupported_in_circuitpython.inc
|
.. include:: ../templates/unsupported_in_circuitpython.inc
|
||||||
|
|
||||||
@ -11,26 +11,12 @@
|
|||||||
Functions
|
Functions
|
||||||
---------
|
---------
|
||||||
|
|
||||||
.. function:: exit(retval=0)
|
.. function:: exit(retval=0, /)
|
||||||
|
|
||||||
Terminate current program with a given exit code. Underlyingly, this
|
Terminate current program with a given exit code. Underlyingly, this
|
||||||
function raise as `SystemExit` exception. If an argument is given, its
|
function raise as `SystemExit` exception. If an argument is given, its
|
||||||
value given as an argument to `SystemExit`.
|
value given as an argument to `SystemExit`.
|
||||||
|
|
||||||
.. function:: print_exception(exc, file=sys.stdout)
|
|
||||||
|
|
||||||
Print exception with a traceback to a file-like object *file* (or
|
|
||||||
`sys.stdout` by default).
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
This is simplified version of a function which appears in the
|
|
||||||
``traceback`` module in CPython. Unlike ``traceback.print_exception()``,
|
|
||||||
this function takes just exception value instead of exception type,
|
|
||||||
exception value, and traceback object; *file* argument should be
|
|
||||||
positional; further arguments are not supported.
|
|
||||||
|
|
||||||
Constants
|
Constants
|
||||||
---------
|
---------
|
||||||
|
|
||||||
@ -122,3 +108,9 @@ Constants
|
|||||||
.. data:: version_info
|
.. data:: version_info
|
||||||
|
|
||||||
Python language version that this implementation conforms to, as a tuple of ints.
|
Python language version that this implementation conforms to, as a tuple of ints.
|
||||||
|
|
||||||
|
.. admonition:: Difference to CPython
|
||||||
|
:class: attention
|
||||||
|
|
||||||
|
Only the first three version numbers (major, minor, micro) are supported and
|
||||||
|
they can be referenced only by index, not by name.
|
||||||
|
307
docs/library/uasyncio.rst
Normal file
307
docs/library/uasyncio.rst
Normal file
@ -0,0 +1,307 @@
|
|||||||
|
:mod:`uasyncio` --- asynchronous I/O scheduler
|
||||||
|
==============================================
|
||||||
|
|
||||||
|
.. module:: uasyncio
|
||||||
|
:synopsis: asynchronous I/O scheduler for writing concurrent code
|
||||||
|
|
||||||
|
|see_cpython_module|
|
||||||
|
`asyncio <https://docs.python.org/3.8/library/asyncio.html>`_
|
||||||
|
|
||||||
|
Example::
|
||||||
|
|
||||||
|
import uasyncio
|
||||||
|
|
||||||
|
async def blink(led, period_ms):
|
||||||
|
while True:
|
||||||
|
led.on()
|
||||||
|
await uasyncio.sleep_ms(5)
|
||||||
|
led.off()
|
||||||
|
await uasyncio.sleep_ms(period_ms)
|
||||||
|
|
||||||
|
async def main(led1, led2):
|
||||||
|
uasyncio.create_task(blink(led1, 700))
|
||||||
|
uasyncio.create_task(blink(led2, 400))
|
||||||
|
await uasyncio.sleep_ms(10_000)
|
||||||
|
|
||||||
|
# Running on a pyboard
|
||||||
|
from pyb import LED
|
||||||
|
uasyncio.run(main(LED(1), LED(2)))
|
||||||
|
|
||||||
|
# Running on a generic board
|
||||||
|
from machine import Pin
|
||||||
|
uasyncio.run(main(Pin(1), Pin(2)))
|
||||||
|
|
||||||
|
Core functions
|
||||||
|
--------------
|
||||||
|
|
||||||
|
.. function:: create_task(coro)
|
||||||
|
|
||||||
|
Create a new task from the given coroutine and schedule it to run.
|
||||||
|
|
||||||
|
Returns the corresponding `Task` object.
|
||||||
|
|
||||||
|
.. function:: current_task()
|
||||||
|
|
||||||
|
Return the `Task` object associated with the currently running task.
|
||||||
|
|
||||||
|
.. function:: run(coro)
|
||||||
|
|
||||||
|
Create a new task from the given coroutine and run it until it completes.
|
||||||
|
|
||||||
|
Returns the value returned by *coro*.
|
||||||
|
|
||||||
|
.. function:: sleep(t)
|
||||||
|
|
||||||
|
Sleep for *t* seconds (can be a float).
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. function:: sleep_ms(t)
|
||||||
|
|
||||||
|
Sleep for *t* milliseconds.
|
||||||
|
|
||||||
|
This is a coroutine, and a MicroPython extension.
|
||||||
|
|
||||||
|
Additional functions
|
||||||
|
--------------------
|
||||||
|
|
||||||
|
.. function:: wait_for(awaitable, timeout)
|
||||||
|
|
||||||
|
Wait for the *awaitable* to complete, but cancel it if it takes longer
|
||||||
|
that *timeout* seconds. If *awaitable* is not a task then a task will be
|
||||||
|
created from it.
|
||||||
|
|
||||||
|
If a timeout occurs, it cancels the task and raises ``asyncio.TimeoutError``:
|
||||||
|
this should be trapped by the caller.
|
||||||
|
|
||||||
|
Returns the return value of *awaitable*.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. function:: wait_for_ms(awaitable, timeout)
|
||||||
|
|
||||||
|
Similar to `wait_for` but *timeout* is an integer in milliseconds.
|
||||||
|
|
||||||
|
This is a coroutine, and a MicroPython extension.
|
||||||
|
|
||||||
|
.. function:: gather(*awaitables, return_exceptions=False)
|
||||||
|
|
||||||
|
Run all *awaitables* concurrently. Any *awaitables* that are not tasks are
|
||||||
|
promoted to tasks.
|
||||||
|
|
||||||
|
Returns a list of return values of all *awaitables*.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
class Task
|
||||||
|
----------
|
||||||
|
|
||||||
|
.. class:: Task()
|
||||||
|
|
||||||
|
This object wraps a coroutine into a running task. Tasks can be waited on
|
||||||
|
using ``await task``, which will wait for the task to complete and return
|
||||||
|
the return value of the task.
|
||||||
|
|
||||||
|
Tasks should not be created directly, rather use `create_task` to create them.
|
||||||
|
|
||||||
|
.. method:: Task.cancel()
|
||||||
|
|
||||||
|
Cancel the task by injecting a ``CancelledError`` into it. The task may
|
||||||
|
or may not ignore this exception.
|
||||||
|
|
||||||
|
class Event
|
||||||
|
-----------
|
||||||
|
|
||||||
|
.. class:: Event()
|
||||||
|
|
||||||
|
Create a new event which can be used to synchronise tasks. Events start
|
||||||
|
in the cleared state.
|
||||||
|
|
||||||
|
.. method:: Event.is_set()
|
||||||
|
|
||||||
|
Returns ``True`` if the event is set, ``False`` otherwise.
|
||||||
|
|
||||||
|
.. method:: Event.set()
|
||||||
|
|
||||||
|
Set the event. Any tasks waiting on the event will be scheduled to run.
|
||||||
|
|
||||||
|
.. method:: Event.clear()
|
||||||
|
|
||||||
|
Clear the event.
|
||||||
|
|
||||||
|
.. method:: Event.wait()
|
||||||
|
|
||||||
|
Wait for the event to be set. If the event is already set then it returns
|
||||||
|
immediately.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
class Lock
|
||||||
|
----------
|
||||||
|
|
||||||
|
.. class:: Lock()
|
||||||
|
|
||||||
|
Create a new lock which can be used to coordinate tasks. Locks start in
|
||||||
|
the unlocked state.
|
||||||
|
|
||||||
|
In addition to the methods below, locks can be used in an ``async with`` statement.
|
||||||
|
|
||||||
|
.. method:: Lock.locked()
|
||||||
|
|
||||||
|
Returns ``True`` if the lock is locked, otherwise ``False``.
|
||||||
|
|
||||||
|
.. method:: Lock.acquire()
|
||||||
|
|
||||||
|
Wait for the lock to be in the unlocked state and then lock it in an atomic
|
||||||
|
way. Only one task can acquire the lock at any one time.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. method:: Lock.release()
|
||||||
|
|
||||||
|
Release the lock. If any tasks are waiting on the lock then the next one in the
|
||||||
|
queue is scheduled to run and the lock remains locked. Otherwise, no tasks are
|
||||||
|
waiting an the lock becomes unlocked.
|
||||||
|
|
||||||
|
TCP stream connections
|
||||||
|
----------------------
|
||||||
|
|
||||||
|
.. function:: open_connection(host, port)
|
||||||
|
|
||||||
|
Open a TCP connection to the given *host* and *port*. The *host* address will be
|
||||||
|
resolved using `socket.getaddrinfo`, which is currently a blocking call.
|
||||||
|
|
||||||
|
Returns a pair of streams: a reader and a writer stream.
|
||||||
|
Will raise a socket-specific ``OSError`` if the host could not be resolved or if
|
||||||
|
the connection could not be made.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. function:: start_server(callback, host, port, backlog=5)
|
||||||
|
|
||||||
|
Start a TCP server on the given *host* and *port*. The *callback* will be
|
||||||
|
called with incoming, accepted connections, and be passed 2 arguments: reader
|
||||||
|
and writer streams for the connection.
|
||||||
|
|
||||||
|
Returns a `Server` object.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. class:: Stream()
|
||||||
|
|
||||||
|
This represents a TCP stream connection. To minimise code this class implements
|
||||||
|
both a reader and a writer, and both ``StreamReader`` and ``StreamWriter`` alias to
|
||||||
|
this class.
|
||||||
|
|
||||||
|
.. method:: Stream.get_extra_info(v)
|
||||||
|
|
||||||
|
Get extra information about the stream, given by *v*. The valid values for *v* are:
|
||||||
|
``peername``.
|
||||||
|
|
||||||
|
.. method:: Stream.close()
|
||||||
|
|
||||||
|
Close the stream.
|
||||||
|
|
||||||
|
.. method:: Stream.wait_closed()
|
||||||
|
|
||||||
|
Wait for the stream to close.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. method:: Stream.read(n)
|
||||||
|
|
||||||
|
Read up to *n* bytes and return them.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. method:: Stream.readline()
|
||||||
|
|
||||||
|
Read a line and return it.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. method:: Stream.write(buf)
|
||||||
|
|
||||||
|
Accumulated *buf* to the output buffer. The data is only flushed when
|
||||||
|
`Stream.drain` is called. It is recommended to call `Stream.drain` immediately
|
||||||
|
after calling this function.
|
||||||
|
|
||||||
|
.. method:: Stream.drain()
|
||||||
|
|
||||||
|
Drain (write) all buffered output data out to the stream.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
.. class:: Server()
|
||||||
|
|
||||||
|
This represents the server class returned from `start_server`. It can be used
|
||||||
|
in an ``async with`` statement to close the server upon exit.
|
||||||
|
|
||||||
|
.. method:: Server.close()
|
||||||
|
|
||||||
|
Close the server.
|
||||||
|
|
||||||
|
.. method:: Server.wait_closed()
|
||||||
|
|
||||||
|
Wait for the server to close.
|
||||||
|
|
||||||
|
This is a coroutine.
|
||||||
|
|
||||||
|
Event Loop
|
||||||
|
----------
|
||||||
|
|
||||||
|
.. function:: get_event_loop()
|
||||||
|
|
||||||
|
Return the event loop used to schedule and run tasks. See `Loop`.
|
||||||
|
|
||||||
|
.. function:: new_event_loop()
|
||||||
|
|
||||||
|
Reset the event loop and return it.
|
||||||
|
|
||||||
|
Note: since MicroPython only has a single event loop this function just
|
||||||
|
resets the loop's state, it does not create a new one.
|
||||||
|
|
||||||
|
.. class:: Loop()
|
||||||
|
|
||||||
|
This represents the object which schedules and runs tasks. It cannot be
|
||||||
|
created, use `get_event_loop` instead.
|
||||||
|
|
||||||
|
.. method:: Loop.create_task(coro)
|
||||||
|
|
||||||
|
Create a task from the given *coro* and return the new `Task` object.
|
||||||
|
|
||||||
|
.. method:: Loop.run_forever()
|
||||||
|
|
||||||
|
Run the event loop until `stop()` is called.
|
||||||
|
|
||||||
|
.. method:: Loop.run_until_complete(awaitable)
|
||||||
|
|
||||||
|
Run the given *awaitable* until it completes. If *awaitable* is not a task
|
||||||
|
then it will be promoted to one.
|
||||||
|
|
||||||
|
.. method:: Loop.stop()
|
||||||
|
|
||||||
|
Stop the event loop.
|
||||||
|
|
||||||
|
.. method:: Loop.close()
|
||||||
|
|
||||||
|
Close the event loop.
|
||||||
|
|
||||||
|
.. method:: Loop.set_exception_handler(handler)
|
||||||
|
|
||||||
|
Set the exception handler to call when a Task raises an exception that is not
|
||||||
|
caught. The *handler* should accept two arguments: ``(loop, context)``.
|
||||||
|
|
||||||
|
.. method:: Loop.get_exception_handler()
|
||||||
|
|
||||||
|
Get the current exception handler. Returns the handler, or ``None`` if no
|
||||||
|
custom handler is set.
|
||||||
|
|
||||||
|
.. method:: Loop.default_exception_handler(context)
|
||||||
|
|
||||||
|
The default exception handler that is called.
|
||||||
|
|
||||||
|
.. method:: Loop.call_exception_handler(context)
|
||||||
|
|
||||||
|
Call the current exception handler. The argument *context* is passed through and
|
||||||
|
is a dictionary containing keys: ``'message'``, ``'exception'``, ``'future'``.
|
@ -13,19 +13,91 @@ module is to define data structure layout with about the same power as the
|
|||||||
C language allows, and then access it using familiar dot-syntax to reference
|
C language allows, and then access it using familiar dot-syntax to reference
|
||||||
sub-fields.
|
sub-fields.
|
||||||
|
|
||||||
|
.. warning::
|
||||||
|
|
||||||
|
``uctypes`` module allows access to arbitrary memory addresses of the
|
||||||
|
machine (including I/O and control registers). Uncareful usage of it
|
||||||
|
may lead to crashes, data loss, and even hardware malfunction.
|
||||||
|
|
||||||
.. seealso::
|
.. seealso::
|
||||||
|
|
||||||
Module :mod:`struct`
|
Module :mod:`struct`
|
||||||
Standard Python way to access binary data structures (doesn't scale
|
Standard Python way to access binary data structures (doesn't scale
|
||||||
well to large and complex structures).
|
well to large and complex structures).
|
||||||
|
|
||||||
|
Usage examples::
|
||||||
|
|
||||||
|
import uctypes
|
||||||
|
|
||||||
|
# Example 1: Subset of ELF file header
|
||||||
|
# https://wikipedia.org/wiki/Executable_and_Linkable_Format#File_header
|
||||||
|
ELF_HEADER = {
|
||||||
|
"EI_MAG": (0x0 | uctypes.ARRAY, 4 | uctypes.UINT8),
|
||||||
|
"EI_DATA": 0x5 | uctypes.UINT8,
|
||||||
|
"e_machine": 0x12 | uctypes.UINT16,
|
||||||
|
}
|
||||||
|
|
||||||
|
# "f" is an ELF file opened in binary mode
|
||||||
|
buf = f.read(uctypes.sizeof(ELF_HEADER, uctypes.LITTLE_ENDIAN))
|
||||||
|
header = uctypes.struct(uctypes.addressof(buf), ELF_HEADER, uctypes.LITTLE_ENDIAN)
|
||||||
|
assert header.EI_MAG == b"\x7fELF"
|
||||||
|
assert header.EI_DATA == 1, "Oops, wrong endianness. Could retry with uctypes.BIG_ENDIAN."
|
||||||
|
print("machine:", hex(header.e_machine))
|
||||||
|
|
||||||
|
|
||||||
|
# Example 2: In-memory data structure, with pointers
|
||||||
|
COORD = {
|
||||||
|
"x": 0 | uctypes.FLOAT32,
|
||||||
|
"y": 4 | uctypes.FLOAT32,
|
||||||
|
}
|
||||||
|
|
||||||
|
STRUCT1 = {
|
||||||
|
"data1": 0 | uctypes.UINT8,
|
||||||
|
"data2": 4 | uctypes.UINT32,
|
||||||
|
"ptr": (8 | uctypes.PTR, COORD),
|
||||||
|
}
|
||||||
|
|
||||||
|
# Suppose you have address of a structure of type STRUCT1 in "addr"
|
||||||
|
# uctypes.NATIVE is optional (used by default)
|
||||||
|
struct1 = uctypes.struct(addr, STRUCT1, uctypes.NATIVE)
|
||||||
|
print("x:", struct1.ptr[0].x)
|
||||||
|
|
||||||
|
|
||||||
|
# Example 3: Access to CPU registers. Subset of STM32F4xx WWDG block
|
||||||
|
WWDG_LAYOUT = {
|
||||||
|
"WWDG_CR": (0, {
|
||||||
|
# BFUINT32 here means size of the WWDG_CR register
|
||||||
|
"WDGA": 7 << uctypes.BF_POS | 1 << uctypes.BF_LEN | uctypes.BFUINT32,
|
||||||
|
"T": 0 << uctypes.BF_POS | 7 << uctypes.BF_LEN | uctypes.BFUINT32,
|
||||||
|
}),
|
||||||
|
"WWDG_CFR": (4, {
|
||||||
|
"EWI": 9 << uctypes.BF_POS | 1 << uctypes.BF_LEN | uctypes.BFUINT32,
|
||||||
|
"WDGTB": 7 << uctypes.BF_POS | 2 << uctypes.BF_LEN | uctypes.BFUINT32,
|
||||||
|
"W": 0 << uctypes.BF_POS | 7 << uctypes.BF_LEN | uctypes.BFUINT32,
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
|
||||||
|
WWDG = uctypes.struct(0x40002c00, WWDG_LAYOUT)
|
||||||
|
|
||||||
|
WWDG.WWDG_CFR.WDGTB = 0b10
|
||||||
|
WWDG.WWDG_CR.WDGA = 1
|
||||||
|
print("Current counter:", WWDG.WWDG_CR.T)
|
||||||
|
|
||||||
Defining structure layout
|
Defining structure layout
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
Structure layout is defined by a "descriptor" - a Python dictionary which
|
Structure layout is defined by a "descriptor" - a Python dictionary which
|
||||||
encodes field names as keys and other properties required to access them as
|
encodes field names as keys and other properties required to access them as
|
||||||
associated values. Currently, uctypes requires explicit specification of
|
associated values::
|
||||||
offsets for each field. Offset are given in bytes from a structure start.
|
|
||||||
|
{
|
||||||
|
"field1": <properties>,
|
||||||
|
"field2": <properties>,
|
||||||
|
...
|
||||||
|
}
|
||||||
|
|
||||||
|
Currently, ``uctypes`` requires explicit specification of offsets for each
|
||||||
|
field. Offset are given in bytes from the structure start.
|
||||||
|
|
||||||
Following are encoding examples for various field types:
|
Following are encoding examples for various field types:
|
||||||
|
|
||||||
@ -33,7 +105,7 @@ Following are encoding examples for various field types:
|
|||||||
|
|
||||||
"field_name": offset | uctypes.UINT32
|
"field_name": offset | uctypes.UINT32
|
||||||
|
|
||||||
in other words, value is scalar type identifier ORed with field offset
|
in other words, the value is a scalar type identifier ORed with a field offset
|
||||||
(in bytes) from the start of the structure.
|
(in bytes) from the start of the structure.
|
||||||
|
|
||||||
* Recursive structures::
|
* Recursive structures::
|
||||||
@ -43,9 +115,11 @@ Following are encoding examples for various field types:
|
|||||||
"b1": 1 | uctypes.UINT8,
|
"b1": 1 | uctypes.UINT8,
|
||||||
})
|
})
|
||||||
|
|
||||||
i.e. value is a 2-tuple, first element of which is offset, and second is
|
i.e. value is a 2-tuple, first element of which is an offset, and second is
|
||||||
a structure descriptor dictionary (note: offsets in recursive descriptors
|
a structure descriptor dictionary (note: offsets in recursive descriptors
|
||||||
are relative to the structure it defines).
|
are relative to the structure it defines). Of course, recursive structures
|
||||||
|
can be specified not just by a literal dictionary, but by referring to a
|
||||||
|
structure descriptor dictionary (defined earlier) by name.
|
||||||
|
|
||||||
* Arrays of primitive types::
|
* Arrays of primitive types::
|
||||||
|
|
||||||
@ -53,42 +127,42 @@ Following are encoding examples for various field types:
|
|||||||
|
|
||||||
i.e. value is a 2-tuple, first element of which is ARRAY flag ORed
|
i.e. value is a 2-tuple, first element of which is ARRAY flag ORed
|
||||||
with offset, and second is scalar element type ORed number of elements
|
with offset, and second is scalar element type ORed number of elements
|
||||||
in array.
|
in the array.
|
||||||
|
|
||||||
* Arrays of aggregate types::
|
* Arrays of aggregate types::
|
||||||
|
|
||||||
"arr2": (offset | uctypes.ARRAY, size, {"b": 0 | uctypes.UINT8}),
|
"arr2": (offset | uctypes.ARRAY, size, {"b": 0 | uctypes.UINT8}),
|
||||||
|
|
||||||
i.e. value is a 3-tuple, first element of which is ARRAY flag ORed
|
i.e. value is a 3-tuple, first element of which is ARRAY flag ORed
|
||||||
with offset, second is a number of elements in array, and third is
|
with offset, second is a number of elements in the array, and third is
|
||||||
descriptor of element type.
|
a descriptor of element type.
|
||||||
|
|
||||||
* Pointer to a primitive type::
|
* Pointer to a primitive type::
|
||||||
|
|
||||||
"ptr": (offset | uctypes.PTR, uctypes.UINT8),
|
"ptr": (offset | uctypes.PTR, uctypes.UINT8),
|
||||||
|
|
||||||
i.e. value is a 2-tuple, first element of which is PTR flag ORed
|
i.e. value is a 2-tuple, first element of which is PTR flag ORed
|
||||||
with offset, and second is scalar element type.
|
with offset, and second is a scalar element type.
|
||||||
|
|
||||||
* Pointer to an aggregate type::
|
* Pointer to an aggregate type::
|
||||||
|
|
||||||
"ptr2": (offset | uctypes.PTR, {"b": 0 | uctypes.UINT8}),
|
"ptr2": (offset | uctypes.PTR, {"b": 0 | uctypes.UINT8}),
|
||||||
|
|
||||||
i.e. value is a 2-tuple, first element of which is PTR flag ORed
|
i.e. value is a 2-tuple, first element of which is PTR flag ORed
|
||||||
with offset, second is descriptor of type pointed to.
|
with offset, second is a descriptor of type pointed to.
|
||||||
|
|
||||||
* Bitfields::
|
* Bitfields::
|
||||||
|
|
||||||
"bitf0": offset | uctypes.BFUINT16 | lsbit << uctypes.BF_POS | bitsize << uctypes.BF_LEN,
|
"bitf0": offset | uctypes.BFUINT16 | lsbit << uctypes.BF_POS | bitsize << uctypes.BF_LEN,
|
||||||
|
|
||||||
i.e. value is type of scalar value containing given bitfield (typenames are
|
i.e. value is a type of scalar value containing given bitfield (typenames are
|
||||||
similar to scalar types, but prefixes with "BF"), ORed with offset for
|
similar to scalar types, but prefixes with ``BF``), ORed with offset for
|
||||||
scalar value containing the bitfield, and further ORed with values for
|
scalar value containing the bitfield, and further ORed with values for
|
||||||
bit offset and bit length of the bitfield within scalar value, shifted by
|
bit position and bit length of the bitfield within the scalar value, shifted by
|
||||||
BF_POS and BF_LEN positions, respectively. Bitfield position is counted
|
BF_POS and BF_LEN bits, respectively. A bitfield position is counted
|
||||||
from the least significant bit, and is the number of right-most bit of a
|
from the least significant bit of the scalar (having position of 0), and
|
||||||
field (in other words, it's a number of bits a scalar needs to be shifted
|
is the number of right-most bit of a field (in other words, it's a number
|
||||||
right to extract the bitfield).
|
of bits a scalar needs to be shifted right to extract the bitfield).
|
||||||
|
|
||||||
In the example above, first a UINT16 value will be extracted at offset 0
|
In the example above, first a UINT16 value will be extracted at offset 0
|
||||||
(this detail may be important when accessing hardware registers, where
|
(this detail may be important when accessing hardware registers, where
|
||||||
@ -108,7 +182,7 @@ Following are encoding examples for various field types:
|
|||||||
Module contents
|
Module contents
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
.. class:: struct(addr, descriptor, layout_type=NATIVE)
|
.. class:: struct(addr, descriptor, layout_type=NATIVE, /)
|
||||||
|
|
||||||
Instantiate a "foreign data structure" object based on structure address in
|
Instantiate a "foreign data structure" object based on structure address in
|
||||||
memory, descriptor (encoded as a dictionary), and layout type (see below).
|
memory, descriptor (encoded as a dictionary), and layout type (see below).
|
||||||
@ -128,10 +202,11 @@ Module contents
|
|||||||
Layout type for a native structure - with data endianness and alignment
|
Layout type for a native structure - with data endianness and alignment
|
||||||
conforming to the ABI of the system on which MicroPython runs.
|
conforming to the ABI of the system on which MicroPython runs.
|
||||||
|
|
||||||
.. function:: sizeof(struct)
|
.. function:: sizeof(struct, layout_type=NATIVE, /)
|
||||||
|
|
||||||
Return size of data structure in bytes. Argument can be either structure
|
Return size of data structure in bytes. The *struct* argument can be
|
||||||
class or specific instantiated structure object (or its aggregate field).
|
either a structure class or a specific instantiated structure object
|
||||||
|
(or its aggregate field).
|
||||||
|
|
||||||
.. function:: addressof(obj)
|
.. function:: addressof(obj)
|
||||||
|
|
||||||
@ -153,6 +228,35 @@ Module contents
|
|||||||
so it can be both written too, and you will access current value
|
so it can be both written too, and you will access current value
|
||||||
at the given memory address.
|
at the given memory address.
|
||||||
|
|
||||||
|
.. data:: UINT8
|
||||||
|
INT8
|
||||||
|
UINT16
|
||||||
|
INT16
|
||||||
|
UINT32
|
||||||
|
INT32
|
||||||
|
UINT64
|
||||||
|
INT64
|
||||||
|
|
||||||
|
Integer types for structure descriptors. Constants for 8, 16, 32,
|
||||||
|
and 64 bit types are provided, both signed and unsigned.
|
||||||
|
|
||||||
|
.. data:: FLOAT32
|
||||||
|
FLOAT64
|
||||||
|
|
||||||
|
Floating-point types for structure descriptors.
|
||||||
|
|
||||||
|
.. data:: VOID
|
||||||
|
|
||||||
|
``VOID`` is an alias for ``UINT8``, and is provided to conviniently define
|
||||||
|
C's void pointers: ``(uctypes.PTR, uctypes.VOID)``.
|
||||||
|
|
||||||
|
.. data:: PTR
|
||||||
|
ARRAY
|
||||||
|
|
||||||
|
Type constants for pointers and arrays. Note that there is no explicit
|
||||||
|
constant for structures, it's implicit: an aggregate type without ``PTR``
|
||||||
|
or ``ARRAY`` flags is a structure.
|
||||||
|
|
||||||
Structure descriptors and instantiating structure objects
|
Structure descriptors and instantiating structure objects
|
||||||
---------------------------------------------------------
|
---------------------------------------------------------
|
||||||
|
|
||||||
@ -165,7 +269,7 @@ following sources:
|
|||||||
system. Lookup these addresses in datasheet for a particular MCU/SoC.
|
system. Lookup these addresses in datasheet for a particular MCU/SoC.
|
||||||
* As a return value from a call to some FFI (Foreign Function Interface)
|
* As a return value from a call to some FFI (Foreign Function Interface)
|
||||||
function.
|
function.
|
||||||
* From uctypes.addressof(), when you want to pass arguments to an FFI
|
* From `uctypes.addressof()`, when you want to pass arguments to an FFI
|
||||||
function, or alternatively, to access some data for I/O (for example,
|
function, or alternatively, to access some data for I/O (for example,
|
||||||
data read from a file or network socket).
|
data read from a file or network socket).
|
||||||
|
|
||||||
@ -183,30 +287,41 @@ the standard subscript operator ``[]`` - both read and assigned to.
|
|||||||
|
|
||||||
If a field is a pointer, it can be dereferenced using ``[0]`` syntax
|
If a field is a pointer, it can be dereferenced using ``[0]`` syntax
|
||||||
(corresponding to C ``*`` operator, though ``[0]`` works in C too).
|
(corresponding to C ``*`` operator, though ``[0]`` works in C too).
|
||||||
Subscripting a pointer with other integer values but 0 are supported too,
|
Subscripting a pointer with other integer values but 0 are also supported,
|
||||||
with the same semantics as in C.
|
with the same semantics as in C.
|
||||||
|
|
||||||
Summing up, accessing structure fields generally follows C syntax,
|
Summing up, accessing structure fields generally follows the C syntax,
|
||||||
except for pointer dereference, when you need to use ``[0]`` operator
|
except for pointer dereference, when you need to use ``[0]`` operator
|
||||||
instead of ``*``.
|
instead of ``*``.
|
||||||
|
|
||||||
Limitations
|
Limitations
|
||||||
-----------
|
-----------
|
||||||
|
|
||||||
Accessing non-scalar fields leads to allocation of intermediate objects
|
1. Accessing non-scalar fields leads to allocation of intermediate objects
|
||||||
to represent them. This means that special care should be taken to
|
to represent them. This means that special care should be taken to
|
||||||
layout a structure which needs to be accessed when memory allocation
|
layout a structure which needs to be accessed when memory allocation
|
||||||
is disabled (e.g. from an interrupt). The recommendations are:
|
is disabled (e.g. from an interrupt). The recommendations are:
|
||||||
|
|
||||||
* Avoid nested structures. For example, instead of
|
* Avoid accessing nested structures. For example, instead of
|
||||||
``mcu_registers.peripheral_a.register1``, define separate layout
|
``mcu_registers.peripheral_a.register1``, define separate layout
|
||||||
descriptors for each peripheral, to be accessed as
|
descriptors for each peripheral, to be accessed as
|
||||||
``peripheral_a.register1``.
|
``peripheral_a.register1``. Or just cache a particular peripheral:
|
||||||
* Avoid other non-scalar data, like array. For example, instead of
|
``peripheral_a = mcu_registers.peripheral_a``. If a register
|
||||||
``peripheral_a.register[0]`` use ``peripheral_a.register0``.
|
consists of multiple bitfields, you would need to cache references
|
||||||
|
to a particular register: ``reg_a = mcu_registers.peripheral_a.reg_a``.
|
||||||
|
* Avoid other non-scalar data, like arrays. For example, instead of
|
||||||
|
``peripheral_a.register[0]`` use ``peripheral_a.register0``. Again,
|
||||||
|
an alternative is to cache intermediate values, e.g.
|
||||||
|
``register0 = peripheral_a.register[0]``.
|
||||||
|
|
||||||
Note that these recommendations will lead to decreased readability
|
2. Range of offsets supported by the ``uctypes`` module is limited.
|
||||||
and conciseness of layouts, so they should be used only if the need
|
The exact range supported is considered an implementation detail,
|
||||||
to access structure fields without allocation is anticipated (it's
|
and the general suggestion is to split structure definitions to
|
||||||
even possible to define 2 parallel layouts - one for normal usage,
|
cover from a few kilobytes to a few dozen of kilobytes maximum.
|
||||||
and a restricted one to use when memory allocation is prohibited).
|
In most cases, this is a natural situation anyway, e.g. it doesn't make
|
||||||
|
sense to define all registers of an MCU (spread over 32-bit address
|
||||||
|
space) in one structure, but rather a peripheral block by peripheral
|
||||||
|
block. In some extreme cases, you may need to split a structure in
|
||||||
|
several parts artificially (e.g. if accessing native data structure
|
||||||
|
with multi-megabyte array in the middle, though that would be a very
|
||||||
|
synthetic case).
|
||||||
|
@ -47,15 +47,20 @@ Methods
|
|||||||
|
|
||||||
*eventmask* defaults to ``uselect.POLLIN | uselect.POLLOUT``.
|
*eventmask* defaults to ``uselect.POLLIN | uselect.POLLOUT``.
|
||||||
|
|
||||||
|
It is OK to call this function multiple times for the same *obj*.
|
||||||
|
Successive calls will update *obj*'s eventmask to the value of
|
||||||
|
*eventmask* (i.e. will behave as `modify()`).
|
||||||
|
|
||||||
.. method:: poll.unregister(obj)
|
.. method:: poll.unregister(obj)
|
||||||
|
|
||||||
Unregister *obj* from polling.
|
Unregister *obj* from polling.
|
||||||
|
|
||||||
.. method:: poll.modify(obj, eventmask)
|
.. method:: poll.modify(obj, eventmask)
|
||||||
|
|
||||||
Modify the *eventmask* for *obj*.
|
Modify the *eventmask* for *obj*. If *obj* is not registered, `OSError`
|
||||||
|
is raised with error of ENOENT.
|
||||||
|
|
||||||
.. method:: poll.poll(timeout=-1)
|
.. method:: poll.poll(timeout=-1, /)
|
||||||
|
|
||||||
Wait for at least one of the registered objects to become ready or have an
|
Wait for at least one of the registered objects to become ready or have an
|
||||||
exceptional condition, with optional timeout in milliseconds (if *timeout*
|
exceptional condition, with optional timeout in milliseconds (if *timeout*
|
||||||
@ -78,7 +83,7 @@ Methods
|
|||||||
|
|
||||||
Tuples returned may contain more than 2 elements as described above.
|
Tuples returned may contain more than 2 elements as described above.
|
||||||
|
|
||||||
.. method:: poll.ipoll(timeout=-1, flags=0)
|
.. method:: poll.ipoll(timeout=-1, flags=0, /)
|
||||||
|
|
||||||
Like :meth:`poll.poll`, but instead returns an iterator which yields a
|
Like :meth:`poll.poll`, but instead returns an iterator which yields a
|
||||||
``callee-owned tuples``. This function provides efficient, allocation-free
|
``callee-owned tuples``. This function provides efficient, allocation-free
|
||||||
|
@ -1,336 +0,0 @@
|
|||||||
*******************************
|
|
||||||
:mod:`usocket` -- socket module
|
|
||||||
*******************************
|
|
||||||
|
|
||||||
.. include:: ../templates/unsupported_in_circuitpython.inc
|
|
||||||
|
|
||||||
.. module:: usocket
|
|
||||||
:synopsis: socket module
|
|
||||||
|
|
||||||
|see_cpython_module| :mod:`cpython:socket`.
|
|
||||||
|
|
||||||
This module provides access to the BSD socket interface.
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
For efficiency and consistency, socket objects in MicroPython implement a ``stream``
|
|
||||||
(file-like) interface directly. In CPython, you need to convert a socket to
|
|
||||||
a file-like object using `makefile()` method. This method is still supported
|
|
||||||
by MicroPython (but is a no-op), so where compatibility with CPython matters,
|
|
||||||
be sure to use it.
|
|
||||||
|
|
||||||
Socket address format(s)
|
|
||||||
------------------------
|
|
||||||
|
|
||||||
The native socket address format of the ``usocket`` module is an opaque data type
|
|
||||||
returned by `getaddrinfo` function, which must be used to resolve textual address
|
|
||||||
(including numeric addresses)::
|
|
||||||
|
|
||||||
sockaddr = usocket.getaddrinfo('www.micropython.org', 80)[0][-1]
|
|
||||||
# You must use getaddrinfo() even for numeric addresses
|
|
||||||
sockaddr = usocket.getaddrinfo('127.0.0.1', 80)[0][-1]
|
|
||||||
# Now you can use that address
|
|
||||||
sock.connect(addr)
|
|
||||||
|
|
||||||
Using `getaddrinfo` is the most efficient (both in terms of memory and processing
|
|
||||||
power) and portable way to work with addresses.
|
|
||||||
|
|
||||||
However, ``socket`` module (note the difference with native MicroPython
|
|
||||||
``usocket`` module described here) provides CPython-compatible way to specify
|
|
||||||
addresses using tuples, as described below.
|
|
||||||
|
|
||||||
Summing up:
|
|
||||||
|
|
||||||
* Always use `getaddrinfo` when writing portable applications.
|
|
||||||
* Tuple addresses described below can be used as a shortcut for
|
|
||||||
quick hacks and interactive use, if your port supports them.
|
|
||||||
|
|
||||||
Tuple address format for ``socket`` module:
|
|
||||||
|
|
||||||
* IPv4: *(ipv4_address, port)*, where *ipv4_address* is a string with
|
|
||||||
dot-notation numeric IPv4 address, e.g. ``"8.8.8.8"``, and *port* is and
|
|
||||||
integer port number in the range 1-65535. Note the domain names are not
|
|
||||||
accepted as *ipv4_address*, they should be resolved first using
|
|
||||||
`usocket.getaddrinfo()`.
|
|
||||||
* IPv6: *(ipv6_address, port, flowinfo, scopeid)*, where *ipv6_address*
|
|
||||||
is a string with colon-notation numeric IPv6 address, e.g. ``"2001:db8::1"``,
|
|
||||||
and *port* is an integer port number in the range 1-65535. *flowinfo*
|
|
||||||
must be 0. *scopeid* is the interface scope identifier for link-local
|
|
||||||
addresses. Note the domain names are not accepted as *ipv6_address*,
|
|
||||||
they should be resolved first using `usocket.getaddrinfo()`.
|
|
||||||
|
|
||||||
Functions
|
|
||||||
---------
|
|
||||||
|
|
||||||
.. function:: socket(af=AF_INET, type=SOCK_STREAM, proto=IPPROTO_TCP)
|
|
||||||
|
|
||||||
Create a new socket using the given address family, socket type and
|
|
||||||
protocol number. Note that specifying *proto* in most cases is not
|
|
||||||
required (and not recommended, as some MicroPython ports may omit
|
|
||||||
``IPPROTO_*`` constants). Instead, *type* argument will select needed
|
|
||||||
protocol automatically::
|
|
||||||
|
|
||||||
# Create STREAM TCP socket
|
|
||||||
socket(AF_INET, SOCK_STREAM)
|
|
||||||
# Create DGRAM UDP socket
|
|
||||||
socket(AF_INET, SOCK_DGRAM)
|
|
||||||
|
|
||||||
.. function:: getaddrinfo(host, port)
|
|
||||||
|
|
||||||
Translate the host/port argument into a sequence of 5-tuples that contain all the
|
|
||||||
necessary arguments for creating a socket connected to that service. The list of
|
|
||||||
5-tuples has following structure::
|
|
||||||
|
|
||||||
(family, type, proto, canonname, sockaddr)
|
|
||||||
|
|
||||||
The following example shows how to connect to a given url::
|
|
||||||
|
|
||||||
s = usocket.socket()
|
|
||||||
s.connect(usocket.getaddrinfo('www.micropython.org', 80)[0][-1])
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
CPython raises a ``socket.gaierror`` exception (`OSError` subclass) in case
|
|
||||||
of error in this function. MicroPython doesn't have ``socket.gaierror``
|
|
||||||
and raises OSError directly. Note that error numbers of `getaddrinfo()`
|
|
||||||
form a separate namespace and may not match error numbers from
|
|
||||||
:py:mod:`uerrno` module. To distinguish `getaddrinfo()` errors, they are
|
|
||||||
represented by negative numbers, whereas standard system errors are
|
|
||||||
positive numbers (error numbers are accessible using ``e.args[0]`` property
|
|
||||||
from an exception object). The use of negative values is a provisional
|
|
||||||
detail which may change in the future.
|
|
||||||
|
|
||||||
.. function:: inet_ntop(af, bin_addr)
|
|
||||||
|
|
||||||
Convert a binary network address *bin_addr* of the given address family *af*
|
|
||||||
to a textual representation::
|
|
||||||
|
|
||||||
>>> usocket.inet_ntop(usocket.AF_INET, b"\x7f\0\0\1")
|
|
||||||
'127.0.0.1'
|
|
||||||
|
|
||||||
.. function:: inet_pton(af, txt_addr)
|
|
||||||
|
|
||||||
Convert a textual network address *txt_addr* of the given address family *af*
|
|
||||||
to a binary representation::
|
|
||||||
|
|
||||||
>>> usocket.inet_pton(usocket.AF_INET, "1.2.3.4")
|
|
||||||
b'\x01\x02\x03\x04'
|
|
||||||
|
|
||||||
Constants
|
|
||||||
---------
|
|
||||||
|
|
||||||
.. data:: AF_INET
|
|
||||||
AF_INET6
|
|
||||||
|
|
||||||
Address family types. Availability depends on a particular ``MicroPython port``.
|
|
||||||
|
|
||||||
.. data:: SOCK_STREAM
|
|
||||||
SOCK_DGRAM
|
|
||||||
|
|
||||||
Socket types.
|
|
||||||
|
|
||||||
.. data:: IPPROTO_UDP
|
|
||||||
IPPROTO_TCP
|
|
||||||
|
|
||||||
IP protocol numbers. Availability depends on a particular ``MicroPython port``.
|
|
||||||
Note that you don't need to specify these in a call to `usocket.socket()`,
|
|
||||||
because `SOCK_STREAM` socket type automatically selects `IPPROTO_TCP`, and
|
|
||||||
`SOCK_DGRAM` - `IPPROTO_UDP`. Thus, the only real use of these constants
|
|
||||||
is as an argument to `usocket.socket.setsockopt()`.
|
|
||||||
|
|
||||||
.. data:: usocket.SOL_*
|
|
||||||
|
|
||||||
Socket option levels (an argument to `usocket.socket.setsockopt()`). The exact
|
|
||||||
inventory depends on a ``MicroPython port``.
|
|
||||||
|
|
||||||
.. data:: usocket.SO_*
|
|
||||||
|
|
||||||
Socket options (an argument to `usocket.socket.setsockopt()`). The exact
|
|
||||||
inventory depends on a ``MicroPython port``.
|
|
||||||
|
|
||||||
Constants specific to WiPy:
|
|
||||||
|
|
||||||
.. data:: IPPROTO_SEC
|
|
||||||
|
|
||||||
Special protocol value to create SSL-compatible socket.
|
|
||||||
|
|
||||||
class socket
|
|
||||||
============
|
|
||||||
|
|
||||||
Methods
|
|
||||||
-------
|
|
||||||
|
|
||||||
.. method:: socket.close()
|
|
||||||
|
|
||||||
Mark the socket closed and release all resources. Once that happens, all future operations
|
|
||||||
on the socket object will fail. The remote end will receive EOF indication if
|
|
||||||
supported by protocol.
|
|
||||||
|
|
||||||
Sockets are automatically closed when they are garbage-collected, but it is recommended
|
|
||||||
to `close()` them explicitly as soon you finished working with them.
|
|
||||||
|
|
||||||
.. method:: socket.bind(address)
|
|
||||||
|
|
||||||
Bind the socket to *address*. The socket must not already be bound.
|
|
||||||
|
|
||||||
.. method:: socket.listen([backlog])
|
|
||||||
|
|
||||||
Enable a server to accept connections. If *backlog* is specified, it must be at least 0
|
|
||||||
(if it's lower, it will be set to 0); and specifies the number of unaccepted connections
|
|
||||||
that the system will allow before refusing new connections. If not specified, a default
|
|
||||||
reasonable value is chosen.
|
|
||||||
|
|
||||||
.. method:: socket.accept()
|
|
||||||
|
|
||||||
Accept a connection. The socket must be bound to an address and listening for connections.
|
|
||||||
The return value is a pair (conn, address) where conn is a new socket object usable to send
|
|
||||||
and receive data on the connection, and address is the address bound to the socket on the
|
|
||||||
other end of the connection.
|
|
||||||
|
|
||||||
.. method:: socket.connect(address)
|
|
||||||
|
|
||||||
Connect to a remote socket at *address*.
|
|
||||||
|
|
||||||
.. method:: socket.send(bytes)
|
|
||||||
|
|
||||||
Send data to the socket. The socket must be connected to a remote socket.
|
|
||||||
Returns number of bytes sent, which may be smaller than the length of data
|
|
||||||
("short write").
|
|
||||||
|
|
||||||
.. method:: socket.sendall(bytes)
|
|
||||||
|
|
||||||
Send all data to the socket. The socket must be connected to a remote socket.
|
|
||||||
Unlike `send()`, this method will try to send all of data, by sending data
|
|
||||||
chunk by chunk consecutively.
|
|
||||||
|
|
||||||
The behavior of this method on non-blocking sockets is undefined. Due to this,
|
|
||||||
on MicroPython, it's recommended to use `write()` method instead, which
|
|
||||||
has the same "no short writes" policy for blocking sockets, and will return
|
|
||||||
number of bytes sent on non-blocking sockets.
|
|
||||||
|
|
||||||
.. method:: socket.recv(bufsize)
|
|
||||||
|
|
||||||
Receive data from the socket. The return value is a bytes object representing the data
|
|
||||||
received. The maximum amount of data to be received at once is specified by bufsize.
|
|
||||||
|
|
||||||
.. method:: socket.sendto(bytes, address)
|
|
||||||
|
|
||||||
Send data to the socket. The socket should not be connected to a remote socket, since the
|
|
||||||
destination socket is specified by *address*.
|
|
||||||
|
|
||||||
.. method:: socket.recvfrom(bufsize)
|
|
||||||
|
|
||||||
Receive data from the socket. The return value is a pair *(bytes, address)* where *bytes* is a
|
|
||||||
bytes object representing the data received and *address* is the address of the socket sending
|
|
||||||
the data.
|
|
||||||
|
|
||||||
.. method:: socket.setsockopt(level, optname, value)
|
|
||||||
|
|
||||||
Set the value of the given socket option. The needed symbolic constants are defined in the
|
|
||||||
socket module (SO_* etc.). The *value* can be an integer or a bytes-like object representing
|
|
||||||
a buffer.
|
|
||||||
|
|
||||||
.. method:: socket.settimeout(value)
|
|
||||||
|
|
||||||
**Note**: Not every port supports this method, see below.
|
|
||||||
|
|
||||||
Set a timeout on blocking socket operations. The value argument can be a nonnegative floating
|
|
||||||
point number expressing seconds, or None. If a non-zero value is given, subsequent socket operations
|
|
||||||
will raise an `OSError` exception if the timeout period value has elapsed before the operation has
|
|
||||||
completed. If zero is given, the socket is put in non-blocking mode. If None is given, the socket
|
|
||||||
is put in blocking mode.
|
|
||||||
|
|
||||||
Not every ``MicroPython port`` supports this method. A more portable and
|
|
||||||
generic solution is to use `uselect.poll` object. This allows to wait on
|
|
||||||
multiple objects at the same time (and not just on sockets, but on generic
|
|
||||||
``stream`` objects which support polling). Example::
|
|
||||||
|
|
||||||
# Instead of:
|
|
||||||
s.settimeout(1.0) # time in seconds
|
|
||||||
s.read(10) # may timeout
|
|
||||||
|
|
||||||
# Use:
|
|
||||||
poller = uselect.poll()
|
|
||||||
poller.register(s, uselect.POLLIN)
|
|
||||||
res = poller.poll(1000) # time in milliseconds
|
|
||||||
if not res:
|
|
||||||
# s is still not ready for input, i.e. operation timed out
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
CPython raises a ``socket.timeout`` exception in case of timeout,
|
|
||||||
which is an `OSError` subclass. MicroPython raises an OSError directly
|
|
||||||
instead. If you use ``except OSError:`` to catch the exception,
|
|
||||||
your code will work both in MicroPython and CPython.
|
|
||||||
|
|
||||||
.. method:: socket.setblocking(flag)
|
|
||||||
|
|
||||||
Set blocking or non-blocking mode of the socket: if flag is false, the socket is set to non-blocking,
|
|
||||||
else to blocking mode.
|
|
||||||
|
|
||||||
This method is a shorthand for certain `settimeout()` calls:
|
|
||||||
|
|
||||||
* ``sock.setblocking(True)`` is equivalent to ``sock.settimeout(None)``
|
|
||||||
* ``sock.setblocking(False)`` is equivalent to ``sock.settimeout(0)``
|
|
||||||
|
|
||||||
.. method:: socket.makefile(mode='rb', buffering=0)
|
|
||||||
|
|
||||||
Return a file object associated with the socket. The exact returned type depends on the arguments
|
|
||||||
given to makefile(). The support is limited to binary modes only ('rb', 'wb', and 'rwb').
|
|
||||||
CPython's arguments: *encoding*, *errors* and *newline* are not supported.
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
As MicroPython doesn't support buffered streams, values of *buffering*
|
|
||||||
parameter is ignored and treated as if it was 0 (unbuffered).
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
Closing the file object returned by makefile() WILL close the
|
|
||||||
original socket as well.
|
|
||||||
|
|
||||||
.. method:: socket.read([size])
|
|
||||||
|
|
||||||
Read up to size bytes from the socket. Return a bytes object. If *size* is not given, it
|
|
||||||
reads all data available from the socket until EOF; as such the method will not return until
|
|
||||||
the socket is closed. This function tries to read as much data as
|
|
||||||
requested (no "short reads"). This may be not possible with
|
|
||||||
non-blocking socket though, and then less data will be returned.
|
|
||||||
|
|
||||||
.. method:: socket.readinto(buf[, nbytes])
|
|
||||||
|
|
||||||
Read bytes into the *buf*. If *nbytes* is specified then read at most
|
|
||||||
that many bytes. Otherwise, read at most *len(buf)* bytes. Just as
|
|
||||||
`read()`, this method follows "no short reads" policy.
|
|
||||||
|
|
||||||
Return value: number of bytes read and stored into *buf*.
|
|
||||||
|
|
||||||
.. method:: socket.readline()
|
|
||||||
|
|
||||||
Read a line, ending in a newline character.
|
|
||||||
|
|
||||||
Return value: the line read.
|
|
||||||
|
|
||||||
.. method:: socket.write(buf)
|
|
||||||
|
|
||||||
Write the buffer of bytes to the socket. This function will try to
|
|
||||||
write all data to a socket (no "short writes"). This may be not possible
|
|
||||||
with a non-blocking socket though, and returned value will be less than
|
|
||||||
the length of *buf*.
|
|
||||||
|
|
||||||
Return value: number of bytes written.
|
|
||||||
|
|
||||||
.. exception:: usocket.error
|
|
||||||
|
|
||||||
MicroPython does NOT have this exception.
|
|
||||||
|
|
||||||
.. admonition:: Difference to CPython
|
|
||||||
:class: attention
|
|
||||||
|
|
||||||
CPython used to have a ``socket.error`` exception which is now deprecated,
|
|
||||||
and is an alias of `OSError`. In MicroPython, use `OSError` directly.
|
|
@ -1,50 +0,0 @@
|
|||||||
:mod:`ussl` -- SSL/TLS module
|
|
||||||
=============================
|
|
||||||
|
|
||||||
.. include:: ../templates/unsupported_in_circuitpython.inc
|
|
||||||
|
|
||||||
.. module:: ussl
|
|
||||||
:synopsis: TLS/SSL wrapper for socket objects
|
|
||||||
|
|
||||||
|see_cpython_module| :mod:`cpython:ssl`.
|
|
||||||
|
|
||||||
This module provides access to Transport Layer Security (previously and
|
|
||||||
widely known as “Secure Sockets Layer”) encryption and peer authentication
|
|
||||||
facilities for network sockets, both client-side and server-side.
|
|
||||||
|
|
||||||
Functions
|
|
||||||
---------
|
|
||||||
|
|
||||||
.. function:: ussl.wrap_socket(sock, server_side=False, keyfile=None, certfile=None, cert_reqs=CERT_NONE, ca_certs=None)
|
|
||||||
|
|
||||||
Takes a ``stream`` *sock* (usually usocket.socket instance of ``SOCK_STREAM`` type),
|
|
||||||
and returns an instance of ssl.SSLSocket, which wraps the underlying stream in
|
|
||||||
an SSL context. Returned object has the usual ``stream`` interface methods like
|
|
||||||
``read()``, ``write()``, etc. In MicroPython, the returned object does not expose
|
|
||||||
socket interface and methods like ``recv()``, ``send()``. In particular, a
|
|
||||||
server-side SSL socket should be created from a normal socket returned from
|
|
||||||
:meth:`~usocket.socket.accept()` on a non-SSL listening server socket.
|
|
||||||
|
|
||||||
Depending on the underlying module implementation in a particular
|
|
||||||
``MicroPython port``, some or all keyword arguments above may be not supported.
|
|
||||||
|
|
||||||
.. warning::
|
|
||||||
|
|
||||||
Some implementations of ``ussl`` module do NOT validate server certificates,
|
|
||||||
which makes an SSL connection established prone to man-in-the-middle attacks.
|
|
||||||
|
|
||||||
Exceptions
|
|
||||||
----------
|
|
||||||
|
|
||||||
.. data:: ssl.SSLError
|
|
||||||
|
|
||||||
This exception does NOT exist. Instead its base class, OSError, is used.
|
|
||||||
|
|
||||||
Constants
|
|
||||||
---------
|
|
||||||
|
|
||||||
.. data:: ussl.CERT_NONE
|
|
||||||
ussl.CERT_OPTIONAL
|
|
||||||
ussl.CERT_REQUIRED
|
|
||||||
|
|
||||||
Supported values for *cert_reqs* parameter.
|
|
@ -16,7 +16,7 @@ is not yet implemented.
|
|||||||
Functions
|
Functions
|
||||||
---------
|
---------
|
||||||
|
|
||||||
.. function:: decompress(data, wbits=0, bufsize=0)
|
.. function:: decompress(data, wbits=0, bufsize=0, /)
|
||||||
|
|
||||||
Return decompressed *data* as bytes. *wbits* is DEFLATE dictionary window
|
Return decompressed *data* as bytes. *wbits* is DEFLATE dictionary window
|
||||||
size used during compression (8-15, the dictionary size is power of 2 of
|
size used during compression (8-15, the dictionary size is power of 2 of
|
||||||
@ -25,7 +25,7 @@ Functions
|
|||||||
to be raw DEFLATE stream. *bufsize* parameter is for compatibility with
|
to be raw DEFLATE stream. *bufsize* parameter is for compatibility with
|
||||||
CPython and is ignored.
|
CPython and is ignored.
|
||||||
|
|
||||||
.. class:: DecompIO(stream, wbits=0)
|
.. class:: DecompIO(stream, wbits=0, /)
|
||||||
|
|
||||||
Create a ``stream`` wrapper which allows transparent decompression of
|
Create a ``stream`` wrapper which allows transparent decompression of
|
||||||
compressed data in another *stream*. This allows to process compressed
|
compressed data in another *stream*. This allows to process compressed
|
||||||
|
175
docs/reference/glossary.rst
Normal file
175
docs/reference/glossary.rst
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
Glossary
|
||||||
|
========
|
||||||
|
|
||||||
|
.. glossary::
|
||||||
|
|
||||||
|
baremetal
|
||||||
|
A system without a (full-fledged) operating system, for example an
|
||||||
|
:term:`MCU`-based system. When running on a baremetal system,
|
||||||
|
MicroPython effectively functions like a small operating system,
|
||||||
|
running user programs and providing a command interpreter
|
||||||
|
(:term:`REPL`).
|
||||||
|
|
||||||
|
buffer protocol
|
||||||
|
Any Python object that can be automatically converted into bytes, such
|
||||||
|
as ``bytes``, ``bytearray``, ``memoryview`` and ``str`` objects, which
|
||||||
|
all implement the "buffer protocol".
|
||||||
|
|
||||||
|
board
|
||||||
|
Typically this refers to a printed circuit board (PCB) containing a
|
||||||
|
:term:`microcontroller <MCU>` and supporting components.
|
||||||
|
MicroPython firmware is typically provided per-board, as the firmware
|
||||||
|
contains both MCU-specific functionality but also board-level
|
||||||
|
functionality such as drivers or pin names.
|
||||||
|
|
||||||
|
bytecode
|
||||||
|
A compact representation of a Python program that generated by
|
||||||
|
compiling the Python source code. This is what the VM actually
|
||||||
|
executes. Bytecode is typically generated automatically at runtime and
|
||||||
|
is invisible to the user. Note that while :term:`CPython` and
|
||||||
|
MicroPython both use bytecode, the format is different. You can also
|
||||||
|
pre-compile source code offline using the :term:`cross-compiler`.
|
||||||
|
|
||||||
|
callee-owned tuple
|
||||||
|
This is a MicroPython-specific construct where, for efficiency
|
||||||
|
reasons, some built-in functions or methods may re-use the same
|
||||||
|
underlying tuple object to return data. This avoids having to allocate
|
||||||
|
a new tuple for every call, and reduces heap fragmentation. Programs
|
||||||
|
should not hold references to callee-owned tuples and instead only
|
||||||
|
extract data from them (or make a copy).
|
||||||
|
|
||||||
|
CircuitPython
|
||||||
|
A variant of MicroPython developed by `Adafruit Industries
|
||||||
|
<https://circuitpython.org>`_.
|
||||||
|
|
||||||
|
CPython
|
||||||
|
CPython is the reference implementation of the Python programming
|
||||||
|
language, and the most well-known one. It is, however, one of many
|
||||||
|
implementations (including Jython, IronPython, PyPy, and MicroPython).
|
||||||
|
While MicroPython's implementation differs substantially from CPython,
|
||||||
|
it aims to maintain as much compatibility as possible.
|
||||||
|
|
||||||
|
cross-compiler
|
||||||
|
Also known as ``mpy-cross``. This tool runs on your PC and converts a
|
||||||
|
:term:`.py file` containing MicroPython code into a :term:`.mpy file`
|
||||||
|
containing MicroPython bytecode. This means it loads faster (the board
|
||||||
|
doesn't have to compile the code), and uses less space on flash (the
|
||||||
|
bytecode is more space efficient).
|
||||||
|
|
||||||
|
driver
|
||||||
|
A MicroPython library that implements support for a particular
|
||||||
|
component, such as a sensor or display.
|
||||||
|
|
||||||
|
FFI
|
||||||
|
Acronym for Foreign Function Interface. A mechanism used by the
|
||||||
|
:term:`MicroPython Unix port` to access operating system functionality.
|
||||||
|
This is not available on :term:`baremetal` ports.
|
||||||
|
|
||||||
|
filesystem
|
||||||
|
Most MicroPython ports and boards provide a filesystem stored in flash
|
||||||
|
that is available to user code via the standard Python file APIs such
|
||||||
|
as ``open()``. Some boards also make this internal filesystem
|
||||||
|
accessible to the host via USB mass-storage.
|
||||||
|
|
||||||
|
frozen module
|
||||||
|
A Python module that has been cross compiled and bundled into the
|
||||||
|
firmware image. This reduces RAM requirements as the code is executed
|
||||||
|
directly from flash.
|
||||||
|
|
||||||
|
Garbage Collector
|
||||||
|
A background process that runs in Python (and MicroPython) to reclaim
|
||||||
|
unused memory in the :term:`heap`.
|
||||||
|
|
||||||
|
GPIO
|
||||||
|
General-purpose input/output. The simplest means to control electrical
|
||||||
|
signals (commonly referred to as "pins") on a microcontroller. GPIO
|
||||||
|
typically allows pins to be either input or output, and to set or get
|
||||||
|
their digital value (logical "0" or "1"). MicroPython abstracts GPIO
|
||||||
|
access using the :class:`machine.Pin` and :class:`machine.Signal`
|
||||||
|
classes.
|
||||||
|
|
||||||
|
GPIO port
|
||||||
|
A group of :term:`GPIO` pins, usually based on hardware properties of
|
||||||
|
these pins (e.g. controllable by the same register).
|
||||||
|
|
||||||
|
heap
|
||||||
|
A region of RAM where MicroPython stores dynamic data. It is managed
|
||||||
|
automatically by the :term:`Garbage Collector`. Different MCUs and
|
||||||
|
boards have vastly different amounts of RAM available for the heap, so
|
||||||
|
this will affect how complex your program can be.
|
||||||
|
|
||||||
|
interned string
|
||||||
|
An optimisation used by MicroPython to improve the efficiency of
|
||||||
|
working with strings. An interned string is referenced by its (unique)
|
||||||
|
identity rather than its address and can therefore be quickly compared
|
||||||
|
just by its identifier. It also means that identical strings can be
|
||||||
|
de-duplicated in memory. String interning is almost always invisible to
|
||||||
|
the user.
|
||||||
|
|
||||||
|
MCU
|
||||||
|
Microcontroller. Microcontrollers usually have much less resources
|
||||||
|
than a desktop, laptop, or phone, but are smaller, cheaper and
|
||||||
|
require much less power. MicroPython is designed to be small and
|
||||||
|
optimized enough to run on an average modern microcontroller.
|
||||||
|
|
||||||
|
MicroPython port
|
||||||
|
MicroPython supports different :term:`boards <board>`, RTOSes, and
|
||||||
|
OSes, and can be relatively easily adapted to new systems. MicroPython
|
||||||
|
with support for a particular system is called a "port" to that
|
||||||
|
system. Different ports may have widely different functionality. This
|
||||||
|
documentation is intended to be a reference of the generic APIs
|
||||||
|
available across different ports ("MicroPython core"). Note that some
|
||||||
|
ports may still omit some APIs described here (e.g. due to resource
|
||||||
|
constraints). Any such differences, and port-specific extensions
|
||||||
|
beyond the MicroPython core functionality, would be described in the
|
||||||
|
separate port-specific documentation.
|
||||||
|
|
||||||
|
MicroPython Unix port
|
||||||
|
The unix port is one of the major :term:`MicroPython ports
|
||||||
|
<MicroPython port>`. It is intended to run on POSIX-compatible
|
||||||
|
operating systems, like Linux, MacOS, FreeBSD, Solaris, etc. It also
|
||||||
|
serves as the basis of Windows port. The Unix port is very useful for
|
||||||
|
quick development and testing of the MicroPython language and
|
||||||
|
machine-independent features. It can also function in a similar way to
|
||||||
|
:term:`CPython`'s ``python`` executable.
|
||||||
|
|
||||||
|
.mpy file
|
||||||
|
The output of the :term:`cross-compiler`. A compiled form of a
|
||||||
|
:term:`.py file` that contains MicroPython bytecode instead of Python
|
||||||
|
source code.
|
||||||
|
|
||||||
|
native
|
||||||
|
Usually refers to "native code", i.e. machine code for the target
|
||||||
|
microcontroller (such as ARM Thumb, Xtensa, x86/x64). The ``@native``
|
||||||
|
decorator can be applied to a MicroPython function to generate native
|
||||||
|
code instead of bytecode for that function, which will likely be
|
||||||
|
faster but use more RAM.
|
||||||
|
|
||||||
|
port
|
||||||
|
Usually short for :term:`MicroPython port`, but could also refer to
|
||||||
|
:term:`GPIO port`.
|
||||||
|
|
||||||
|
.py file
|
||||||
|
A file containing Python source code.
|
||||||
|
|
||||||
|
REPL
|
||||||
|
An acronym for "Read, Eval, Print, Loop". This is the interactive
|
||||||
|
Python prompt, useful for debugging or testing short snippets of code.
|
||||||
|
Most MicroPython boards make a REPL available over a UART, and this is
|
||||||
|
typically accessible on a host PC via USB.
|
||||||
|
|
||||||
|
stream
|
||||||
|
Also known as a "file-like object". An Python object which provides
|
||||||
|
sequential read-write access to the underlying data. A stream object
|
||||||
|
implements a corresponding interface, which consists of methods like
|
||||||
|
``read()``, ``write()``, ``readinto()``, ``seek()``, ``flush()``,
|
||||||
|
``close()``, etc. A stream is an important concept in MicroPython;
|
||||||
|
many I/O objects implement the stream interface, and thus can be used
|
||||||
|
consistently and interchangeably in different contexts. For more
|
||||||
|
information on streams in MicroPython, see the `io` module.
|
||||||
|
|
||||||
|
UART
|
||||||
|
Acronym for "Universal Asynchronous Receiver/Transmitter". This is a
|
||||||
|
peripheral that sends data over a pair of pins (TX & RX). Many boards
|
||||||
|
include a way to make at least one of the UARTs available to a host PC
|
||||||
|
as a serial port over USB.
|
@ -1,7 +1,8 @@
|
|||||||
sphinx<4
|
sphinx<4
|
||||||
recommonmark==0.6.0
|
myst-parser==0.14.0
|
||||||
sphinxcontrib-svg2pdfconverter==0.1.0
|
sphinxcontrib-svg2pdfconverter==0.1.0
|
||||||
astroid
|
astroid
|
||||||
sphinx-autoapi
|
sphinx-autoapi
|
||||||
isort
|
isort
|
||||||
black
|
black
|
||||||
|
readthedocs-sphinx-search
|
||||||
|
@ -155,7 +155,8 @@ def get_settings_from_makefile(port_dir, board_name):
|
|||||||
|
|
||||||
settings = {}
|
settings = {}
|
||||||
for line in contents.stdout.split('\n'):
|
for line in contents.stdout.split('\n'):
|
||||||
m = re.match(r'^([A-Z][A-Z0-9_]*) = (.*)$', line)
|
# Handle both = and := definitions.
|
||||||
|
m = re.match(r'^([A-Z][A-Z0-9_]*) :?= (.*)$', line)
|
||||||
if m:
|
if m:
|
||||||
settings[m.group(1)] = m.group(2)
|
settings[m.group(1)] = m.group(2)
|
||||||
|
|
||||||
@ -224,7 +225,7 @@ def support_matrix_by_board(use_branded_name=True):
|
|||||||
# flatmap with comprehensions
|
# flatmap with comprehensions
|
||||||
boards = dict(sorted([board for matrix in mapped_exec for board in matrix]))
|
boards = dict(sorted([board for matrix in mapped_exec for board in matrix]))
|
||||||
|
|
||||||
#print(json.dumps(boards, indent=2))
|
# print(json.dumps(boards, indent=2))
|
||||||
return boards
|
return boards
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
17
docs/static/filter.css
vendored
Normal file
17
docs/static/filter.css
vendored
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#support-matrix-filter-block { position: relative; }
|
||||||
|
#support-matrix-filter {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
#support-matrix-filter-num {
|
||||||
|
position: absolute;
|
||||||
|
right: 10px;
|
||||||
|
top: 4px;
|
||||||
|
}
|
||||||
|
.support-matrix-table .this_module code,
|
||||||
|
.support-matrix-table .this_module span {
|
||||||
|
background: black;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
.support-matrix-table .board_hidden {
|
||||||
|
display: none;
|
||||||
|
}
|
86
docs/static/filter.js
vendored
Normal file
86
docs/static/filter.js
vendored
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
$(() => {
|
||||||
|
var urlTimeout = null;
|
||||||
|
function setURL(query, value) {
|
||||||
|
clearTimeout(urlTimeout);
|
||||||
|
|
||||||
|
urlTimeout = setTimeout(function() {
|
||||||
|
var url = new URL(window.location.href);
|
||||||
|
console.log(query,value,value.length,!value.length);
|
||||||
|
if (!value.length) {
|
||||||
|
console.log
|
||||||
|
url.searchParams.delete(query);
|
||||||
|
} else if (Array.isArray(value)) {
|
||||||
|
url.searchParams.delete(query);
|
||||||
|
value.forEach(function(v) {
|
||||||
|
url.searchParams.append(query, v);
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
url.searchParams.set(query, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
window.history.pushState(null, document.title, url.href);
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
function handlePageLoad() {
|
||||||
|
var url = new URL(window.location.href);
|
||||||
|
//get values from URL
|
||||||
|
var filters = url.searchParams.getAll('filter');
|
||||||
|
search_terms = filters.join(" ");
|
||||||
|
$("#support-matrix-filter").val(search_terms);
|
||||||
|
run_filter();
|
||||||
|
}
|
||||||
|
|
||||||
|
function filter_boards(search_string) {
|
||||||
|
$(".board_hidden").removeClass("board_hidden");
|
||||||
|
$(".this_module").removeClass("this_module");
|
||||||
|
var nboards = $(".support-matrix-table tbody tr").length;
|
||||||
|
if(search_string.trim() == "") {
|
||||||
|
$("#support-matrix-filter-num").html("(all)");
|
||||||
|
setURL("filter",[]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var list_search = search_string.split(" ").filter(i => i);
|
||||||
|
var nvisible = 0;
|
||||||
|
$(".support-matrix-table tbody tr").each( (index,item) => {
|
||||||
|
var name = $(item).find("td:first-child p").html();
|
||||||
|
var modules = $(item).find("a.reference.internal");
|
||||||
|
var matching_all = true;
|
||||||
|
//
|
||||||
|
list_search.forEach((sstring) => {
|
||||||
|
var matching = (sstring[0] == "-");
|
||||||
|
for(var modi = 0; modi < modules.length; ++modi) {
|
||||||
|
module = modules[modi];
|
||||||
|
var mod_name = module.firstChild.firstChild.textContent;
|
||||||
|
if(sstring[0] == "-") {
|
||||||
|
if(mod_name.match(sstring.substr(1))) {
|
||||||
|
matching = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if(mod_name.match(sstring)) {
|
||||||
|
$(module).addClass("this_module");
|
||||||
|
matching = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
matching_all = matching_all && matching;
|
||||||
|
});
|
||||||
|
if(!matching_all) {
|
||||||
|
$(item).addClass("board_hidden");
|
||||||
|
} else {
|
||||||
|
nvisible += 1;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$("#support-matrix-filter-num").html(`(${nvisible}/${nboards})`);
|
||||||
|
setURL("filter",list_search);
|
||||||
|
}
|
||||||
|
|
||||||
|
function run_filter() {
|
||||||
|
var search_string = $("#support-matrix-filter").val();
|
||||||
|
filter_boards(search_string);
|
||||||
|
}
|
||||||
|
$("#support-matrix-filter").on("keyup", run_filter);
|
||||||
|
// $(document).on("keyup", "#support-matrix-filter", run_filter);
|
||||||
|
handlePageLoad();
|
||||||
|
});
|
6
docs/templates/replace.inc
vendored
6
docs/templates/replace.inc
vendored
@ -4,6 +4,6 @@
|
|||||||
|
|
||||||
.. |see_cpython_module| replace::
|
.. |see_cpython_module| replace::
|
||||||
|
|
||||||
*This module implements a subset of the corresponding* ``CPython`` *module,
|
*This module implements a subset of the corresponding* :term:`CPython` *module,
|
||||||
as described below. For more information, refer to the original*
|
as described below. For more information, refer to the original
|
||||||
``CPython`` *documentation:*
|
CPython documentation:*
|
||||||
|
@ -1,203 +0,0 @@
|
|||||||
/*
|
|
||||||
* This file is part of the MicroPython project, http://micropython.org/
|
|
||||||
*
|
|
||||||
* The MIT License (MIT)
|
|
||||||
*
|
|
||||||
* SPDX-FileCopyrightText: Copyright (c) 2017-2018 Damien P. George
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "drivers/bus/qspi.h"
|
|
||||||
|
|
||||||
#define CS_LOW(self) mp_hal_pin_write(self->cs, 0)
|
|
||||||
#define CS_HIGH(self) mp_hal_pin_write(self->cs, 1)
|
|
||||||
|
|
||||||
#ifdef MICROPY_HW_SOFTQSPI_SCK_LOW
|
|
||||||
|
|
||||||
// Use externally provided functions for SCK control and IO reading
|
|
||||||
#define SCK_LOW(self) MICROPY_HW_SOFTQSPI_SCK_LOW(self)
|
|
||||||
#define SCK_HIGH(self) MICROPY_HW_SOFTQSPI_SCK_HIGH(self)
|
|
||||||
#define NIBBLE_READ(self) MICROPY_HW_SOFTQSPI_NIBBLE_READ(self)
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
// Use generic pin functions for SCK control and IO reading
|
|
||||||
#define SCK_LOW(self) mp_hal_pin_write(self->clk, 0)
|
|
||||||
#define SCK_HIGH(self) mp_hal_pin_write(self->clk, 1)
|
|
||||||
#define NIBBLE_READ(self) ( \
|
|
||||||
mp_hal_pin_read(self->io0) \
|
|
||||||
| (mp_hal_pin_read(self->io1) << 1) \
|
|
||||||
| (mp_hal_pin_read(self->io2) << 2) \
|
|
||||||
| (mp_hal_pin_read(self->io3) << 3))
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
STATIC void nibble_write(mp_soft_qspi_obj_t *self, uint8_t v) {
|
|
||||||
mp_hal_pin_write(self->io0, v & 1);
|
|
||||||
mp_hal_pin_write(self->io1, (v >> 1) & 1);
|
|
||||||
mp_hal_pin_write(self->io2, (v >> 2) & 1);
|
|
||||||
mp_hal_pin_write(self->io3, (v >> 3) & 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC int mp_soft_qspi_ioctl(void *self_in, uint32_t cmd) {
|
|
||||||
mp_soft_qspi_obj_t *self = (mp_soft_qspi_obj_t *)self_in;
|
|
||||||
|
|
||||||
switch (cmd) {
|
|
||||||
case MP_QSPI_IOCTL_INIT:
|
|
||||||
mp_hal_pin_high(self->cs);
|
|
||||||
mp_hal_pin_output(self->cs);
|
|
||||||
|
|
||||||
// Configure pins
|
|
||||||
mp_hal_pin_write(self->clk, 0);
|
|
||||||
mp_hal_pin_output(self->clk);
|
|
||||||
// mp_hal_pin_write(self->clk, 1);
|
|
||||||
mp_hal_pin_output(self->io0);
|
|
||||||
mp_hal_pin_input(self->io1);
|
|
||||||
mp_hal_pin_write(self->io2, 1);
|
|
||||||
mp_hal_pin_output(self->io2);
|
|
||||||
mp_hal_pin_write(self->io3, 1);
|
|
||||||
mp_hal_pin_output(self->io3);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0; // success
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void mp_soft_qspi_transfer(mp_soft_qspi_obj_t *self, size_t len, const uint8_t *src, uint8_t *dest) {
|
|
||||||
// Will run as fast as possible, limited only by CPU speed and GPIO time
|
|
||||||
mp_hal_pin_input(self->io1);
|
|
||||||
mp_hal_pin_output(self->io0);
|
|
||||||
if (self->io3) {
|
|
||||||
mp_hal_pin_write(self->io2, 1);
|
|
||||||
mp_hal_pin_output(self->io2);
|
|
||||||
mp_hal_pin_write(self->io3, 1);
|
|
||||||
mp_hal_pin_output(self->io3);
|
|
||||||
}
|
|
||||||
if (src) {
|
|
||||||
for (size_t i = 0; i < len; ++i) {
|
|
||||||
uint8_t data_out = src[i];
|
|
||||||
uint8_t data_in = 0;
|
|
||||||
for (int j = 0; j < 8; ++j, data_out <<= 1) {
|
|
||||||
mp_hal_pin_write(self->io0, (data_out >> 7) & 1);
|
|
||||||
mp_hal_pin_write(self->clk, 1);
|
|
||||||
data_in = (data_in << 1) | mp_hal_pin_read(self->io1);
|
|
||||||
mp_hal_pin_write(self->clk, 0);
|
|
||||||
}
|
|
||||||
if (dest != NULL) {
|
|
||||||
dest[i] = data_in;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for (size_t i = 0; i < len; ++i) {
|
|
||||||
uint8_t data_in = 0;
|
|
||||||
for (int j = 0; j < 8; ++j) {
|
|
||||||
mp_hal_pin_write(self->clk, 1);
|
|
||||||
data_in = (data_in << 1) | mp_hal_pin_read(self->io1);
|
|
||||||
mp_hal_pin_write(self->clk, 0);
|
|
||||||
}
|
|
||||||
if (dest != NULL) {
|
|
||||||
dest[i] = data_in;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void mp_soft_qspi_qread(mp_soft_qspi_obj_t *self, size_t len, uint8_t *buf) {
|
|
||||||
// Make all IO lines input
|
|
||||||
mp_hal_pin_input(self->io2);
|
|
||||||
mp_hal_pin_input(self->io3);
|
|
||||||
mp_hal_pin_input(self->io0);
|
|
||||||
mp_hal_pin_input(self->io1);
|
|
||||||
|
|
||||||
// Will run as fast as possible, limited only by CPU speed and GPIO time
|
|
||||||
while (len--) {
|
|
||||||
SCK_HIGH(self);
|
|
||||||
uint8_t data_in = NIBBLE_READ(self);
|
|
||||||
SCK_LOW(self);
|
|
||||||
SCK_HIGH(self);
|
|
||||||
*buf++ = (data_in << 4) | NIBBLE_READ(self);
|
|
||||||
SCK_LOW(self);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void mp_soft_qspi_qwrite(mp_soft_qspi_obj_t *self, size_t len, const uint8_t *buf) {
|
|
||||||
// Make all IO lines output
|
|
||||||
mp_hal_pin_output(self->io2);
|
|
||||||
mp_hal_pin_output(self->io3);
|
|
||||||
mp_hal_pin_output(self->io0);
|
|
||||||
mp_hal_pin_output(self->io1);
|
|
||||||
|
|
||||||
// Will run as fast as possible, limited only by CPU speed and GPIO time
|
|
||||||
for (size_t i = 0; i < len; ++i) {
|
|
||||||
nibble_write(self, buf[i] >> 4);
|
|
||||||
SCK_HIGH(self);
|
|
||||||
SCK_LOW(self);
|
|
||||||
|
|
||||||
nibble_write(self, buf[i]);
|
|
||||||
SCK_HIGH(self);
|
|
||||||
SCK_LOW(self);
|
|
||||||
}
|
|
||||||
|
|
||||||
// mp_hal_pin_input(self->io1);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void mp_soft_qspi_write_cmd_data(void *self_in, uint8_t cmd, size_t len, uint32_t data) {
|
|
||||||
mp_soft_qspi_obj_t *self = (mp_soft_qspi_obj_t *)self_in;
|
|
||||||
uint32_t cmd_buf = cmd | data << 8;
|
|
||||||
CS_LOW(self);
|
|
||||||
mp_soft_qspi_transfer(self, 1 + len, (uint8_t *)&cmd_buf, NULL);
|
|
||||||
CS_HIGH(self);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void mp_soft_qspi_write_cmd_addr_data(void *self_in, uint8_t cmd, uint32_t addr, size_t len, const uint8_t *src) {
|
|
||||||
mp_soft_qspi_obj_t *self = (mp_soft_qspi_obj_t *)self_in;
|
|
||||||
uint8_t cmd_buf[4] = {cmd, addr >> 16, addr >> 8, addr};
|
|
||||||
CS_LOW(self);
|
|
||||||
mp_soft_qspi_transfer(self, 4, cmd_buf, NULL);
|
|
||||||
mp_soft_qspi_transfer(self, len, src, NULL);
|
|
||||||
CS_HIGH(self);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC uint32_t mp_soft_qspi_read_cmd(void *self_in, uint8_t cmd, size_t len) {
|
|
||||||
mp_soft_qspi_obj_t *self = (mp_soft_qspi_obj_t *)self_in;
|
|
||||||
uint32_t cmd_buf = cmd;
|
|
||||||
CS_LOW(self);
|
|
||||||
mp_soft_qspi_transfer(self, 1 + len, (uint8_t *)&cmd_buf, (uint8_t *)&cmd_buf);
|
|
||||||
CS_HIGH(self);
|
|
||||||
return cmd_buf >> 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void mp_soft_qspi_read_cmd_qaddr_qdata(void *self_in, uint8_t cmd, uint32_t addr, size_t len, uint8_t *dest) {
|
|
||||||
mp_soft_qspi_obj_t *self = (mp_soft_qspi_obj_t *)self_in;
|
|
||||||
uint8_t cmd_buf[7] = {cmd, addr >> 16, addr >> 8, addr};
|
|
||||||
CS_LOW(self);
|
|
||||||
mp_soft_qspi_transfer(self, 1, cmd_buf, NULL);
|
|
||||||
mp_soft_qspi_qwrite(self, 6, &cmd_buf[1]); // 3 addr bytes, 1 extra byte (0), 2 dummy bytes (4 dummy cycles)
|
|
||||||
mp_soft_qspi_qread(self, len, dest);
|
|
||||||
CS_HIGH(self);
|
|
||||||
}
|
|
||||||
|
|
||||||
const mp_qspi_proto_t mp_soft_qspi_proto = {
|
|
||||||
.ioctl = mp_soft_qspi_ioctl,
|
|
||||||
.write_cmd_data = mp_soft_qspi_write_cmd_data,
|
|
||||||
.write_cmd_addr_data = mp_soft_qspi_write_cmd_addr_data,
|
|
||||||
.read_cmd = mp_soft_qspi_read_cmd,
|
|
||||||
.read_cmd_qaddr_qdata = mp_soft_qspi_read_cmd_qaddr_qdata,
|
|
||||||
};
|
|
@ -1,105 +0,0 @@
|
|||||||
/*
|
|
||||||
* This file is part of the MicroPython project, http://micropython.org/
|
|
||||||
*
|
|
||||||
* The MIT License (MIT)
|
|
||||||
*
|
|
||||||
* SPDX-FileCopyrightText: Copyright (c) 2016-2018 Damien P. George
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "drivers/bus/spi.h"
|
|
||||||
|
|
||||||
int mp_soft_spi_ioctl(void *self_in, uint32_t cmd) {
|
|
||||||
mp_soft_spi_obj_t *self = (mp_soft_spi_obj_t *)self_in;
|
|
||||||
|
|
||||||
switch (cmd) {
|
|
||||||
case MP_SPI_IOCTL_INIT:
|
|
||||||
mp_hal_pin_write(self->sck, self->polarity);
|
|
||||||
mp_hal_pin_output(self->sck);
|
|
||||||
mp_hal_pin_output(self->mosi);
|
|
||||||
mp_hal_pin_input(self->miso);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case MP_SPI_IOCTL_DEINIT:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void mp_soft_spi_transfer(void *self_in, size_t len, const uint8_t *src, uint8_t *dest) {
|
|
||||||
mp_soft_spi_obj_t *self = (mp_soft_spi_obj_t *)self_in;
|
|
||||||
uint32_t delay_half = self->delay_half;
|
|
||||||
|
|
||||||
// only MSB transfer is implemented
|
|
||||||
|
|
||||||
// If a port defines MICROPY_HW_SOFTSPI_MIN_DELAY, and the configured
|
|
||||||
// delay_half is equal to this value, then the software SPI implementation
|
|
||||||
// will run as fast as possible, limited only by CPU speed and GPIO time.
|
|
||||||
#ifdef MICROPY_HW_SOFTSPI_MIN_DELAY
|
|
||||||
if (delay_half == MICROPY_HW_SOFTSPI_MIN_DELAY) {
|
|
||||||
for (size_t i = 0; i < len; ++i) {
|
|
||||||
uint8_t data_out = src[i];
|
|
||||||
uint8_t data_in = 0;
|
|
||||||
for (int j = 0; j < 8; ++j, data_out <<= 1) {
|
|
||||||
mp_hal_pin_write(self->mosi, (data_out >> 7) & 1);
|
|
||||||
mp_hal_pin_write(self->sck, 1 - self->polarity);
|
|
||||||
data_in = (data_in << 1) | mp_hal_pin_read(self->miso);
|
|
||||||
mp_hal_pin_write(self->sck, self->polarity);
|
|
||||||
}
|
|
||||||
if (dest != NULL) {
|
|
||||||
dest[i] = data_in;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
for (size_t i = 0; i < len; ++i) {
|
|
||||||
uint8_t data_out = src[i];
|
|
||||||
uint8_t data_in = 0;
|
|
||||||
for (int j = 0; j < 8; ++j, data_out <<= 1) {
|
|
||||||
mp_hal_pin_write(self->mosi, (data_out >> 7) & 1);
|
|
||||||
if (self->phase == 0) {
|
|
||||||
mp_hal_delay_us_fast(delay_half);
|
|
||||||
mp_hal_pin_write(self->sck, 1 - self->polarity);
|
|
||||||
} else {
|
|
||||||
mp_hal_pin_write(self->sck, 1 - self->polarity);
|
|
||||||
mp_hal_delay_us_fast(delay_half);
|
|
||||||
}
|
|
||||||
data_in = (data_in << 1) | mp_hal_pin_read(self->miso);
|
|
||||||
if (self->phase == 0) {
|
|
||||||
mp_hal_delay_us_fast(delay_half);
|
|
||||||
mp_hal_pin_write(self->sck, self->polarity);
|
|
||||||
} else {
|
|
||||||
mp_hal_pin_write(self->sck, self->polarity);
|
|
||||||
mp_hal_delay_us_fast(delay_half);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (dest != NULL) {
|
|
||||||
dest[i] = data_in;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const mp_spi_proto_t mp_soft_spi_proto = {
|
|
||||||
.ioctl = mp_soft_spi_ioctl,
|
|
||||||
.transfer = mp_soft_spi_transfer,
|
|
||||||
};
|
|
@ -1,6 +0,0 @@
|
|||||||
This is the driver for the WIZnet5x00 series of Ethernet controllers.
|
|
||||||
|
|
||||||
Adapted for MicroPython.
|
|
||||||
|
|
||||||
Original source: https://github.com/Wiznet/W5500_EVB/tree/master/ioLibrary
|
|
||||||
Taken on: 30 August 2014
|
|
@ -1,718 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file socket.c
|
|
||||||
//! \brief SOCKET APIs Implements file.
|
|
||||||
//! \details SOCKET APIs like as Berkeley Socket APIs.
|
|
||||||
//! \version 1.0.3
|
|
||||||
//! \date 2013/10/21
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2018/10/09> Nick Moore fixes for CircuitPython
|
|
||||||
//! <2014/05/01> V1.0.3. Refer to M20140501
|
|
||||||
//! 1. Implicit type casting -> Explicit type casting.
|
|
||||||
//! 2. replace 0x01 with PACK_REMAINED in recvfrom()
|
|
||||||
//! 3. Validation a destination ip in connect() & sendto():
|
|
||||||
//! It occurs a fatal error on converting unint32 address if uint8* addr parameter is not aligned by 4byte address.
|
|
||||||
//! Copy 4 byte addr value into temporary uint32 variable and then compares it.
|
|
||||||
//! <2013/12/20> V1.0.2 Refer to M20131220
|
|
||||||
//! Remove Warning.
|
|
||||||
//! <2013/11/04> V1.0.1 2nd Release. Refer to "20131104".
|
|
||||||
//! In sendto(), Add to clear timeout interrupt status (Sn_IR_TIMEOUT)
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! \author MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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 <string.h>
|
|
||||||
|
|
||||||
#include "py/mpthread.h"
|
|
||||||
#include "socket.h"
|
|
||||||
|
|
||||||
#define SOCK_ANY_PORT_NUM 0xC000;
|
|
||||||
|
|
||||||
static uint16_t sock_any_port = SOCK_ANY_PORT_NUM;
|
|
||||||
static uint16_t sock_io_mode = 0;
|
|
||||||
static uint16_t sock_is_sending = 0;
|
|
||||||
static uint16_t sock_remained_size[_WIZCHIP_SOCK_NUM_] = {0,0,};
|
|
||||||
static uint8_t sock_pack_info[_WIZCHIP_SOCK_NUM_] = {0,};
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
static uint16_t sock_next_rd[_WIZCHIP_SOCK_NUM_] ={0,};
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define CHECK_SOCKNUM() \
|
|
||||||
do{ \
|
|
||||||
if(sn > _WIZCHIP_SOCK_NUM_) return SOCKERR_SOCKNUM; \
|
|
||||||
}while(0); \
|
|
||||||
|
|
||||||
#define CHECK_SOCKMODE(mode) \
|
|
||||||
do{ \
|
|
||||||
if((getSn_MR(sn) & 0x0F) != mode) return SOCKERR_SOCKMODE; \
|
|
||||||
}while(0); \
|
|
||||||
|
|
||||||
#define CHECK_SOCKINIT() \
|
|
||||||
do{ \
|
|
||||||
if((getSn_SR(sn) != SOCK_INIT)) return SOCKERR_SOCKINIT; \
|
|
||||||
}while(0); \
|
|
||||||
|
|
||||||
#define CHECK_SOCKDATA() \
|
|
||||||
do{ \
|
|
||||||
if(len == 0) return SOCKERR_DATALEN; \
|
|
||||||
}while(0); \
|
|
||||||
|
|
||||||
void WIZCHIP_EXPORT(socket_reset)(void) {
|
|
||||||
sock_any_port = SOCK_ANY_PORT_NUM;
|
|
||||||
sock_io_mode = 0;
|
|
||||||
sock_is_sending = 0;
|
|
||||||
/*
|
|
||||||
memset(sock_remained_size, 0, _WIZCHIP_SOCK_NUM_ * sizeof(uint16_t));
|
|
||||||
memset(sock_pack_info, 0, _WIZCHIP_SOCK_NUM_ * sizeof(uint8_t));
|
|
||||||
*/
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
memset(sock_next_rd, 0, _WIZCHIP_SOCK_NUM_ * sizeof(uint16_t));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(socket)(uint8_t sn, uint8_t protocol, uint16_t port, uint8_t flag)
|
|
||||||
{
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
switch(protocol)
|
|
||||||
{
|
|
||||||
case Sn_MR_TCP :
|
|
||||||
case Sn_MR_UDP :
|
|
||||||
case Sn_MR_MACRAW :
|
|
||||||
break;
|
|
||||||
#if ( _WIZCHIP_ < 5200 )
|
|
||||||
case Sn_MR_IPRAW :
|
|
||||||
case Sn_MR_PPPoE :
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
default :
|
|
||||||
return SOCKERR_SOCKMODE;
|
|
||||||
}
|
|
||||||
if((flag & 0x06) != 0) return SOCKERR_SOCKFLAG;
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
if(flag & 0x10) return SOCKERR_SOCKFLAG;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if(flag != 0)
|
|
||||||
{
|
|
||||||
switch(protocol)
|
|
||||||
{
|
|
||||||
case Sn_MR_TCP:
|
|
||||||
if((flag & (SF_TCP_NODELAY|SF_IO_NONBLOCK))==0) return SOCKERR_SOCKFLAG;
|
|
||||||
break;
|
|
||||||
case Sn_MR_UDP:
|
|
||||||
if(flag & SF_IGMP_VER2)
|
|
||||||
{
|
|
||||||
if((flag & SF_MULTI_ENABLE)==0) return SOCKERR_SOCKFLAG;
|
|
||||||
}
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
if(flag & SF_UNI_BLOCK)
|
|
||||||
{
|
|
||||||
if((flag & SF_MULTI_ENABLE) == 0) return SOCKERR_SOCKFLAG;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
setSn_MR(sn, (protocol | (flag & 0xF0)));
|
|
||||||
if(!port)
|
|
||||||
{
|
|
||||||
port = sock_any_port++;
|
|
||||||
if(sock_any_port == 0xFFF0) sock_any_port = SOCK_ANY_PORT_NUM;
|
|
||||||
}
|
|
||||||
setSn_PORT(sn,port);
|
|
||||||
setSn_CR(sn,Sn_CR_OPEN);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
sock_io_mode |= ((flag & SF_IO_NONBLOCK) << sn);
|
|
||||||
sock_is_sending &= ~(1<<sn);
|
|
||||||
sock_remained_size[sn] = 0;
|
|
||||||
sock_pack_info[sn] = 0;
|
|
||||||
while(getSn_SR(sn) == SOCK_CLOSED);
|
|
||||||
return (int8_t)sn;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(close)(uint8_t sn)
|
|
||||||
{
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
|
|
||||||
setSn_CR(sn,Sn_CR_CLOSE);
|
|
||||||
/* wait to process the command... */
|
|
||||||
while( getSn_CR(sn) );
|
|
||||||
/* clear all interrupt of the socket. */
|
|
||||||
setSn_IR(sn, 0xFF);
|
|
||||||
sock_is_sending &= ~(1<<sn);
|
|
||||||
sock_remained_size[sn] = 0;
|
|
||||||
sock_pack_info[sn] = 0;
|
|
||||||
while(getSn_SR(sn) != SOCK_CLOSED);
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(listen)(uint8_t sn)
|
|
||||||
{
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
CHECK_SOCKINIT();
|
|
||||||
setSn_CR(sn,Sn_CR_LISTEN);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
while(getSn_SR(sn) != SOCK_LISTEN)
|
|
||||||
{
|
|
||||||
if(getSn_CR(sn) == SOCK_CLOSED)
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return SOCKERR_SOCKCLOSED;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(connect)(uint8_t sn, uint8_t * addr, uint16_t port)
|
|
||||||
{
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
CHECK_SOCKINIT();
|
|
||||||
//M20140501 : For avoiding fatal error on memory align mismatched
|
|
||||||
//if( *((uint32_t*)addr) == 0xFFFFFFFF || *((uint32_t*)addr) == 0) return SOCKERR_IPINVALID;
|
|
||||||
{
|
|
||||||
uint32_t taddr;
|
|
||||||
taddr = ((uint32_t)addr[0] & 0x000000FF);
|
|
||||||
taddr = (taddr << 8) + ((uint32_t)addr[1] & 0x000000FF);
|
|
||||||
taddr = (taddr << 8) + ((uint32_t)addr[2] & 0x000000FF);
|
|
||||||
taddr = (taddr << 8) + ((uint32_t)addr[3] & 0x000000FF);
|
|
||||||
if (taddr == 0xFFFFFFFF || taddr == 0) return SOCKERR_IPINVALID;
|
|
||||||
}
|
|
||||||
//
|
|
||||||
|
|
||||||
if(port == 0) return SOCKERR_PORTZERO;
|
|
||||||
setSn_DIPR(sn,addr);
|
|
||||||
setSn_DPORT(sn,port);
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR(wizchip_getsubn());
|
|
||||||
#endif
|
|
||||||
setSn_CR(sn,Sn_CR_CONNECT);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
if(sock_io_mode & (1<<sn)) return SOCK_BUSY;
|
|
||||||
while(getSn_SR(sn) != SOCK_ESTABLISHED)
|
|
||||||
{
|
|
||||||
if (getSn_SR(sn) == SOCK_CLOSED) {
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR((uint8_t*)"\x00\x00\x00\x00");
|
|
||||||
#endif
|
|
||||||
return SOCKERR_SOCKCLOSED;
|
|
||||||
}
|
|
||||||
if (getSn_IR(sn) & Sn_IR_TIMEOUT)
|
|
||||||
{
|
|
||||||
setSn_IR(sn, Sn_IR_TIMEOUT);
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR((uint8_t*)"\x00\x00\x00\x00");
|
|
||||||
#endif
|
|
||||||
return SOCKERR_TIMEOUT;
|
|
||||||
}
|
|
||||||
MICROPY_THREAD_YIELD();
|
|
||||||
}
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR((uint8_t*)"\x00\x00\x00\x00");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(disconnect)(uint8_t sn)
|
|
||||||
{
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
setSn_CR(sn,Sn_CR_DISCON);
|
|
||||||
/* wait to process the command... */
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
sock_is_sending &= ~(1<<sn);
|
|
||||||
if(sock_io_mode & (1<<sn)) return SOCK_BUSY;
|
|
||||||
while(getSn_SR(sn) != SOCK_CLOSED)
|
|
||||||
{
|
|
||||||
if(getSn_IR(sn) & Sn_IR_TIMEOUT)
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return SOCKERR_TIMEOUT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int32_t WIZCHIP_EXPORT(send)(uint8_t sn, uint8_t * buf, uint16_t len)
|
|
||||||
{
|
|
||||||
uint8_t tmp=0;
|
|
||||||
uint16_t freesize=0;
|
|
||||||
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
CHECK_SOCKDATA();
|
|
||||||
tmp = getSn_SR(sn);
|
|
||||||
if(tmp != SOCK_ESTABLISHED && tmp != SOCK_CLOSE_WAIT) return SOCKERR_SOCKSTATUS;
|
|
||||||
if( sock_is_sending & (1<<sn) )
|
|
||||||
{
|
|
||||||
tmp = getSn_IR(sn);
|
|
||||||
if(tmp & Sn_IR_SENDOK)
|
|
||||||
{
|
|
||||||
setSn_IR(sn, Sn_IR_SENDOK);
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
if(getSn_TX_RD(sn) != sock_next_rd[sn])
|
|
||||||
{
|
|
||||||
setSn_CR(sn,Sn_CR_SEND);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
return SOCKERR_BUSY;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
sock_is_sending &= ~(1<<sn);
|
|
||||||
}
|
|
||||||
else if(tmp & Sn_IR_TIMEOUT)
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return SOCKERR_TIMEOUT;
|
|
||||||
}
|
|
||||||
else return SOCK_BUSY;
|
|
||||||
}
|
|
||||||
freesize = getSn_TxMAX(sn);
|
|
||||||
if (len > freesize) len = freesize; // check size not to exceed MAX size.
|
|
||||||
while(1)
|
|
||||||
{
|
|
||||||
freesize = getSn_TX_FSR(sn);
|
|
||||||
tmp = getSn_SR(sn);
|
|
||||||
if ((tmp != SOCK_ESTABLISHED) && (tmp != SOCK_CLOSE_WAIT))
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return SOCKERR_SOCKSTATUS;
|
|
||||||
}
|
|
||||||
if( (sock_io_mode & (1<<sn)) && (len > freesize) ) return SOCK_BUSY;
|
|
||||||
if(len <= freesize) break;
|
|
||||||
MICROPY_THREAD_YIELD();
|
|
||||||
}
|
|
||||||
wiz_send_data(sn, buf, len);
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
sock_next_rd[sn] = getSn_TX_RD(sn) + len;
|
|
||||||
#endif
|
|
||||||
setSn_CR(sn,Sn_CR_SEND);
|
|
||||||
/* wait to process the command... */
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
sock_is_sending |= (1 << sn);
|
|
||||||
return len;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int32_t WIZCHIP_EXPORT(recv)(uint8_t sn, uint8_t * buf, uint16_t len)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
uint16_t recvsize = 0;
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
CHECK_SOCKDATA();
|
|
||||||
|
|
||||||
recvsize = getSn_RxMAX(sn);
|
|
||||||
if(recvsize < len) len = recvsize;
|
|
||||||
while(1)
|
|
||||||
{
|
|
||||||
recvsize = getSn_RX_RSR(sn);
|
|
||||||
tmp = getSn_SR(sn);
|
|
||||||
if (tmp != SOCK_ESTABLISHED)
|
|
||||||
{
|
|
||||||
if(tmp == SOCK_CLOSE_WAIT)
|
|
||||||
{
|
|
||||||
if(recvsize != 0) break;
|
|
||||||
else if(getSn_TX_FSR(sn) == getSn_TxMAX(sn))
|
|
||||||
{
|
|
||||||
// dpgeorge: Getting here seems to be an orderly shutdown of the
|
|
||||||
// socket, and trying to get POSIX behaviour we return 0 because:
|
|
||||||
// "If no messages are available to be received and the peer has per‐
|
|
||||||
// formed an orderly shutdown, recv() shall return 0".
|
|
||||||
// TODO this return value clashes with SOCK_BUSY in non-blocking mode.
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return SOCKERR_SOCKSTATUS;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if((sock_io_mode & (1<<sn)) && (recvsize == 0)) return SOCK_BUSY;
|
|
||||||
if(recvsize != 0) break;
|
|
||||||
MICROPY_THREAD_YIELD();
|
|
||||||
};
|
|
||||||
if(recvsize < len) len = recvsize;
|
|
||||||
wiz_recv_data(sn, buf, len);
|
|
||||||
setSn_CR(sn,Sn_CR_RECV);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
return len;
|
|
||||||
}
|
|
||||||
|
|
||||||
int32_t WIZCHIP_EXPORT(sendto)(uint8_t sn, uint8_t * buf, uint16_t len, uint8_t * addr, uint16_t port)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
uint16_t freesize = 0;
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
switch(getSn_MR(sn) & 0x0F)
|
|
||||||
{
|
|
||||||
case Sn_MR_UDP:
|
|
||||||
case Sn_MR_MACRAW:
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return SOCKERR_SOCKMODE;
|
|
||||||
}
|
|
||||||
CHECK_SOCKDATA();
|
|
||||||
//M20140501 : For avoiding fatal error on memory align mismatched
|
|
||||||
//if(*((uint32_t*)addr) == 0) return SOCKERR_IPINVALID;
|
|
||||||
if ((addr[0] | addr[1] | addr[2] | addr[3]) == 0) return SOCKERR_IPINVALID;
|
|
||||||
if(port == 0) return SOCKERR_PORTZERO;
|
|
||||||
tmp = getSn_SR(sn);
|
|
||||||
if(tmp != SOCK_MACRAW && tmp != SOCK_UDP) return SOCKERR_SOCKSTATUS;
|
|
||||||
|
|
||||||
setSn_DIPR(sn,addr);
|
|
||||||
setSn_DPORT(sn,port);
|
|
||||||
freesize = getSn_TxMAX(sn);
|
|
||||||
if (len > freesize) len = freesize; // check size not to exceed MAX size.
|
|
||||||
while(1)
|
|
||||||
{
|
|
||||||
freesize = getSn_TX_FSR(sn);
|
|
||||||
if(getSn_SR(sn) == SOCK_CLOSED) return SOCKERR_SOCKCLOSED;
|
|
||||||
if( (sock_io_mode & (1<<sn)) && (len > freesize) ) return SOCK_BUSY;
|
|
||||||
if(len <= freesize) break;
|
|
||||||
MICROPY_THREAD_YIELD();
|
|
||||||
};
|
|
||||||
wiz_send_data(sn, buf, len);
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR(wizchip_getsubn());
|
|
||||||
#endif
|
|
||||||
|
|
||||||
setSn_CR(sn,Sn_CR_SEND);
|
|
||||||
/* wait to process the command... */
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
while(1)
|
|
||||||
{
|
|
||||||
tmp = getSn_IR(sn);
|
|
||||||
if(tmp & Sn_IR_SENDOK)
|
|
||||||
{
|
|
||||||
setSn_IR(sn, Sn_IR_SENDOK);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
//M:20131104
|
|
||||||
//else if(tmp & Sn_IR_TIMEOUT) return SOCKERR_TIMEOUT;
|
|
||||||
else if(tmp & Sn_IR_TIMEOUT)
|
|
||||||
{
|
|
||||||
setSn_IR(sn, Sn_IR_TIMEOUT);
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR((uint8_t*)"\x00\x00\x00\x00");
|
|
||||||
#endif
|
|
||||||
return SOCKERR_TIMEOUT;
|
|
||||||
}
|
|
||||||
////////////
|
|
||||||
MICROPY_THREAD_YIELD();
|
|
||||||
}
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
setSUBR((uint8_t*)"\x00\x00\x00\x00");
|
|
||||||
#endif
|
|
||||||
return len;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
int32_t WIZCHIP_EXPORT(recvfrom)(uint8_t sn, uint8_t * buf, uint16_t len, uint8_t * addr, uint16_t *port)
|
|
||||||
{
|
|
||||||
uint8_t mr;
|
|
||||||
uint8_t head[8];
|
|
||||||
uint16_t pack_len=0;
|
|
||||||
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
//CHECK_SOCKMODE(Sn_MR_UDP);
|
|
||||||
switch((mr=getSn_MR(sn)) & 0x0F)
|
|
||||||
{
|
|
||||||
case Sn_MR_UDP:
|
|
||||||
case Sn_MR_MACRAW:
|
|
||||||
break;
|
|
||||||
#if ( _WIZCHIP_ < 5200 )
|
|
||||||
case Sn_MR_IPRAW:
|
|
||||||
case Sn_MR_PPPoE:
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
default:
|
|
||||||
return SOCKERR_SOCKMODE;
|
|
||||||
}
|
|
||||||
CHECK_SOCKDATA();
|
|
||||||
if(sock_remained_size[sn] == 0)
|
|
||||||
{
|
|
||||||
while(1)
|
|
||||||
{
|
|
||||||
pack_len = getSn_RX_RSR(sn);
|
|
||||||
if(getSn_SR(sn) == SOCK_CLOSED) return SOCKERR_SOCKCLOSED;
|
|
||||||
if( (sock_io_mode & (1<<sn)) && (pack_len == 0) ) return SOCK_BUSY;
|
|
||||||
if(pack_len != 0) break;
|
|
||||||
MICROPY_THREAD_YIELD();
|
|
||||||
};
|
|
||||||
}
|
|
||||||
sock_pack_info[sn] = PACK_COMPLETED;
|
|
||||||
switch (mr & 0x07)
|
|
||||||
{
|
|
||||||
case Sn_MR_UDP :
|
|
||||||
if(sock_remained_size[sn] == 0)
|
|
||||||
{
|
|
||||||
wiz_recv_data(sn, head, 8);
|
|
||||||
setSn_CR(sn,Sn_CR_RECV);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
// read peer's IP address, port number & packet length
|
|
||||||
addr[0] = head[0];
|
|
||||||
addr[1] = head[1];
|
|
||||||
addr[2] = head[2];
|
|
||||||
addr[3] = head[3];
|
|
||||||
*port = head[4];
|
|
||||||
*port = (*port << 8) + head[5];
|
|
||||||
sock_remained_size[sn] = head[6];
|
|
||||||
sock_remained_size[sn] = (sock_remained_size[sn] << 8) + head[7];
|
|
||||||
sock_pack_info[sn] = PACK_FIRST;
|
|
||||||
}
|
|
||||||
if(len < sock_remained_size[sn]) pack_len = len;
|
|
||||||
else pack_len = sock_remained_size[sn];
|
|
||||||
//
|
|
||||||
// Need to packet length check (default 1472)
|
|
||||||
//
|
|
||||||
wiz_recv_data(sn, buf, pack_len); // data copy.
|
|
||||||
break;
|
|
||||||
case Sn_MR_MACRAW :
|
|
||||||
if(sock_remained_size[sn] == 0)
|
|
||||||
{
|
|
||||||
wiz_recv_data(sn, head, 2);
|
|
||||||
setSn_CR(sn,Sn_CR_RECV);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
// read peer's IP address, port number & packet length
|
|
||||||
sock_remained_size[sn] = head[0];
|
|
||||||
sock_remained_size[sn] = (sock_remained_size[sn] <<8) + head[1];
|
|
||||||
sock_remained_size[sn] -= 2; // len includes 2 len bytes
|
|
||||||
if(sock_remained_size[sn] > 1514)
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(sn);
|
|
||||||
return SOCKFATAL_PACKLEN;
|
|
||||||
}
|
|
||||||
sock_pack_info[sn] = PACK_FIRST;
|
|
||||||
}
|
|
||||||
if(len < sock_remained_size[sn]) pack_len = len;
|
|
||||||
else pack_len = sock_remained_size[sn];
|
|
||||||
wiz_recv_data(sn,buf,pack_len);
|
|
||||||
break;
|
|
||||||
#if ( _WIZCHIP_ < 5200 )
|
|
||||||
case Sn_MR_IPRAW:
|
|
||||||
if(sock_remained_size[sn] == 0)
|
|
||||||
{
|
|
||||||
wiz_recv_data(sn, head, 6);
|
|
||||||
setSn_CR(sn,Sn_CR_RECV);
|
|
||||||
while(getSn_CR(sn));
|
|
||||||
addr[0] = head[0];
|
|
||||||
addr[1] = head[1];
|
|
||||||
addr[2] = head[2];
|
|
||||||
addr[3] = head[3];
|
|
||||||
sock_remained_size[sn] = head[4];
|
|
||||||
sock_remaiend_size[sn] = (sock_remained_size[sn] << 8) + head[5];
|
|
||||||
sock_pack_info[sn] = PACK_FIRST;
|
|
||||||
}
|
|
||||||
//
|
|
||||||
// Need to packet length check
|
|
||||||
//
|
|
||||||
if(len < sock_remained_size[sn]) pack_len = len;
|
|
||||||
else pack_len = sock_remained_size[sn];
|
|
||||||
wiz_recv_data(sn, buf, pack_len); // data copy.
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
default:
|
|
||||||
wiz_recv_ignore(sn, pack_len); // data copy.
|
|
||||||
sock_remained_size[sn] = pack_len;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
setSn_CR(sn,Sn_CR_RECV);
|
|
||||||
/* wait to process the command... */
|
|
||||||
while(getSn_CR(sn)) ;
|
|
||||||
sock_remained_size[sn] -= pack_len;
|
|
||||||
//M20140501 : replace 0x01 with PACK_REMAINED
|
|
||||||
//if(sock_remained_size[sn] != 0) sock_pack_info[sn] |= 0x01;
|
|
||||||
if(sock_remained_size[sn] != 0) sock_pack_info[sn] |= PACK_REMAINED;
|
|
||||||
//
|
|
||||||
return pack_len;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(ctlsocket)(uint8_t sn, ctlsock_type cstype, void* arg)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
switch(cstype)
|
|
||||||
{
|
|
||||||
case CS_SET_IOMODE:
|
|
||||||
tmp = *((uint8_t*)arg);
|
|
||||||
if(tmp == SOCK_IO_NONBLOCK) sock_io_mode |= (1<<sn);
|
|
||||||
else if(tmp == SOCK_IO_BLOCK) sock_io_mode &= ~(1<<sn);
|
|
||||||
else return SOCKERR_ARG;
|
|
||||||
break;
|
|
||||||
case CS_GET_IOMODE:
|
|
||||||
//M20140501 : implict type casting -> explict type casting
|
|
||||||
//*((uint8_t*)arg) = (sock_io_mode >> sn) & 0x0001;
|
|
||||||
*((uint8_t*)arg) = (uint8_t)((sock_io_mode >> sn) & 0x0001);
|
|
||||||
//
|
|
||||||
break;
|
|
||||||
case CS_GET_MAXTXBUF:
|
|
||||||
*((uint16_t*)arg) = getSn_TxMAX(sn);
|
|
||||||
break;
|
|
||||||
case CS_GET_MAXRXBUF:
|
|
||||||
*((uint16_t*)arg) = getSn_RxMAX(sn);
|
|
||||||
break;
|
|
||||||
case CS_CLR_INTERRUPT:
|
|
||||||
if( (*(uint8_t*)arg) > SIK_ALL) return SOCKERR_ARG;
|
|
||||||
setSn_IR(sn,*(uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
case CS_GET_INTERRUPT:
|
|
||||||
*((uint8_t*)arg) = getSn_IR(sn);
|
|
||||||
break;
|
|
||||||
case CS_SET_INTMASK:
|
|
||||||
if( (*(uint8_t*)arg) > SIK_ALL) return SOCKERR_ARG;
|
|
||||||
setSn_IMR(sn,*(uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
case CS_GET_INTMASK:
|
|
||||||
*((uint8_t*)arg) = getSn_IMR(sn);
|
|
||||||
default:
|
|
||||||
return SOCKERR_ARG;
|
|
||||||
}
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(setsockopt)(uint8_t sn, sockopt_type sotype, void* arg)
|
|
||||||
{
|
|
||||||
// M20131220 : Remove warning
|
|
||||||
//uint8_t tmp;
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
switch(sotype)
|
|
||||||
{
|
|
||||||
case SO_TTL:
|
|
||||||
setSn_TTL(sn,*(uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
case SO_TOS:
|
|
||||||
setSn_TOS(sn,*(uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
case SO_MSS:
|
|
||||||
setSn_MSSR(sn,*(uint16_t*)arg);
|
|
||||||
break;
|
|
||||||
case SO_DESTIP:
|
|
||||||
setSn_DIPR(sn, (uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
case SO_DESTPORT:
|
|
||||||
setSn_DPORT(sn, *(uint16_t*)arg);
|
|
||||||
break;
|
|
||||||
#if _WIZCHIP_ != 5100
|
|
||||||
case SO_KEEPALIVESEND:
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
#if _WIZCHIP_ > 5200
|
|
||||||
if(getSn_KPALVTR(sn) != 0) return SOCKERR_SOCKOPT;
|
|
||||||
#endif
|
|
||||||
setSn_CR(sn,Sn_CR_SEND_KEEP);
|
|
||||||
while(getSn_CR(sn) != 0)
|
|
||||||
{
|
|
||||||
// M20131220
|
|
||||||
//if ((tmp = getSn_IR(sn)) & Sn_IR_TIMEOUT)
|
|
||||||
if (getSn_IR(sn) & Sn_IR_TIMEOUT)
|
|
||||||
{
|
|
||||||
setSn_IR(sn, Sn_IR_TIMEOUT);
|
|
||||||
return SOCKERR_TIMEOUT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#if _WIZCHIP_ > 5200
|
|
||||||
case SO_KEEPALIVEAUTO:
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
setSn_KPALVTR(sn,*(uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
default:
|
|
||||||
return SOCKERR_ARG;
|
|
||||||
}
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t WIZCHIP_EXPORT(getsockopt)(uint8_t sn, sockopt_type sotype, void* arg)
|
|
||||||
{
|
|
||||||
CHECK_SOCKNUM();
|
|
||||||
switch(sotype)
|
|
||||||
{
|
|
||||||
case SO_FLAG:
|
|
||||||
*(uint8_t*)arg = getSn_MR(sn) & 0xF0;
|
|
||||||
break;
|
|
||||||
case SO_TTL:
|
|
||||||
*(uint8_t*) arg = getSn_TTL(sn);
|
|
||||||
break;
|
|
||||||
case SO_TOS:
|
|
||||||
*(uint8_t*) arg = getSn_TOS(sn);
|
|
||||||
break;
|
|
||||||
case SO_MSS:
|
|
||||||
*(uint8_t*) arg = getSn_MSSR(sn);
|
|
||||||
case SO_DESTIP:
|
|
||||||
getSn_DIPR(sn, (uint8_t*)arg);
|
|
||||||
break;
|
|
||||||
case SO_DESTPORT:
|
|
||||||
*(uint16_t*) arg = getSn_DPORT(sn);
|
|
||||||
break;
|
|
||||||
#if _WIZCHIP_ > 5200
|
|
||||||
case SO_KEEPALIVEAUTO:
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
*(uint16_t*) arg = getSn_KPALVTR(sn);
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
case SO_SENDBUF:
|
|
||||||
*(uint16_t*) arg = getSn_TX_FSR(sn);
|
|
||||||
case SO_RECVBUF:
|
|
||||||
*(uint16_t*) arg = getSn_RX_RSR(sn);
|
|
||||||
case SO_STATUS:
|
|
||||||
*(uint8_t*) arg = getSn_SR(sn);
|
|
||||||
break;
|
|
||||||
case SO_REMAINSIZE:
|
|
||||||
if(getSn_MR(sn) == Sn_MR_TCP)
|
|
||||||
*(uint16_t*)arg = getSn_RX_RSR(sn);
|
|
||||||
else
|
|
||||||
*(uint16_t*)arg = sock_remained_size[sn];
|
|
||||||
break;
|
|
||||||
case SO_PACKINFO:
|
|
||||||
CHECK_SOCKMODE(Sn_MR_TCP);
|
|
||||||
*(uint8_t*)arg = sock_pack_info[sn];
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return SOCKERR_SOCKOPT;
|
|
||||||
}
|
|
||||||
return SOCK_OK;
|
|
||||||
}
|
|
@ -1,472 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file socket.h
|
|
||||||
//! \brief SOCKET APIs Header file.
|
|
||||||
//! \details SOCKET APIs like as berkeley socket api.
|
|
||||||
//! \version 1.0.2
|
|
||||||
//! \date 2013/10/21
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2014/05/01> V1.0.2. Refer to M20140501
|
|
||||||
//! 1. Modify the comment : SO_REMAINED -> PACK_REMAINED
|
|
||||||
//! 2. Add the comment as zero byte udp data reception in getsockopt().
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! \author MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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.
|
|
||||||
//
|
|
||||||
//*****************************************************************************
|
|
||||||
/**
|
|
||||||
* @defgroup WIZnet_socket_APIs 1. WIZnet socket APIs
|
|
||||||
* @brief WIZnet socket APIs are based on Berkeley socket APIs, thus it has much similar name and interface.
|
|
||||||
* But there is a little bit of difference.
|
|
||||||
* @details
|
|
||||||
* <b> Comparison between WIZnet and Berkeley SOCKET APIs </b>
|
|
||||||
* <table>
|
|
||||||
* <tr> <td><b>API</b></td> <td><b>WIZnet</b></td> <td><b>Berkeley</b></td> </tr>
|
|
||||||
* <tr> <td>socket()</td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>bind()</b></td> <td>X</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>listen()</b></td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>connect()</b></td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>accept()</b></td> <td>X</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>recv()</b></td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>send()</b></td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>recvfrom()</b></td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>sendto()</b></td> <td>O</td> <td>O</td> </tr>
|
|
||||||
* <tr> <td><b>closesocket()</b></td> <td>O<br>close() & disconnect()</td> <td>O</td> </tr>
|
|
||||||
* </table>
|
|
||||||
* There are @b bind() and @b accept() functions in @b Berkeley SOCKET API but,
|
|
||||||
* not in @b WIZnet SOCKET API. Because socket() of WIZnet is not only creating a SOCKET but also binding a local port number,
|
|
||||||
* and listen() of WIZnet is not only listening to connection request from client but also accepting the connection request. \n
|
|
||||||
* When you program "TCP SERVER" with Berkeley SOCKET API, you can use only one listen port.
|
|
||||||
* When the listen SOCKET accepts a connection request from a client, it keeps listening.
|
|
||||||
* After accepting the connection request, a new SOCKET is created and the new SOCKET is used in communication with the client. \n
|
|
||||||
* Following figure shows network flow diagram by Berkeley SOCKET API.
|
|
||||||
* @image html Berkeley_SOCKET.jpg "<Berkeley SOCKET API>"
|
|
||||||
* But, When you program "TCP SERVER" with WIZnet SOCKET API, you can use as many as 8 listen SOCKET with same port number. \n
|
|
||||||
* Because there's no accept() in WIZnet SOCKET APIs, when the listen SOCKET accepts a connection request from a client,
|
|
||||||
* it is changed in order to communicate with the client.
|
|
||||||
* And the changed SOCKET is not listening any more and is dedicated for communicating with the client. \n
|
|
||||||
* If there're many listen SOCKET with same listen port number and a client requests a connection,
|
|
||||||
* the SOCKET which has the smallest SOCKET number accepts the request and is changed as communication SOCKET. \n
|
|
||||||
* Following figure shows network flow diagram by WIZnet SOCKET API.
|
|
||||||
* @image html WIZnet_SOCKET.jpg "<WIZnet SOCKET API>"
|
|
||||||
*/
|
|
||||||
#ifndef _WIZCHIP_SOCKET_H_
|
|
||||||
#define _WIZCHIP_SOCKET_H_
|
|
||||||
|
|
||||||
// use this macro for exported names to avoid name clashes
|
|
||||||
#define WIZCHIP_EXPORT(name) wizchip_ ## name
|
|
||||||
|
|
||||||
#include "wizchip_conf.h"
|
|
||||||
|
|
||||||
#define SOCKET uint8_t ///< SOCKET type define for legacy driver
|
|
||||||
|
|
||||||
#define SOCK_OK 1 ///< Result is OK about socket process.
|
|
||||||
#define SOCK_BUSY 0 ///< Socket is busy on processing the operation. Valid only Non-block IO Mode.
|
|
||||||
#define SOCK_FATAL -1000 ///< Result is fatal error about socket process.
|
|
||||||
|
|
||||||
#define SOCK_ERROR 0
|
|
||||||
#define SOCKERR_SOCKNUM (SOCK_ERROR - 1) ///< Invalid socket number
|
|
||||||
#define SOCKERR_SOCKOPT (SOCK_ERROR - 2) ///< Invalid socket option
|
|
||||||
#define SOCKERR_SOCKINIT (SOCK_ERROR - 3) ///< Socket is not initialized
|
|
||||||
#define SOCKERR_SOCKCLOSED (SOCK_ERROR - 4) ///< Socket unexpectedly closed.
|
|
||||||
#define SOCKERR_SOCKMODE (SOCK_ERROR - 5) ///< Invalid socket mode for socket operation.
|
|
||||||
#define SOCKERR_SOCKFLAG (SOCK_ERROR - 6) ///< Invalid socket flag
|
|
||||||
#define SOCKERR_SOCKSTATUS (SOCK_ERROR - 7) ///< Invalid socket status for socket operation.
|
|
||||||
#define SOCKERR_ARG (SOCK_ERROR - 10) ///< Invalid argument.
|
|
||||||
#define SOCKERR_PORTZERO (SOCK_ERROR - 11) ///< Port number is zero
|
|
||||||
#define SOCKERR_IPINVALID (SOCK_ERROR - 12) ///< Invalid IP address
|
|
||||||
#define SOCKERR_TIMEOUT (SOCK_ERROR - 13) ///< Timeout occurred
|
|
||||||
#define SOCKERR_DATALEN (SOCK_ERROR - 14) ///< Data length is zero or greater than buffer max size.
|
|
||||||
#define SOCKERR_BUFFER (SOCK_ERROR - 15) ///< Socket buffer is not enough for data communication.
|
|
||||||
|
|
||||||
#define SOCKFATAL_PACKLEN (SOCK_FATAL - 1) ///< Invalid packet length. Fatal Error.
|
|
||||||
|
|
||||||
/*
|
|
||||||
* SOCKET FLAG
|
|
||||||
*/
|
|
||||||
#define SF_ETHER_OWN (Sn_MR_MFEN) ///< In \ref Sn_MR_MACRAW, Receive only the packet as broadcast, multicast and own packet
|
|
||||||
#define SF_IGMP_VER2 (Sn_MR_MC) ///< In \ref Sn_MR_UDP with \ref SF_MULTI_ENABLE, Select IGMP version 2.
|
|
||||||
#define SF_TCP_NODELAY (Sn_MR_ND) ///< In \ref Sn_MR_TCP, Use to nodelayed ack.
|
|
||||||
#define SF_MULTI_ENABLE (Sn_MR_MULTI) ///< In \ref Sn_MR_UDP, Enable multicast mode.
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
#define SF_BROAD_BLOCK (Sn_MR_BCASTB) ///< In \ref Sn_MR_UDP or \ref Sn_MR_MACRAW, Block broadcast packet. Valid only in W5500
|
|
||||||
#define SF_MULTI_BLOCK (Sn_MR_MMB) ///< In \ref Sn_MR_MACRAW, Block multicast packet. Valid only in W5500
|
|
||||||
#define SF_IPv6_BLOCK (Sn_MR_MIP6B) ///< In \ref Sn_MR_MACRAW, Block IPv6 packet. Valid only in W5500
|
|
||||||
#define SF_UNI_BLOCK (Sn_MR_UCASTB) ///< In \ref Sn_MR_UDP with \ref SF_MULTI_ENABLE. Valid only in W5500
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define SF_IO_NONBLOCK 0x01 ///< Socket nonblock io mode. It used parameter in \ref socket().
|
|
||||||
|
|
||||||
/*
|
|
||||||
* UDP & MACRAW Packet Infomation
|
|
||||||
*/
|
|
||||||
#define PACK_FIRST 0x80 ///< In Non-TCP packet, It indicates to start receiving a packet.
|
|
||||||
#define PACK_REMAINED 0x01 ///< In Non-TCP packet, It indicates to remain a packet to be received.
|
|
||||||
#define PACK_COMPLETED 0x00 ///< In Non-TCP packet, It indicates to complete to receive a packet.
|
|
||||||
|
|
||||||
// resets all global state associated with the socket interface
|
|
||||||
void WIZCHIP_EXPORT(socket_reset)(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Open a socket.
|
|
||||||
* @details Initializes the socket with 'sn' passed as parameter and open.
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @param protocol Protocol type to operate such as TCP, UDP and MACRAW.
|
|
||||||
* @param port Port number to be bined.
|
|
||||||
* @param flag Socket flags as \ref SF_ETHER_OWN, \ref SF_IGMP_VER2, \ref SF_TCP_NODELAY, \ref SF_MULTI_ENABLE, \ref SF_IO_NONBLOCK and so on.\n
|
|
||||||
* Valid flags only in W5500 : @ref SF_BROAD_BLOCK, @ref SF_MULTI_BLOCK, @ref SF_IPv6_BLOCK, and @ref SF_UNI_BLOCK.
|
|
||||||
* @sa Sn_MR
|
|
||||||
*
|
|
||||||
* @return @b Success : The socket number @b 'sn' passed as parameter\n
|
|
||||||
* @b Fail :\n @ref SOCKERR_SOCKNUM - Invalid socket number\n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Not support socket mode as TCP, UDP, and so on. \n
|
|
||||||
* @ref SOCKERR_SOCKFLAG - Invaild socket flag.
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(socket)(uint8_t sn, uint8_t protocol, uint16_t port, uint8_t flag);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Close a socket.
|
|
||||||
* @details It closes the socket with @b'sn' passed as parameter.
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
*
|
|
||||||
* @return @b Success : @ref SOCK_OK \n
|
|
||||||
* @b Fail : @ref SOCKERR_SOCKNUM - Invalid socket number
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(close)(uint8_t sn);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Listen to a connection request from a client.
|
|
||||||
* @details It is listening to a connection request from a client.
|
|
||||||
* If connection request is accepted successfully, the connection is established. Socket sn is used in passive(server) mode.
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @return @b Success : @ref SOCK_OK \n
|
|
||||||
* @b Fail :\n @ref SOCKERR_SOCKINIT - Socket is not initialized \n
|
|
||||||
* @ref SOCKERR_SOCKCLOSED - Socket closed unexpectedly.
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(listen)(uint8_t sn);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Try to connect a server.
|
|
||||||
* @details It requests connection to the server with destination IP address and port number passed as parameter.\n
|
|
||||||
* @note It is valid only in TCP client mode.
|
|
||||||
* In block io mode, it does not return until connection is completed.
|
|
||||||
* In Non-block io mode, it return @ref SOCK_BUSY immediately.
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @param addr Pointer variable of destination IP address. It should be allocated 4 bytes.
|
|
||||||
* @param port Destination port number.
|
|
||||||
*
|
|
||||||
* @return @b Success : @ref SOCK_OK \n
|
|
||||||
* @b Fail :\n @ref SOCKERR_SOCKNUM - Invalid socket number\n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Invalid socket mode\n
|
|
||||||
* @ref SOCKERR_SOCKINIT - Socket is not initialized\n
|
|
||||||
* @ref SOCKERR_IPINVALID - Wrong server IP address\n
|
|
||||||
* @ref SOCKERR_PORTZERO - Server port zero\n
|
|
||||||
* @ref SOCKERR_TIMEOUT - Timeout occurred during request connection\n
|
|
||||||
* @ref SOCK_BUSY - In non-block io mode, it returned immediately\n
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(connect)(uint8_t sn, uint8_t * addr, uint16_t port);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Try to disconnect a connection socket.
|
|
||||||
* @details It sends request message to disconnect the TCP socket 'sn' passed as parameter to the server or client.
|
|
||||||
* @note It is valid only in TCP server or client mode. \n
|
|
||||||
* In block io mode, it does not return until disconnection is completed. \n
|
|
||||||
* In Non-block io mode, it return @ref SOCK_BUSY immediately. \n
|
|
||||||
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @return @b Success : @ref SOCK_OK \n
|
|
||||||
* @b Fail :\n @ref SOCKERR_SOCKNUM - Invalid socket number \n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Invalid operation in the socket \n
|
|
||||||
* @ref SOCKERR_TIMEOUT - Timeout occurred \n
|
|
||||||
* @ref SOCK_BUSY - Socket is busy.
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(disconnect)(uint8_t sn);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Send data to the connected peer in TCP socket.
|
|
||||||
* @details It is used to send outgoing data to the connected socket.
|
|
||||||
* @note It is valid only in TCP server or client mode. It can't send data greater than socket buffer size. \n
|
|
||||||
* In block io mode, It doesn't return until data send is completed - socket buffer size is greater than data. \n
|
|
||||||
* In non-block io mode, It return @ref SOCK_BUSY immediately when socket buffer is not enough. \n
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @param buf Pointer buffer containing data to be sent.
|
|
||||||
* @param len The byte length of data in buf.
|
|
||||||
* @return @b Success : The sent data size \n
|
|
||||||
* @b Fail : \n @ref SOCKERR_SOCKSTATUS - Invalid socket status for socket operation \n
|
|
||||||
* @ref SOCKERR_TIMEOUT - Timeout occurred \n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Invalid operation in the socket \n
|
|
||||||
* @ref SOCKERR_SOCKNUM - Invalid socket number \n
|
|
||||||
* @ref SOCKERR_DATALEN - zero data length \n
|
|
||||||
* @ref SOCK_BUSY - Socket is busy.
|
|
||||||
*/
|
|
||||||
int32_t WIZCHIP_EXPORT(send)(uint8_t sn, uint8_t * buf, uint16_t len);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Receive data from the connected peer.
|
|
||||||
* @details It is used to read incoming data from the connected socket.\n
|
|
||||||
* It waits for data as much as the application wants to receive.
|
|
||||||
* @note It is valid only in TCP server or client mode. It can't receive data greater than socket buffer size. \n
|
|
||||||
* In block io mode, it doesn't return until data reception is completed - data is filled as <I>len</I> in socket buffer. \n
|
|
||||||
* In non-block io mode, it return @ref SOCK_BUSY immediately when <I>len</I> is greater than data size in socket buffer. \n
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @param buf Pointer buffer to read incoming data.
|
|
||||||
* @param len The max data length of data in buf.
|
|
||||||
* @return @b Success : The real received data size \n
|
|
||||||
* @b Fail :\n
|
|
||||||
* @ref SOCKERR_SOCKSTATUS - Invalid socket status for socket operation \n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Invalid operation in the socket \n
|
|
||||||
* @ref SOCKERR_SOCKNUM - Invalid socket number \n
|
|
||||||
* @ref SOCKERR_DATALEN - zero data length \n
|
|
||||||
* @ref SOCK_BUSY - Socket is busy.
|
|
||||||
*/
|
|
||||||
int32_t WIZCHIP_EXPORT(recv)(uint8_t sn, uint8_t * buf, uint16_t len);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Sends datagram to the peer with destination IP address and port number passed as parameter.
|
|
||||||
* @details It sends datagram of UDP or MACRAW to the peer with destination IP address and port number passed as parameter.\n
|
|
||||||
* Even if the connectionless socket has been previously connected to a specific address,
|
|
||||||
* the address and port number parameters override the destination address for that particular datagram only.
|
|
||||||
* @note In block io mode, It doesn't return until data send is completed - socket buffer size is greater than <I>len</I>.
|
|
||||||
* In non-block io mode, It return @ref SOCK_BUSY immediately when socket buffer is not enough.
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @param buf Pointer buffer to send outgoing data.
|
|
||||||
* @param len The byte length of data in buf.
|
|
||||||
* @param addr Pointer variable of destination IP address. It should be allocated 4 bytes.
|
|
||||||
* @param port Destination port number.
|
|
||||||
*
|
|
||||||
* @return @b Success : The sent data size \n
|
|
||||||
* @b Fail :\n @ref SOCKERR_SOCKNUM - Invalid socket number \n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Invalid operation in the socket \n
|
|
||||||
* @ref SOCKERR_SOCKSTATUS - Invalid socket status for socket operation \n
|
|
||||||
* @ref SOCKERR_DATALEN - zero data length \n
|
|
||||||
* @ref SOCKERR_IPINVALID - Wrong server IP address\n
|
|
||||||
* @ref SOCKERR_PORTZERO - Server port zero\n
|
|
||||||
* @ref SOCKERR_SOCKCLOSED - Socket unexpectedly closed \n
|
|
||||||
* @ref SOCKERR_TIMEOUT - Timeout occurred \n
|
|
||||||
* @ref SOCK_BUSY - Socket is busy.
|
|
||||||
*/
|
|
||||||
int32_t WIZCHIP_EXPORT(sendto)(uint8_t sn, uint8_t * buf, uint16_t len, uint8_t * addr, uint16_t port);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Receive datagram of UDP or MACRAW
|
|
||||||
* @details This function is an application I/F function which is used to receive the data in other then TCP mode. \n
|
|
||||||
* This function is used to receive UDP and MAC_RAW mode, and handle the header as well.
|
|
||||||
* This function can divide to received the packet data.
|
|
||||||
* On the MACRAW SOCKET, the addr and port parameters are ignored.
|
|
||||||
* @note In block io mode, it doesn't return until data reception is completed - data is filled as <I>len</I> in socket buffer
|
|
||||||
* In non-block io mode, it return @ref SOCK_BUSY immediately when <I>len</I> is greater than data size in socket buffer.
|
|
||||||
*
|
|
||||||
* @param sn Socket number. It should be <b>0 ~ @ref \_WIZCHIP_SOCK_NUM_</b>.
|
|
||||||
* @param buf Pointer buffer to read incoming data.
|
|
||||||
* @param len The max data length of data in buf.
|
|
||||||
* When the received packet size <= len, receives data as packet sized.
|
|
||||||
* When others, receives data as len.
|
|
||||||
* @param addr Pointer variable of destination IP address. It should be allocated 4 bytes.
|
|
||||||
* It is valid only when the first call recvfrom for receiving the packet.
|
|
||||||
* When it is valid, @ref packinfo[7] should be set as '1' after call @ref getsockopt(sn, SO_PACKINFO, &packinfo).
|
|
||||||
* @param port Pointer variable of destination port number.
|
|
||||||
* It is valid only when the first call recvform for receiving the packet.
|
|
||||||
* When it is valid, @ref packinfo[7] should be set as '1' after call @ref getsockopt(sn, SO_PACKINFO, &packinfo).
|
|
||||||
*
|
|
||||||
* @return @b Success : This function return real received data size for success.\n
|
|
||||||
* @b Fail : @ref SOCKERR_DATALEN - zero data length \n
|
|
||||||
* @ref SOCKERR_SOCKMODE - Invalid operation in the socket \n
|
|
||||||
* @ref SOCKERR_SOCKNUM - Invalid socket number \n
|
|
||||||
* @ref SOCKBUSY - Socket is busy.
|
|
||||||
*/
|
|
||||||
int32_t WIZCHIP_EXPORT(recvfrom)(uint8_t sn, uint8_t * buf, uint16_t len, uint8_t * addr, uint16_t *port);
|
|
||||||
|
|
||||||
|
|
||||||
/////////////////////////////
|
|
||||||
// SOCKET CONTROL & OPTION //
|
|
||||||
/////////////////////////////
|
|
||||||
#define SOCK_IO_BLOCK 0 ///< Socket Block IO Mode in @ref setsockopt().
|
|
||||||
#define SOCK_IO_NONBLOCK 1 ///< Socket Non-block IO Mode in @ref setsockopt().
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @defgroup DATA_TYPE DATA TYPE
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* @brief The kind of Socket Interrupt.
|
|
||||||
* @sa Sn_IR, Sn_IMR, setSn_IR(), getSn_IR(), setSn_IMR(), getSn_IMR()
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
SIK_CONNECTED = (1 << 0), ///< connected
|
|
||||||
SIK_DISCONNECTED = (1 << 1), ///< disconnected
|
|
||||||
SIK_RECEIVED = (1 << 2), ///< data received
|
|
||||||
SIK_TIMEOUT = (1 << 3), ///< timeout occurred
|
|
||||||
SIK_SENT = (1 << 4), ///< send ok
|
|
||||||
SIK_ALL = 0x1F, ///< all interrupt
|
|
||||||
}sockint_kind;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* @brief The type of @ref ctlsocket().
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
CS_SET_IOMODE, ///< set socket IO mode with @ref SOCK_IO_BLOCK or @ref SOCK_IO_NONBLOCK
|
|
||||||
CS_GET_IOMODE, ///< get socket IO mode
|
|
||||||
CS_GET_MAXTXBUF, ///< get the size of socket buffer allocated in TX memory
|
|
||||||
CS_GET_MAXRXBUF, ///< get the size of socket buffer allocated in RX memory
|
|
||||||
CS_CLR_INTERRUPT, ///< clear the interrupt of socket with @ref sockint_kind
|
|
||||||
CS_GET_INTERRUPT, ///< get the socket interrupt. refer to @ref sockint_kind
|
|
||||||
CS_SET_INTMASK, ///< set the interrupt mask of socket with @ref sockint_kind
|
|
||||||
CS_GET_INTMASK ///< get the masked interrupt of socket. refer to @ref sockint_kind
|
|
||||||
}ctlsock_type;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* @brief The type of socket option in @ref setsockopt() or @ref getsockopt()
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
SO_FLAG, ///< Valid only in getsockopt(), For set flag of socket refer to <I>flag</I> in @ref socket().
|
|
||||||
SO_TTL, ///< Set/Get TTL. @ref Sn_TTL ( @ref setSn_TTL(), @ref getSn_TTL() )
|
|
||||||
SO_TOS, ///< Set/Get TOS. @ref Sn_TOS ( @ref setSn_TOS(), @ref getSn_TOS() )
|
|
||||||
SO_MSS, ///< Set/Get MSS. @ref Sn_MSSR ( @ref setSn_MSSR(), @ref getSn_MSSR() )
|
|
||||||
SO_DESTIP, ///< Set/Get the destination IP address. @ref Sn_DIPR ( @ref setSn_DIPR(), @ref getSn_DIPR() )
|
|
||||||
SO_DESTPORT, ///< Set/Get the destination Port number. @ref Sn_DPORT ( @ref setSn_DPORT(), @ref getSn_DPORT() )
|
|
||||||
#if _WIZCHIP_ != 5100
|
|
||||||
SO_KEEPALIVESEND, ///< Valid only in setsockopt. Manually send keep-alive packet in TCP mode
|
|
||||||
#if _WIZCHIP_ > 5200
|
|
||||||
SO_KEEPALIVEAUTO, ///< Set/Get keep-alive auto transmission timer in TCP mode
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
SO_SENDBUF, ///< Valid only in getsockopt. Get the free data size of Socekt TX buffer. @ref Sn_TX_FSR, @ref getSn_TX_FSR()
|
|
||||||
SO_RECVBUF, ///< Valid only in getsockopt. Get the received data size in socket RX buffer. @ref Sn_RX_RSR, @ref getSn_RX_RSR()
|
|
||||||
SO_STATUS, ///< Valid only in getsockopt. Get the socket status. @ref Sn_SR, @ref getSn_SR()
|
|
||||||
SO_REMAINSIZE, ///< Valid only in getsockopt. Get the remained packet size in other then TCP mode.
|
|
||||||
SO_PACKINFO ///< Valid only in getsockopt. Get the packet information as @ref PACK_FIRST, @ref PACK_REMAINED, and @ref PACK_COMPLETED in other then TCP mode.
|
|
||||||
}sockopt_type;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief Control socket.
|
|
||||||
* @details Control IO mode, Interrupt & Mask of socket and get the socket buffer information.
|
|
||||||
* Refer to @ref ctlsock_type.
|
|
||||||
* @param sn socket number
|
|
||||||
* @param cstype type of control socket. refer to @ref ctlsock_type.
|
|
||||||
* @param arg Data type and value is determined according to @ref ctlsock_type. \n
|
|
||||||
* <table>
|
|
||||||
* <tr> <td> @b cstype </td> <td> @b data type</td><td>@b value</td></tr>
|
|
||||||
* <tr> <td> @ref CS_SET_IOMODE \n @ref CS_GET_IOMODE </td> <td> uint8_t </td><td>@ref SOCK_IO_BLOCK @ref SOCK_IO_NONBLOCK</td></tr>
|
|
||||||
* <tr> <td> @ref CS_GET_MAXTXBUF \n @ref CS_GET_MAXRXBUF </td> <td> uint16_t </td><td> 0 ~ 16K </td></tr>
|
|
||||||
* <tr> <td> @ref CS_CLR_INTERRUPT \n @ref CS_GET_INTERRUPT \n @ref CS_SET_INTMASK \n @ref CS_GET_INTMASK </td> <td> @ref sockint_kind </td><td> @ref SIK_CONNECTED, etc. </td></tr>
|
|
||||||
* </table>
|
|
||||||
* @return @b Success @ref SOCK_OK \n
|
|
||||||
* @b fail @ref SOCKERR_ARG - Invalid argument\n
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(ctlsocket)(uint8_t sn, ctlsock_type cstype, void* arg);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief set socket options
|
|
||||||
* @details Set socket option like as TTL, MSS, TOS, and so on. Refer to @ref sockopt_type.
|
|
||||||
*
|
|
||||||
* @param sn socket number
|
|
||||||
* @param sotype socket option type. refer to @ref sockopt_type
|
|
||||||
* @param arg Data type and value is determined according to <I>sotype</I>. \n
|
|
||||||
* <table>
|
|
||||||
* <tr> <td> @b sotype </td> <td> @b data type</td><td>@b value</td></tr>
|
|
||||||
* <tr> <td> @ref SO_TTL </td> <td> uint8_t </td><td> 0 ~ 255 </td> </tr>
|
|
||||||
* <tr> <td> @ref SO_TOS </td> <td> uint8_t </td><td> 0 ~ 255 </td> </tr>
|
|
||||||
* <tr> <td> @ref SO_MSS </td> <td> uint16_t </td><td> 0 ~ 65535 </td> </tr>
|
|
||||||
* <tr> <td> @ref SO_DESTIP </td> <td> uint8_t[4] </td><td> </td></tr>
|
|
||||||
* <tr> <td> @ref SO_DESTPORT </td> <td> uint16_t </td><td> 0 ~ 65535 </td></tr>
|
|
||||||
* <tr> <td> @ref SO_KEEPALIVESEND </td> <td> null </td><td> null </td></tr>
|
|
||||||
* <tr> <td> @ref SO_KEEPALIVEAUTO </td> <td> uint8_t </td><td> 0 ~ 255 </td></tr>
|
|
||||||
* </table>
|
|
||||||
* @return
|
|
||||||
* - @b Success : @ref SOCK_OK \n
|
|
||||||
* - @b Fail
|
|
||||||
* - @ref SOCKERR_SOCKNUM - Invalid Socket number \n
|
|
||||||
* - @ref SOCKERR_SOCKMODE - Invalid socket mode \n
|
|
||||||
* - @ref SOCKERR_SOCKOPT - Invalid socket option or its value \n
|
|
||||||
* - @ref SOCKERR_TIMEOUT - Timeout occurred when sending keep-alive packet \n
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(setsockopt)(uint8_t sn, sockopt_type sotype, void* arg);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup WIZnet_socket_APIs
|
|
||||||
* @brief get socket options
|
|
||||||
* @details Get socket option like as FLAG, TTL, MSS, and so on. Refer to @ref sockopt_type
|
|
||||||
* @param sn socket number
|
|
||||||
* @param sotype socket option type. refer to @ref sockopt_type
|
|
||||||
* @param arg Data type and value is determined according to <I>sotype</I>. \n
|
|
||||||
* <table>
|
|
||||||
* <tr> <td> @b sotype </td> <td>@b data type</td><td>@b value</td></tr>
|
|
||||||
* <tr> <td> @ref SO_FLAG </td> <td> uint8_t </td><td> @ref SF_ETHER_OWN, etc... </td> </tr>
|
|
||||||
* <tr> <td> @ref SO_TOS </td> <td> uint8_t </td><td> 0 ~ 255 </td> </tr>
|
|
||||||
* <tr> <td> @ref SO_MSS </td> <td> uint16_t </td><td> 0 ~ 65535 </td> </tr>
|
|
||||||
* <tr> <td> @ref SO_DESTIP </td> <td> uint8_t[4] </td><td> </td></tr>
|
|
||||||
* <tr> <td> @ref SO_DESTPORT </td> <td> uint16_t </td><td> </td></tr>
|
|
||||||
* <tr> <td> @ref SO_KEEPALIVEAUTO </td> <td> uint8_t </td><td> 0 ~ 255 </td></tr>
|
|
||||||
* <tr> <td> @ref SO_SENDBUF </td> <td> uint16_t </td><td> 0 ~ 65535 </td></tr>
|
|
||||||
* <tr> <td> @ref SO_RECVBUF </td> <td> uint16_t </td><td> 0 ~ 65535 </td></tr>
|
|
||||||
* <tr> <td> @ref SO_STATUS </td> <td> uint8_t </td><td> @ref SOCK_ESTABLISHED, etc.. </td></tr>
|
|
||||||
* <tr> <td> @ref SO_REMAINSIZE </td> <td> uint16_t </td><td> 0~ 65535 </td></tr>
|
|
||||||
* <tr> <td> @ref SO_PACKINFO </td> <td> uint8_t </td><td> @ref PACK_FIRST, etc... </td></tr>
|
|
||||||
* </table>
|
|
||||||
* @return
|
|
||||||
* - @b Success : @ref SOCK_OK \n
|
|
||||||
* - @b Fail
|
|
||||||
* - @ref SOCKERR_SOCKNUM - Invalid Socket number \n
|
|
||||||
* - @ref SOCKERR_SOCKOPT - Invalid socket option or its value \n
|
|
||||||
* - @ref SOCKERR_SOCKMODE - Invalid socket mode \n
|
|
||||||
* @note
|
|
||||||
* The option as PACK_REMAINED and SO_PACKINFO is valid only in NON-TCP mode and after call @ref recvfrom(). \n
|
|
||||||
* When SO_PACKINFO value is PACK_FIRST and the return value of recvfrom() is zero,
|
|
||||||
* This means the zero byte UDP data(UDP Header only) received.
|
|
||||||
*/
|
|
||||||
int8_t WIZCHIP_EXPORT(getsockopt)(uint8_t sn, sockopt_type sotype, void* arg);
|
|
||||||
|
|
||||||
#endif // _WIZCHIP_SOCKET_H_
|
|
@ -1,206 +0,0 @@
|
|||||||
// dpgeorge: this file taken from w5500/w5500.c and adapted to W5200
|
|
||||||
|
|
||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file w5500.c
|
|
||||||
//! \brief W5500 HAL Interface.
|
|
||||||
//! \version 1.0.1
|
|
||||||
//! \date 2013/10/21
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2014/05/01> V1.0.2
|
|
||||||
//! 1. Implicit type casting -> Explicit type casting. Refer to M20140501
|
|
||||||
//! Fixed the problem on porting into under 32bit MCU
|
|
||||||
//! Issued by Mathias ClauBen, wizwiki forum ID Think01 and bobh
|
|
||||||
//! Thank for your interesting and serious advices.
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! <2013/12/20> V1.0.1
|
|
||||||
//! 1. Remove warning
|
|
||||||
//! 2. WIZCHIP_READ_BUF WIZCHIP_WRITE_BUF in case _WIZCHIP_IO_MODE_SPI_FDM_
|
|
||||||
//! for loop optimized(removed). refer to M20131220
|
|
||||||
//! \author MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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 "w5200.h"
|
|
||||||
|
|
||||||
#define SMASK (0x7ff) /* tx buffer mask */
|
|
||||||
#define RMASK (0x7ff) /* rx buffer mask */
|
|
||||||
#define SSIZE (2048) /* max tx buffer size */
|
|
||||||
#define RSIZE (2048) /* max rx buffer size */
|
|
||||||
|
|
||||||
#define TXBUF_BASE (0x8000)
|
|
||||||
#define RXBUF_BASE (0xc000)
|
|
||||||
#define SBASE(sn) (TXBUF_BASE + SSIZE * (sn)) /* tx buffer base for socket sn */
|
|
||||||
#define RBASE(sn) (RXBUF_BASE + RSIZE * (sn)) /* rx buffer base for socket sn */
|
|
||||||
|
|
||||||
uint8_t WIZCHIP_READ(uint32_t AddrSel) {
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
uint8_t spi_data[4] = {
|
|
||||||
AddrSel >> 8,
|
|
||||||
AddrSel,
|
|
||||||
0x00,
|
|
||||||
0x01,
|
|
||||||
};
|
|
||||||
WIZCHIP.IF.SPI._write_bytes(spi_data, 4);
|
|
||||||
uint8_t ret;
|
|
||||||
WIZCHIP.IF.SPI._read_bytes(&ret, 1);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void WIZCHIP_WRITE(uint32_t AddrSel, uint8_t wb) {
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
uint8_t spi_data[5] = {
|
|
||||||
AddrSel >> 8,
|
|
||||||
AddrSel,
|
|
||||||
0x80,
|
|
||||||
0x01,
|
|
||||||
wb,
|
|
||||||
};
|
|
||||||
WIZCHIP.IF.SPI._write_bytes(spi_data, 5);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WIZCHIP_READ_BUF(uint32_t AddrSel, uint8_t* pBuf, uint16_t len) {
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
uint8_t spi_data[4] = {
|
|
||||||
AddrSel >> 8,
|
|
||||||
AddrSel,
|
|
||||||
0x00 | ((len >> 8) & 0x7f),
|
|
||||||
len & 0xff,
|
|
||||||
};
|
|
||||||
WIZCHIP.IF.SPI._write_bytes(spi_data, 4);
|
|
||||||
WIZCHIP.IF.SPI._read_bytes(pBuf, len);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WIZCHIP_WRITE_BUF(uint32_t AddrSel, uint8_t* pBuf, uint16_t len) {
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
uint8_t spi_data[4] = {
|
|
||||||
AddrSel >> 8,
|
|
||||||
AddrSel,
|
|
||||||
0x80 | ((len >> 8) & 0x7f),
|
|
||||||
len & 0xff,
|
|
||||||
};
|
|
||||||
WIZCHIP.IF.SPI._write_bytes(spi_data, 4);
|
|
||||||
WIZCHIP.IF.SPI._write_bytes(pBuf, len);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
}
|
|
||||||
|
|
||||||
uint16_t getSn_TX_FSR(uint8_t sn) {
|
|
||||||
uint16_t val = 0, val1 = 0;
|
|
||||||
do {
|
|
||||||
val1 = (WIZCHIP_READ(Sn_TX_FSR(sn)) << 8) | WIZCHIP_READ(Sn_TX_FSR(sn) + 1);
|
|
||||||
if (val1 != 0) {
|
|
||||||
val = (WIZCHIP_READ(Sn_TX_FSR(sn)) << 8) | WIZCHIP_READ(Sn_TX_FSR(sn) + 1);
|
|
||||||
}
|
|
||||||
} while (val != val1);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint16_t getSn_RX_RSR(uint8_t sn) {
|
|
||||||
uint16_t val = 0, val1 = 0;
|
|
||||||
do {
|
|
||||||
val1 = (WIZCHIP_READ(Sn_RX_RSR(sn)) << 8) | WIZCHIP_READ(Sn_RX_RSR(sn) + 1);
|
|
||||||
if (val1 != 0) {
|
|
||||||
val = (WIZCHIP_READ(Sn_RX_RSR(sn)) << 8) | WIZCHIP_READ(Sn_RX_RSR(sn) + 1);
|
|
||||||
}
|
|
||||||
} while (val != val1);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
void wiz_send_data(uint8_t sn, uint8_t *wizdata, uint16_t len) {
|
|
||||||
if (len == 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint16_t ptr = getSn_TX_WR(sn);
|
|
||||||
uint16_t offset = ptr & SMASK;
|
|
||||||
uint32_t addr = offset + SBASE(sn);
|
|
||||||
|
|
||||||
if (offset + len > SSIZE) {
|
|
||||||
// implement wrap-around circular buffer
|
|
||||||
uint16_t size = SSIZE - offset;
|
|
||||||
WIZCHIP_WRITE_BUF(addr, wizdata, size);
|
|
||||||
WIZCHIP_WRITE_BUF(SBASE(sn), wizdata + size, len - size);
|
|
||||||
} else {
|
|
||||||
WIZCHIP_WRITE_BUF(addr, wizdata, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
ptr += len;
|
|
||||||
setSn_TX_WR(sn, ptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
void wiz_recv_data(uint8_t sn, uint8_t *wizdata, uint16_t len) {
|
|
||||||
if (len == 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint16_t ptr = getSn_RX_RD(sn);
|
|
||||||
uint16_t offset = ptr & RMASK;
|
|
||||||
uint16_t addr = RBASE(sn) + offset;
|
|
||||||
|
|
||||||
if (offset + len > RSIZE) {
|
|
||||||
// implement wrap-around circular buffer
|
|
||||||
uint16_t size = RSIZE - offset;
|
|
||||||
WIZCHIP_READ_BUF(addr, wizdata, size);
|
|
||||||
WIZCHIP_READ_BUF(RBASE(sn), wizdata + size, len - size);
|
|
||||||
} else {
|
|
||||||
WIZCHIP_READ_BUF(addr, wizdata, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
ptr += len;
|
|
||||||
setSn_RX_RD(sn, ptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
void wiz_recv_ignore(uint8_t sn, uint16_t len) {
|
|
||||||
uint16_t ptr = getSn_RX_RD(sn);
|
|
||||||
ptr += len;
|
|
||||||
setSn_RX_RD(sn, ptr);
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
@ -1,246 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file w5500.c
|
|
||||||
//! \brief W5500 HAL Interface.
|
|
||||||
//! \version 1.0.1
|
|
||||||
//! \date 2013/10/21
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2014/05/01> V1.0.2
|
|
||||||
//! 1. Implicit type casting -> Explicit type casting. Refer to M20140501
|
|
||||||
//! Fixed the problem on porting into under 32bit MCU
|
|
||||||
//! Issued by Mathias ClauBen, wizwiki forum ID Think01 and bobh
|
|
||||||
//! Thank for your interesting and serious advices.
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! <2013/12/20> V1.0.1
|
|
||||||
//! 1. Remove warning
|
|
||||||
//! 2. WIZCHIP_READ_BUF WIZCHIP_WRITE_BUF in case _WIZCHIP_IO_MODE_SPI_FDM_
|
|
||||||
//! for loop optimized(removed). refer to M20131220
|
|
||||||
//! \author MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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 <stdio.h>
|
|
||||||
#include "w5500.h"
|
|
||||||
|
|
||||||
#define _W5500_SPI_VDM_OP_ 0x00
|
|
||||||
#define _W5500_SPI_FDM_OP_LEN1_ 0x01
|
|
||||||
#define _W5500_SPI_FDM_OP_LEN2_ 0x02
|
|
||||||
#define _W5500_SPI_FDM_OP_LEN4_ 0x03
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#define LPC_SSP0 (0)
|
|
||||||
|
|
||||||
static void Chip_SSP_ReadFrames_Blocking(int dummy, uint8_t *buf, uint32_t len) {
|
|
||||||
WIZCHIP.IF.SPI._read_bytes(buf, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void Chip_SSP_WriteFrames_Blocking(int dummy, const uint8_t *buf, uint32_t len) {
|
|
||||||
WIZCHIP.IF.SPI._write_bytes(buf, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t WIZCHIP_READ(uint32_t AddrSel)
|
|
||||||
{
|
|
||||||
uint8_t ret;
|
|
||||||
uint8_t spi_data[3];
|
|
||||||
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
AddrSel |= (_W5500_SPI_READ_ | _W5500_SPI_VDM_OP_);
|
|
||||||
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x00FF0000) >> 16);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x0000FF00) >> 8);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x000000FF) >> 0);
|
|
||||||
//ret = WIZCHIP.IF.SPI._read_byte();
|
|
||||||
spi_data[0] = (AddrSel & 0x00FF0000) >> 16;
|
|
||||||
spi_data[1] = (AddrSel & 0x0000FF00) >> 8;
|
|
||||||
spi_data[2] = (AddrSel & 0x000000FF) >> 0;
|
|
||||||
Chip_SSP_WriteFrames_Blocking(LPC_SSP0, spi_data, 3);
|
|
||||||
Chip_SSP_ReadFrames_Blocking(LPC_SSP0, &ret, 1);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void WIZCHIP_WRITE(uint32_t AddrSel, uint8_t wb )
|
|
||||||
{
|
|
||||||
uint8_t spi_data[4];
|
|
||||||
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
AddrSel |= (_W5500_SPI_WRITE_ | _W5500_SPI_VDM_OP_);
|
|
||||||
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x00FF0000) >> 16);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x0000FF00) >> 8);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x000000FF) >> 0);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte(wb);
|
|
||||||
spi_data[0] = (AddrSel & 0x00FF0000) >> 16;
|
|
||||||
spi_data[1] = (AddrSel & 0x0000FF00) >> 8;
|
|
||||||
spi_data[2] = (AddrSel & 0x000000FF) >> 0;
|
|
||||||
spi_data[3] = wb;
|
|
||||||
Chip_SSP_WriteFrames_Blocking(LPC_SSP0, spi_data, 4);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WIZCHIP_READ_BUF (uint32_t AddrSel, uint8_t* pBuf, uint16_t len)
|
|
||||||
{
|
|
||||||
uint8_t spi_data[3];
|
|
||||||
//uint16_t i;
|
|
||||||
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
AddrSel |= (_W5500_SPI_READ_ | _W5500_SPI_VDM_OP_);
|
|
||||||
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x00FF0000) >> 16);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x0000FF00) >> 8);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x000000FF) >> 0);
|
|
||||||
//for(i = 0; i < len; i++)
|
|
||||||
// pBuf[i] = WIZCHIP.IF.SPI._read_byte();
|
|
||||||
spi_data[0] = (AddrSel & 0x00FF0000) >> 16;
|
|
||||||
spi_data[1] = (AddrSel & 0x0000FF00) >> 8;
|
|
||||||
spi_data[2] = (AddrSel & 0x000000FF) >> 0;
|
|
||||||
Chip_SSP_WriteFrames_Blocking(LPC_SSP0, spi_data, 3);
|
|
||||||
Chip_SSP_ReadFrames_Blocking(LPC_SSP0, pBuf, len);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WIZCHIP_WRITE_BUF(uint32_t AddrSel, uint8_t* pBuf, uint16_t len)
|
|
||||||
{
|
|
||||||
uint8_t spi_data[3];
|
|
||||||
//uint16_t i;
|
|
||||||
|
|
||||||
WIZCHIP_CRITICAL_ENTER();
|
|
||||||
WIZCHIP.CS._select();
|
|
||||||
|
|
||||||
AddrSel |= (_W5500_SPI_WRITE_ | _W5500_SPI_VDM_OP_);
|
|
||||||
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x00FF0000) >> 16);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x0000FF00) >> 8);
|
|
||||||
//WIZCHIP.IF.SPI._write_byte((AddrSel & 0x000000FF) >> 0);
|
|
||||||
//for(i = 0; i < len; i++)
|
|
||||||
// WIZCHIP.IF.SPI._write_byte(pBuf[i]);
|
|
||||||
spi_data[0] = (AddrSel & 0x00FF0000) >> 16;
|
|
||||||
spi_data[1] = (AddrSel & 0x0000FF00) >> 8;
|
|
||||||
spi_data[2] = (AddrSel & 0x000000FF) >> 0;
|
|
||||||
Chip_SSP_WriteFrames_Blocking(LPC_SSP0, spi_data, 3);
|
|
||||||
Chip_SSP_WriteFrames_Blocking(LPC_SSP0, pBuf, len);
|
|
||||||
|
|
||||||
WIZCHIP.CS._deselect();
|
|
||||||
WIZCHIP_CRITICAL_EXIT();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
uint16_t getSn_TX_FSR(uint8_t sn)
|
|
||||||
{
|
|
||||||
uint16_t val=0,val1=0;
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
val1 = WIZCHIP_READ(Sn_TX_FSR(sn));
|
|
||||||
val1 = (val1 << 8) + WIZCHIP_READ(WIZCHIP_OFFSET_INC(Sn_TX_FSR(sn),1));
|
|
||||||
if (val1 != 0)
|
|
||||||
{
|
|
||||||
val = WIZCHIP_READ(Sn_TX_FSR(sn));
|
|
||||||
val = (val << 8) + WIZCHIP_READ(WIZCHIP_OFFSET_INC(Sn_TX_FSR(sn),1));
|
|
||||||
}
|
|
||||||
}while (val != val1);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
uint16_t getSn_RX_RSR(uint8_t sn)
|
|
||||||
{
|
|
||||||
uint16_t val=0,val1=0;
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
val1 = WIZCHIP_READ(Sn_RX_RSR(sn));
|
|
||||||
val1 = (val1 << 8) + WIZCHIP_READ(WIZCHIP_OFFSET_INC(Sn_RX_RSR(sn),1));
|
|
||||||
if (val1 != 0)
|
|
||||||
{
|
|
||||||
val = WIZCHIP_READ(Sn_RX_RSR(sn));
|
|
||||||
val = (val << 8) + WIZCHIP_READ(WIZCHIP_OFFSET_INC(Sn_RX_RSR(sn),1));
|
|
||||||
}
|
|
||||||
}while (val != val1);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
void wiz_send_data(uint8_t sn, uint8_t *wizdata, uint16_t len)
|
|
||||||
{
|
|
||||||
uint16_t ptr = 0;
|
|
||||||
uint32_t addrsel = 0;
|
|
||||||
|
|
||||||
if(len == 0) return;
|
|
||||||
ptr = getSn_TX_WR(sn);
|
|
||||||
//M20140501 : implict type casting -> explict type casting
|
|
||||||
//addrsel = (ptr << 8) + (WIZCHIP_TXBUF_BLOCK(sn) << 3);
|
|
||||||
addrsel = ((uint32_t)ptr << 8) + (WIZCHIP_TXBUF_BLOCK(sn) << 3);
|
|
||||||
//
|
|
||||||
WIZCHIP_WRITE_BUF(addrsel,wizdata, len);
|
|
||||||
|
|
||||||
ptr += len;
|
|
||||||
setSn_TX_WR(sn,ptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
void wiz_recv_data(uint8_t sn, uint8_t *wizdata, uint16_t len)
|
|
||||||
{
|
|
||||||
uint16_t ptr = 0;
|
|
||||||
uint32_t addrsel = 0;
|
|
||||||
|
|
||||||
if(len == 0) return;
|
|
||||||
ptr = getSn_RX_RD(sn);
|
|
||||||
//M20140501 : implict type casting -> explict type casting
|
|
||||||
//addrsel = ((ptr << 8) + (WIZCHIP_RXBUF_BLOCK(sn) << 3);
|
|
||||||
addrsel = ((uint32_t)ptr << 8) + (WIZCHIP_RXBUF_BLOCK(sn) << 3);
|
|
||||||
//
|
|
||||||
WIZCHIP_READ_BUF(addrsel, wizdata, len);
|
|
||||||
ptr += len;
|
|
||||||
|
|
||||||
setSn_RX_RD(sn,ptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void wiz_recv_ignore(uint8_t sn, uint16_t len)
|
|
||||||
{
|
|
||||||
uint16_t ptr = 0;
|
|
||||||
|
|
||||||
ptr = getSn_RX_RD(sn);
|
|
||||||
ptr += len;
|
|
||||||
setSn_RX_RD(sn,ptr);
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
@ -1,662 +0,0 @@
|
|||||||
//****************************************************************************/
|
|
||||||
//!
|
|
||||||
//! \file wizchip_conf.c
|
|
||||||
//! \brief WIZCHIP Config Header File.
|
|
||||||
//! \version 1.0.1
|
|
||||||
//! \date 2013/10/21
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2014/05/01> V1.0.1 Refer to M20140501
|
|
||||||
//! 1. Explicit type casting in wizchip_bus_readbyte() & wizchip_bus_writebyte()
|
|
||||||
// Issued by Mathias ClauBen.
|
|
||||||
//! uint32_t type converts into ptrdiff_t first. And then recoverting it into uint8_t*
|
|
||||||
//! For remove the warning when pointer type size is not 32bit.
|
|
||||||
//! If ptrdiff_t doesn't support in your complier, You should must replace ptrdiff_t into your suitable pointer type.
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! \author MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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.
|
|
||||||
//
|
|
||||||
//*****************************************************************************/
|
|
||||||
//A20140501 : for use the type - ptrdiff_t
|
|
||||||
#include <stddef.h>
|
|
||||||
//
|
|
||||||
|
|
||||||
#include "wizchip_conf.h"
|
|
||||||
#include "socket.h"
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Default function to enable interrupt.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
void wizchip_cris_enter(void) {};
|
|
||||||
/**
|
|
||||||
* @brief Default function to disable interrupt.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
void wizchip_cris_exit(void) {};
|
|
||||||
/**
|
|
||||||
* @brief Default function to select chip.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
void wizchip_cs_select(void) {};
|
|
||||||
/**
|
|
||||||
* @brief Default function to deselect chip.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
void wizchip_cs_deselect(void) {};
|
|
||||||
/**
|
|
||||||
* @brief Default function to read in direct or indirect interface.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
//M20140501 : Explict pointer type casting
|
|
||||||
//uint8_t wizchip_bus_readbyte(uint32_t AddrSel) { return * ((volatile uint8_t *) AddrSel); };
|
|
||||||
uint8_t wizchip_bus_readbyte(uint32_t AddrSel) { return * ((volatile uint8_t *)((ptrdiff_t) AddrSel)); };
|
|
||||||
/**
|
|
||||||
* @brief Default function to write in direct or indirect interface.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
|
|
||||||
//M20140501 : Explict pointer type casting
|
|
||||||
//void wizchip_bus_writebyte(uint32_t AddrSel, uint8_t wb) { *((volatile uint8_t*) AddrSel) = wb; };
|
|
||||||
void wizchip_bus_writebyte(uint32_t AddrSel, uint8_t wb) { *((volatile uint8_t*)((ptrdiff_t)AddrSel)) = wb; };
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Default function to read in SPI interface.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
void wizchip_spi_readbytes(uint8_t *buf, uint32_t len) {}
|
|
||||||
/**
|
|
||||||
* @brief Default function to write in SPI interface.
|
|
||||||
* @note This function help not to access wrong address. If you do not describe this function or register any functions,
|
|
||||||
* null function is called.
|
|
||||||
*/
|
|
||||||
void wizchip_spi_writebytes(const uint8_t *buf, uint32_t len) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @\ref _WIZCHIP instance
|
|
||||||
*/
|
|
||||||
_WIZCHIP WIZCHIP =
|
|
||||||
{
|
|
||||||
.id = _WIZCHIP_ID_,
|
|
||||||
.if_mode = _WIZCHIP_IO_MODE_,
|
|
||||||
.CRIS._enter = wizchip_cris_enter,
|
|
||||||
.CRIS._exit = wizchip_cris_exit,
|
|
||||||
.CS._select = wizchip_cs_select,
|
|
||||||
.CS._deselect = wizchip_cs_deselect,
|
|
||||||
.IF.BUS._read_byte = wizchip_bus_readbyte,
|
|
||||||
.IF.BUS._write_byte = wizchip_bus_writebyte
|
|
||||||
// .IF.SPI._read_byte = wizchip_spi_readbyte,
|
|
||||||
// .IF.SPI._write_byte = wizchip_spi_writebyte
|
|
||||||
};
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
static uint8_t _SUBN_[4]; // subnet
|
|
||||||
#endif
|
|
||||||
static uint8_t _DNS_[4]; // DNS server ip address
|
|
||||||
static dhcp_mode _DHCP_; // DHCP mode
|
|
||||||
|
|
||||||
void reg_wizchip_cris_cbfunc(void(*cris_en)(void), void(*cris_ex)(void))
|
|
||||||
{
|
|
||||||
if(!cris_en || !cris_ex)
|
|
||||||
{
|
|
||||||
WIZCHIP.CRIS._enter = wizchip_cris_enter;
|
|
||||||
WIZCHIP.CRIS._exit = wizchip_cris_exit;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
WIZCHIP.CRIS._enter = cris_en;
|
|
||||||
WIZCHIP.CRIS._exit = cris_ex;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void reg_wizchip_cs_cbfunc(void(*cs_sel)(void), void(*cs_desel)(void))
|
|
||||||
{
|
|
||||||
if(!cs_sel || !cs_desel)
|
|
||||||
{
|
|
||||||
WIZCHIP.CS._select = wizchip_cs_select;
|
|
||||||
WIZCHIP.CS._deselect = wizchip_cs_deselect;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
WIZCHIP.CS._select = cs_sel;
|
|
||||||
WIZCHIP.CS._deselect = cs_desel;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void reg_wizchip_bus_cbfunc(uint8_t(*bus_rb)(uint32_t addr), void (*bus_wb)(uint32_t addr, uint8_t wb))
|
|
||||||
{
|
|
||||||
while(!(WIZCHIP.if_mode & _WIZCHIP_IO_MODE_BUS_));
|
|
||||||
|
|
||||||
if(!bus_rb || !bus_wb)
|
|
||||||
{
|
|
||||||
WIZCHIP.IF.BUS._read_byte = wizchip_bus_readbyte;
|
|
||||||
WIZCHIP.IF.BUS._write_byte = wizchip_bus_writebyte;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
WIZCHIP.IF.BUS._read_byte = bus_rb;
|
|
||||||
WIZCHIP.IF.BUS._write_byte = bus_wb;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void reg_wizchip_spi_cbfunc(void (*spi_rb)(uint8_t *, uint32_t), void (*spi_wb)(const uint8_t *, uint32_t))
|
|
||||||
{
|
|
||||||
while(!(WIZCHIP.if_mode & _WIZCHIP_IO_MODE_SPI_));
|
|
||||||
|
|
||||||
if(!spi_rb || !spi_wb)
|
|
||||||
{
|
|
||||||
WIZCHIP.IF.SPI._read_bytes = wizchip_spi_readbytes;
|
|
||||||
WIZCHIP.IF.SPI._write_bytes = wizchip_spi_writebytes;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
WIZCHIP.IF.SPI._read_bytes = spi_rb;
|
|
||||||
WIZCHIP.IF.SPI._write_bytes = spi_wb;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t ctlwizchip(ctlwizchip_type cwtype, void* arg)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
uint8_t* ptmp[2] = {0,0};
|
|
||||||
switch(cwtype)
|
|
||||||
{
|
|
||||||
case CW_RESET_WIZCHIP:
|
|
||||||
wizchip_sw_reset();
|
|
||||||
break;
|
|
||||||
case CW_INIT_WIZCHIP:
|
|
||||||
if(arg != 0)
|
|
||||||
{
|
|
||||||
ptmp[0] = (uint8_t*)arg;
|
|
||||||
ptmp[1] = ptmp[0] + _WIZCHIP_SOCK_NUM_;
|
|
||||||
}
|
|
||||||
return wizchip_init(ptmp[0], ptmp[1]);
|
|
||||||
case CW_CLR_INTERRUPT:
|
|
||||||
wizchip_clrinterrupt(*((intr_kind*)arg));
|
|
||||||
break;
|
|
||||||
case CW_GET_INTERRUPT:
|
|
||||||
*((intr_kind*)arg) = wizchip_getinterrupt();
|
|
||||||
break;
|
|
||||||
case CW_SET_INTRMASK:
|
|
||||||
wizchip_setinterruptmask(*((intr_kind*)arg));
|
|
||||||
break;
|
|
||||||
case CW_GET_INTRMASK:
|
|
||||||
*((intr_kind*)arg) = wizchip_getinterruptmask();
|
|
||||||
break;
|
|
||||||
#if _WIZCHIP_ > 5100
|
|
||||||
case CW_SET_INTRTIME:
|
|
||||||
setINTLEVEL(*(uint16_t*)arg);
|
|
||||||
break;
|
|
||||||
case CW_GET_INTRTIME:
|
|
||||||
*(uint16_t*)arg = getINTLEVEL();
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
case CW_GET_ID:
|
|
||||||
((uint8_t*)arg)[0] = WIZCHIP.id[0];
|
|
||||||
((uint8_t*)arg)[1] = WIZCHIP.id[1];
|
|
||||||
((uint8_t*)arg)[2] = WIZCHIP.id[2];
|
|
||||||
((uint8_t*)arg)[3] = WIZCHIP.id[3];
|
|
||||||
((uint8_t*)arg)[4] = WIZCHIP.id[4];
|
|
||||||
((uint8_t*)arg)[5] = 0;
|
|
||||||
break;
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
case CW_RESET_PHY:
|
|
||||||
wizphy_reset();
|
|
||||||
break;
|
|
||||||
case CW_SET_PHYCONF:
|
|
||||||
wizphy_setphyconf((wiz_PhyConf*)arg);
|
|
||||||
break;
|
|
||||||
case CW_GET_PHYCONF:
|
|
||||||
wizphy_getphyconf((wiz_PhyConf*)arg);
|
|
||||||
break;
|
|
||||||
case CW_GET_PHYSTATUS:
|
|
||||||
break;
|
|
||||||
case CW_SET_PHYPOWMODE:
|
|
||||||
return wizphy_setphypmode(*(uint8_t*)arg);
|
|
||||||
#endif
|
|
||||||
case CW_GET_PHYPOWMODE:
|
|
||||||
tmp = wizphy_getphypmode();
|
|
||||||
if((int8_t)tmp == -1) return -1;
|
|
||||||
*(uint8_t*)arg = tmp;
|
|
||||||
break;
|
|
||||||
case CW_GET_PHYLINK:
|
|
||||||
tmp = wizphy_getphylink();
|
|
||||||
if((int8_t)tmp == -1) return -1;
|
|
||||||
*(uint8_t*)arg = tmp;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int8_t ctlnetwork(ctlnetwork_type cntype, void* arg)
|
|
||||||
{
|
|
||||||
|
|
||||||
switch(cntype)
|
|
||||||
{
|
|
||||||
case CN_SET_NETINFO:
|
|
||||||
wizchip_setnetinfo((wiz_NetInfo*)arg);
|
|
||||||
break;
|
|
||||||
case CN_GET_NETINFO:
|
|
||||||
wizchip_getnetinfo((wiz_NetInfo*)arg);
|
|
||||||
break;
|
|
||||||
case CN_SET_NETMODE:
|
|
||||||
return wizchip_setnetmode(*(netmode_type*)arg);
|
|
||||||
case CN_GET_NETMODE:
|
|
||||||
*(netmode_type*)arg = wizchip_getnetmode();
|
|
||||||
break;
|
|
||||||
case CN_SET_TIMEOUT:
|
|
||||||
wizchip_settimeout((wiz_NetTimeout*)arg);
|
|
||||||
break;
|
|
||||||
case CN_GET_TIMEOUT:
|
|
||||||
wizchip_gettimeout((wiz_NetTimeout*)arg);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizchip_sw_reset(void)
|
|
||||||
{
|
|
||||||
uint8_t gw[4], sn[4], sip[4];
|
|
||||||
uint8_t mac[6];
|
|
||||||
getSHAR(mac);
|
|
||||||
getGAR(gw); getSUBR(sn); getSIPR(sip);
|
|
||||||
setMR(MR_RST);
|
|
||||||
getMR(); // for delay
|
|
||||||
setSHAR(mac);
|
|
||||||
setGAR(gw);
|
|
||||||
setSUBR(sn);
|
|
||||||
setSIPR(sip);
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t wizchip_init(uint8_t* txsize, uint8_t* rxsize)
|
|
||||||
{
|
|
||||||
int8_t i;
|
|
||||||
int8_t tmp = 0;
|
|
||||||
wizchip_sw_reset();
|
|
||||||
if(txsize)
|
|
||||||
{
|
|
||||||
tmp = 0;
|
|
||||||
for(i = 0 ; i < _WIZCHIP_SOCK_NUM_; i++)
|
|
||||||
tmp += txsize[i];
|
|
||||||
if(tmp > 16) return -1;
|
|
||||||
for(i = 0 ; i < _WIZCHIP_SOCK_NUM_; i++)
|
|
||||||
setSn_TXBUF_SIZE(i, txsize[i]);
|
|
||||||
}
|
|
||||||
if(rxsize)
|
|
||||||
{
|
|
||||||
tmp = 0;
|
|
||||||
for(i = 0 ; i < _WIZCHIP_SOCK_NUM_; i++)
|
|
||||||
tmp += rxsize[i];
|
|
||||||
if(tmp > 16) return -1;
|
|
||||||
for(i = 0 ; i < _WIZCHIP_SOCK_NUM_; i++)
|
|
||||||
setSn_RXBUF_SIZE(i, rxsize[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
WIZCHIP_EXPORT(socket_reset)();
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizchip_clrinterrupt(intr_kind intr)
|
|
||||||
{
|
|
||||||
uint8_t ir = (uint8_t)intr;
|
|
||||||
uint8_t sir = (uint8_t)((uint16_t)intr >> 8);
|
|
||||||
#if _WIZCHIP_ < 5500
|
|
||||||
ir |= (1<<4); // IK_WOL
|
|
||||||
#endif
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
ir |= (1 << 6);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ < 5200
|
|
||||||
sir &= 0x0F;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5100
|
|
||||||
ir |= sir;
|
|
||||||
setIR(ir);
|
|
||||||
#else
|
|
||||||
setIR(ir);
|
|
||||||
setSIR(sir);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
intr_kind wizchip_getinterrupt(void)
|
|
||||||
{
|
|
||||||
uint8_t ir = 0;
|
|
||||||
uint8_t sir = 0;
|
|
||||||
uint16_t ret = 0;
|
|
||||||
#if _WIZCHIP_ == 5100
|
|
||||||
ir = getIR();
|
|
||||||
sir = ir 0x0F;
|
|
||||||
#else
|
|
||||||
ir = getIR();
|
|
||||||
sir = getSIR();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ < 5500
|
|
||||||
ir &= ~(1<<4); // IK_WOL
|
|
||||||
#endif
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
ir &= ~(1 << 6);
|
|
||||||
#endif
|
|
||||||
ret = sir;
|
|
||||||
ret = (ret << 8) + ir;
|
|
||||||
return (intr_kind)ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizchip_setinterruptmask(intr_kind intr)
|
|
||||||
{
|
|
||||||
uint8_t imr = (uint8_t)intr;
|
|
||||||
uint8_t simr = (uint8_t)((uint16_t)intr >> 8);
|
|
||||||
#if _WIZCHIP_ < 5500
|
|
||||||
imr &= ~(1<<4); // IK_WOL
|
|
||||||
#endif
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
imr &= ~(1 << 6);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ < 5200
|
|
||||||
simr &= 0x0F;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5100
|
|
||||||
imr |= simr;
|
|
||||||
setIMR(imr);
|
|
||||||
#else
|
|
||||||
setIMR(imr);
|
|
||||||
setSIMR(simr);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
intr_kind wizchip_getinterruptmask(void)
|
|
||||||
{
|
|
||||||
uint8_t imr = 0;
|
|
||||||
uint8_t simr = 0;
|
|
||||||
uint16_t ret = 0;
|
|
||||||
#if _WIZCHIP_ == 5100
|
|
||||||
imr = getIMR();
|
|
||||||
simr = imr 0x0F;
|
|
||||||
#else
|
|
||||||
imr = getIMR();
|
|
||||||
simr = getSIMR();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ < 5500
|
|
||||||
imr &= ~(1<<4); // IK_WOL
|
|
||||||
#endif
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
imr &= ~(1 << 6); // IK_DEST_UNREACH
|
|
||||||
#endif
|
|
||||||
ret = simr;
|
|
||||||
ret = (ret << 8) + imr;
|
|
||||||
return (intr_kind)ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t wizphy_getphylink(void)
|
|
||||||
{
|
|
||||||
int8_t tmp;
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
if(getPHYSTATUS() & PHYSTATUS_LINK)
|
|
||||||
tmp = PHY_LINK_ON;
|
|
||||||
else
|
|
||||||
tmp = PHY_LINK_OFF;
|
|
||||||
#elif _WIZCHIP_ == 5500
|
|
||||||
if(getPHYCFGR() & PHYCFGR_LNK_ON)
|
|
||||||
tmp = PHY_LINK_ON;
|
|
||||||
else
|
|
||||||
tmp = PHY_LINK_OFF;
|
|
||||||
#else
|
|
||||||
tmp = -1;
|
|
||||||
#endif
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if _WIZCHIP_ > 5100
|
|
||||||
|
|
||||||
int8_t wizphy_getphypmode(void)
|
|
||||||
{
|
|
||||||
int8_t tmp = 0;
|
|
||||||
#if _WIZCHIP_ == 5200
|
|
||||||
if(getPHYSTATUS() & PHYSTATUS_POWERDOWN)
|
|
||||||
tmp = PHY_POWER_DOWN;
|
|
||||||
else
|
|
||||||
tmp = PHY_POWER_NORM;
|
|
||||||
#elif _WIZCHIP_ == 5500
|
|
||||||
if(getPHYCFGR() & PHYCFGR_OPMDC_PDOWN)
|
|
||||||
tmp = PHY_POWER_DOWN;
|
|
||||||
else
|
|
||||||
tmp = PHY_POWER_NORM;
|
|
||||||
#else
|
|
||||||
tmp = -1;
|
|
||||||
#endif
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
void wizphy_reset(void)
|
|
||||||
{
|
|
||||||
uint8_t tmp = getPHYCFGR();
|
|
||||||
tmp &= PHYCFGR_RST;
|
|
||||||
setPHYCFGR(tmp);
|
|
||||||
tmp = getPHYCFGR();
|
|
||||||
tmp |= ~PHYCFGR_RST;
|
|
||||||
setPHYCFGR(tmp);
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizphy_setphyconf(wiz_PhyConf* phyconf)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
if(phyconf->by == PHY_CONFBY_SW)
|
|
||||||
tmp |= PHYCFGR_OPMD;
|
|
||||||
else
|
|
||||||
tmp &= ~PHYCFGR_OPMD;
|
|
||||||
if(phyconf->mode == PHY_MODE_AUTONEGO)
|
|
||||||
tmp |= PHYCFGR_OPMDC_ALLA;
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if(phyconf->duplex == PHY_DUPLEX_FULL)
|
|
||||||
{
|
|
||||||
if(phyconf->speed == PHY_SPEED_100)
|
|
||||||
tmp |= PHYCFGR_OPMDC_100F;
|
|
||||||
else
|
|
||||||
tmp |= PHYCFGR_OPMDC_10F;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if(phyconf->speed == PHY_SPEED_100)
|
|
||||||
tmp |= PHYCFGR_OPMDC_100H;
|
|
||||||
else
|
|
||||||
tmp |= PHYCFGR_OPMDC_10H;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setPHYCFGR(tmp);
|
|
||||||
wizphy_reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizphy_getphyconf(wiz_PhyConf* phyconf)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
tmp = getPHYCFGR();
|
|
||||||
phyconf->by = (tmp & PHYCFGR_OPMD) ? PHY_CONFBY_SW : PHY_CONFBY_HW;
|
|
||||||
switch(tmp & PHYCFGR_OPMDC_ALLA)
|
|
||||||
{
|
|
||||||
case PHYCFGR_OPMDC_ALLA:
|
|
||||||
case PHYCFGR_OPMDC_100FA:
|
|
||||||
phyconf->mode = PHY_MODE_AUTONEGO;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
phyconf->mode = PHY_MODE_MANUAL;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
switch(tmp & PHYCFGR_OPMDC_ALLA)
|
|
||||||
{
|
|
||||||
case PHYCFGR_OPMDC_100FA:
|
|
||||||
case PHYCFGR_OPMDC_100F:
|
|
||||||
case PHYCFGR_OPMDC_100H:
|
|
||||||
phyconf->speed = PHY_SPEED_100;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
phyconf->speed = PHY_SPEED_10;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
switch(tmp & PHYCFGR_OPMDC_ALLA)
|
|
||||||
{
|
|
||||||
case PHYCFGR_OPMDC_100FA:
|
|
||||||
case PHYCFGR_OPMDC_100F:
|
|
||||||
case PHYCFGR_OPMDC_10F:
|
|
||||||
phyconf->duplex = PHY_DUPLEX_FULL;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
phyconf->duplex = PHY_DUPLEX_HALF;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizphy_getphystat(wiz_PhyConf* phyconf)
|
|
||||||
{
|
|
||||||
uint8_t tmp = getPHYCFGR();
|
|
||||||
phyconf->duplex = (tmp & PHYCFGR_DPX_FULL) ? PHY_DUPLEX_FULL : PHY_DUPLEX_HALF;
|
|
||||||
phyconf->speed = (tmp & PHYCFGR_SPD_100) ? PHY_SPEED_100 : PHY_SPEED_10;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t wizphy_setphypmode(uint8_t pmode)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
tmp = getPHYCFGR();
|
|
||||||
if((tmp & PHYCFGR_OPMD)== 0) return -1;
|
|
||||||
tmp &= ~PHYCFGR_OPMDC_ALLA;
|
|
||||||
if( pmode == PHY_POWER_DOWN)
|
|
||||||
tmp |= PHYCFGR_OPMDC_PDOWN;
|
|
||||||
else
|
|
||||||
tmp |= PHYCFGR_OPMDC_ALLA;
|
|
||||||
setPHYCFGR(tmp);
|
|
||||||
wizphy_reset();
|
|
||||||
tmp = getPHYCFGR();
|
|
||||||
if( pmode == PHY_POWER_DOWN)
|
|
||||||
{
|
|
||||||
if(tmp & PHYCFGR_OPMDC_PDOWN) return 0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if(tmp & PHYCFGR_OPMDC_ALLA) return 0;
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
void wizchip_setnetinfo(wiz_NetInfo* pnetinfo)
|
|
||||||
{
|
|
||||||
setSHAR(pnetinfo->mac);
|
|
||||||
setGAR(pnetinfo->gw);
|
|
||||||
setSUBR(pnetinfo->sn);
|
|
||||||
setSIPR(pnetinfo->ip);
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
_SUBN_[0] = pnetinfo->sn[0];
|
|
||||||
_SUBN_[1] = pnetinfo->sn[1];
|
|
||||||
_SUBN_[2] = pnetinfo->sn[2];
|
|
||||||
_SUBN_[3] = pnetinfo->sn[3];
|
|
||||||
#endif
|
|
||||||
_DNS_[0] = pnetinfo->dns[0];
|
|
||||||
_DNS_[1] = pnetinfo->dns[1];
|
|
||||||
_DNS_[2] = pnetinfo->dns[2];
|
|
||||||
_DNS_[3] = pnetinfo->dns[3];
|
|
||||||
_DHCP_ = pnetinfo->dhcp;
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizchip_getnetinfo(wiz_NetInfo* pnetinfo)
|
|
||||||
{
|
|
||||||
getSHAR(pnetinfo->mac);
|
|
||||||
getGAR(pnetinfo->gw);
|
|
||||||
getSUBR(pnetinfo->sn);
|
|
||||||
getSIPR(pnetinfo->ip);
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
pnetinfo->sn[0] = _SUBN_[0];
|
|
||||||
pnetinfo->sn[1] = _SUBN_[1];
|
|
||||||
pnetinfo->sn[2] = _SUBN_[2];
|
|
||||||
pnetinfo->sn[3] = _SUBN_[3];
|
|
||||||
#endif
|
|
||||||
pnetinfo->dns[0]= _DNS_[0];
|
|
||||||
pnetinfo->dns[1]= _DNS_[1];
|
|
||||||
pnetinfo->dns[2]= _DNS_[2];
|
|
||||||
pnetinfo->dns[3]= _DNS_[3];
|
|
||||||
pnetinfo->dhcp = _DHCP_;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
uint8_t *wizchip_getsubn(void) {
|
|
||||||
return _SUBN_;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
int8_t wizchip_setnetmode(netmode_type netmode)
|
|
||||||
{
|
|
||||||
uint8_t tmp = 0;
|
|
||||||
#if _WIZCHIP_ != 5500
|
|
||||||
if(netmode & ~(NM_WAKEONLAN | NM_PPPOE | NM_PINGBLOCK)) return -1;
|
|
||||||
#else
|
|
||||||
if(netmode & ~(NM_WAKEONLAN | NM_PPPOE | NM_PINGBLOCK | NM_FORCEARP)) return -1;
|
|
||||||
#endif
|
|
||||||
tmp = getMR();
|
|
||||||
tmp |= (uint8_t)netmode;
|
|
||||||
setMR(tmp);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
netmode_type wizchip_getnetmode(void)
|
|
||||||
{
|
|
||||||
return (netmode_type) getMR();
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizchip_settimeout(wiz_NetTimeout* nettime)
|
|
||||||
{
|
|
||||||
setRCR(nettime->retry_cnt);
|
|
||||||
setRTR(nettime->time_100us);
|
|
||||||
}
|
|
||||||
|
|
||||||
void wizchip_gettimeout(wiz_NetTimeout* nettime)
|
|
||||||
{
|
|
||||||
nettime->retry_cnt = getRCR();
|
|
||||||
nettime->time_100us = getRTR();
|
|
||||||
}
|
|
@ -1,554 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file wizchip_conf.h
|
|
||||||
//! \brief WIZCHIP Config Header File.
|
|
||||||
//! \version 1.0.0
|
|
||||||
//! \date 2013/10/21
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! \author MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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.
|
|
||||||
//
|
|
||||||
//*****************************************************************************
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @defgroup extra_functions 2. WIZnet Extra Functions
|
|
||||||
*
|
|
||||||
* @brief These functions is optional function. It could be replaced at WIZCHIP I/O function because they were made by WIZCHIP I/O functions.
|
|
||||||
* @details There are functions of configuring WIZCHIP, network, interrupt, phy, network information and timer. \n
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _WIZCHIP_CONF_H_
|
|
||||||
#define _WIZCHIP_CONF_H_
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
/**
|
|
||||||
* @brief Select WIZCHIP.
|
|
||||||
* @todo You should select one, \b 5100, \b 5200 ,\b 5500 or etc. \n\n
|
|
||||||
* ex> <code> #define \_WIZCHIP_ 5500 </code>
|
|
||||||
*/
|
|
||||||
#ifndef _WIZCHIP_
|
|
||||||
#define _WIZCHIP_ 5200 // 5100, 5200, 5500
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define _WIZCHIP_IO_MODE_NONE_ 0x0000
|
|
||||||
#define _WIZCHIP_IO_MODE_BUS_ 0x0100 /**< Bus interface mode */
|
|
||||||
#define _WIZCHIP_IO_MODE_SPI_ 0x0200 /**< SPI interface mode */
|
|
||||||
//#define _WIZCHIP_IO_MODE_IIC_ 0x0400
|
|
||||||
//#define _WIZCHIP_IO_MODE_SDIO_ 0x0800
|
|
||||||
// Add to
|
|
||||||
//
|
|
||||||
|
|
||||||
#define _WIZCHIP_IO_MODE_BUS_DIR_ (_WIZCHIP_IO_MODE_BUS_ + 1) /**< BUS interface mode for direct */
|
|
||||||
#define _WIZCHIP_IO_MODE_BUS_INDIR_ (_WIZCHIP_IO_MODE_BUS_ + 2) /**< BUS interface mode for indirect */
|
|
||||||
|
|
||||||
#define _WIZCHIP_IO_MODE_SPI_VDM_ (_WIZCHIP_IO_MODE_SPI_ + 1) /**< SPI interface mode for variable length data*/
|
|
||||||
#define _WIZCHIP_IO_MODE_SPI_FDM_ (_WIZCHIP_IO_MODE_SPI_ + 2) /**< SPI interface mode for fixed length data mode*/
|
|
||||||
|
|
||||||
|
|
||||||
#if (_WIZCHIP_ == 5100)
|
|
||||||
#define _WIZCHIP_ID_ "W5100\0"
|
|
||||||
/**
|
|
||||||
* @brief Define interface mode.
|
|
||||||
* @todo you should select interface mode as chip. Select one of @ref \_WIZCHIP_IO_MODE_SPI_ , @ref \_WIZCHIP_IO_MODE_BUS_DIR_ or @ref \_WIZCHIP_IO_MODE_BUS_INDIR_
|
|
||||||
*/
|
|
||||||
|
|
||||||
// #define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_BUS_DIR_
|
|
||||||
// #define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_BUS_INDIR_
|
|
||||||
#define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_SPI_
|
|
||||||
|
|
||||||
#elif (_WIZCHIP_ == 5200)
|
|
||||||
#define _WIZCHIP_ID_ "W5200\0"
|
|
||||||
/**
|
|
||||||
* @brief Define interface mode.
|
|
||||||
* @todo you should select interface mode as chip. Select one of @ref \_WIZCHIP_IO_MODE_SPI_ or @ref \_WIZCHIP_IO_MODE_BUS_INDIR_
|
|
||||||
*/
|
|
||||||
// #define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_BUS_INDIR_
|
|
||||||
#define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_SPI_
|
|
||||||
#include "w5200/w5200.h"
|
|
||||||
#elif (_WIZCHIP_ == 5500)
|
|
||||||
#define _WIZCHIP_ID_ "W5500\0"
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Define interface mode. \n
|
|
||||||
* @todo Should select interface mode as chip.
|
|
||||||
* - @ref \_WIZCHIP_IO_MODE_SPI_ \n
|
|
||||||
* -@ref \_WIZCHIP_IO_MODE_SPI_VDM_ : Valid only in @ref \_WIZCHIP_ == 5500 \n
|
|
||||||
* -@ref \_WIZCHIP_IO_MODE_SPI_FDM_ : Valid only in @ref \_WIZCHIP_ == 5500 \n
|
|
||||||
* - @ref \_WIZCHIP_IO_MODE_BUS_ \n
|
|
||||||
* - @ref \_WIZCHIP_IO_MODE_BUS_DIR_ \n
|
|
||||||
* - @ref \_WIZCHIP_IO_MODE_BUS_INDIR_ \n
|
|
||||||
* - Others will be defined in future. \n\n
|
|
||||||
* ex> <code> #define \_WIZCHIP_IO_MODE_ \_WIZCHIP_IO_MODE_SPI_VDM_ </code>
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
//#define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_SPI_FDM_
|
|
||||||
#define _WIZCHIP_IO_MODE_ _WIZCHIP_IO_MODE_SPI_VDM_
|
|
||||||
#include "w5500/w5500.h"
|
|
||||||
#else
|
|
||||||
#error "Unknown defined _WIZCHIP_. You should define one of 5100, 5200, and 5500 !!!"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifndef _WIZCHIP_IO_MODE_
|
|
||||||
#error "Undefined _WIZCHIP_IO_MODE_. You should define it !!!"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Define I/O base address when BUS IF mode.
|
|
||||||
* @todo Should re-define it to fit your system when BUS IF Mode (@ref \_WIZCHIP_IO_MODE_BUS_,
|
|
||||||
* @ref \_WIZCHIP_IO_MODE_BUS_DIR_, @ref \_WIZCHIP_IO_MODE_BUS_INDIR_). \n\n
|
|
||||||
* ex> <code> #define \_WIZCHIP_IO_BASE_ 0x00008000 </code>
|
|
||||||
*/
|
|
||||||
#define _WIZCHIP_IO_BASE_ 0x00000000 //
|
|
||||||
|
|
||||||
#if _WIZCHIP_IO_MODE_ & _WIZCHIP_IO_MODE_BUS_
|
|
||||||
#ifndef _WIZCHIP_IO_BASE_
|
|
||||||
#error "You should be define _WIZCHIP_IO_BASE to fit your system memory map."
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ > 5100
|
|
||||||
#define _WIZCHIP_SOCK_NUM_ 8 ///< The count of independant socket of @b WIZCHIP
|
|
||||||
#else
|
|
||||||
#define _WIZCHIP_SOCK_NUM_ 4 ///< The count of independant socket of @b WIZCHIP
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
/********************************************************
|
|
||||||
* WIZCHIP BASIC IF functions for SPI, SDIO, I2C , ETC.
|
|
||||||
*********************************************************/
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* @brief The set of callback functions for W5500:@ref WIZCHIP_IO_Functions W5200:@ref WIZCHIP_IO_Functions_W5200
|
|
||||||
*/
|
|
||||||
typedef struct __WIZCHIP
|
|
||||||
{
|
|
||||||
uint16_t if_mode; ///< host interface mode
|
|
||||||
uint8_t id[6]; ///< @b WIZCHIP ID such as @b 5100, @b 5200, @b 5500, and so on.
|
|
||||||
/**
|
|
||||||
* The set of critical section callback func.
|
|
||||||
*/
|
|
||||||
struct _CRIS
|
|
||||||
{
|
|
||||||
void (*_enter) (void); ///< crtical section enter
|
|
||||||
void (*_exit) (void); ///< critial section exit
|
|
||||||
}CRIS;
|
|
||||||
/**
|
|
||||||
* The set of @ref\_WIZCHIP_ select control callback func.
|
|
||||||
*/
|
|
||||||
struct _CS
|
|
||||||
{
|
|
||||||
void (*_select) (void); ///< @ref \_WIZCHIP_ selected
|
|
||||||
void (*_deselect)(void); ///< @ref \_WIZCHIP_ deselected
|
|
||||||
}CS;
|
|
||||||
/**
|
|
||||||
* The set of interface IO callback func.
|
|
||||||
*/
|
|
||||||
union _IF
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* For BUS interface IO
|
|
||||||
*/
|
|
||||||
struct
|
|
||||||
{
|
|
||||||
uint8_t (*_read_byte) (uint32_t AddrSel);
|
|
||||||
void (*_write_byte) (uint32_t AddrSel, uint8_t wb);
|
|
||||||
}BUS;
|
|
||||||
/**
|
|
||||||
* For SPI interface IO
|
|
||||||
*/
|
|
||||||
struct
|
|
||||||
{
|
|
||||||
void (*_read_bytes) (uint8_t *buf, uint32_t len);
|
|
||||||
void (*_write_bytes) (const uint8_t *buf, uint32_t len);
|
|
||||||
}SPI;
|
|
||||||
// To be added
|
|
||||||
//
|
|
||||||
}IF;
|
|
||||||
}_WIZCHIP;
|
|
||||||
|
|
||||||
extern _WIZCHIP WIZCHIP;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* WIZCHIP control type enumration used in @ref ctlwizchip().
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
CW_RESET_WIZCHIP, ///< Resets WIZCHIP by softly
|
|
||||||
CW_INIT_WIZCHIP, ///< Inializes to WIZCHIP with SOCKET buffer size 2 or 1 dimension array typed uint8_t.
|
|
||||||
CW_GET_INTERRUPT, ///< Get Interrupt status of WIZCHIP
|
|
||||||
CW_CLR_INTERRUPT, ///< Clears interrupt
|
|
||||||
CW_SET_INTRMASK, ///< Masks interrupt
|
|
||||||
CW_GET_INTRMASK, ///< Get interrupt mask
|
|
||||||
CW_SET_INTRTIME, ///< Set interval time between the current and next interrupt.
|
|
||||||
CW_GET_INTRTIME, ///< Set interval time between the current and next interrupt.
|
|
||||||
CW_GET_ID, ///< Gets WIZCHIP name.
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
CW_RESET_PHY, ///< Resets internal PHY. Valid Only W5000
|
|
||||||
CW_SET_PHYCONF, ///< When PHY configured by interal register, PHY operation mode (Manual/Auto, 10/100, Half/Full). Valid Only W5000
|
|
||||||
CW_GET_PHYCONF, ///< Get PHY operation mode in interal register. Valid Only W5000
|
|
||||||
CW_GET_PHYSTATUS, ///< Get real PHY status on operating. Valid Only W5000
|
|
||||||
CW_SET_PHYPOWMODE, ///< Set PHY power mode as noraml and down when PHYSTATUS.OPMD == 1. Valid Only W5000
|
|
||||||
#endif
|
|
||||||
CW_GET_PHYPOWMODE, ///< Get PHY Power mode as down or normal
|
|
||||||
CW_GET_PHYLINK ///< Get PHY Link status
|
|
||||||
}ctlwizchip_type;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* Network control type enumration used in @ref ctlnetwork().
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
CN_SET_NETINFO, ///< Set Network with @ref wiz_NetInfo
|
|
||||||
CN_GET_NETINFO, ///< Get Network with @ref wiz_NetInfo
|
|
||||||
CN_SET_NETMODE, ///< Set network mode as WOL, PPPoE, Ping Block, and Force ARP mode
|
|
||||||
CN_GET_NETMODE, ///< Get network mode as WOL, PPPoE, Ping Block, and Force ARP mode
|
|
||||||
CN_SET_TIMEOUT, ///< Set network timeout as retry count and time.
|
|
||||||
CN_GET_TIMEOUT, ///< Get network timeout as retry count and time.
|
|
||||||
}ctlnetwork_type;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* Interrupt kind when CW_SET_INTRRUPT, CW_GET_INTERRUPT, CW_SET_INTRMASK
|
|
||||||
* and CW_GET_INTRMASK is used in @ref ctlnetwork().
|
|
||||||
* It can be used with OR operation.
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
#if _WIZCHIP_ > 5200
|
|
||||||
IK_WOL = (1 << 4), ///< Wake On Lan by receiving the magic packet. Valid in W500.
|
|
||||||
#endif
|
|
||||||
|
|
||||||
IK_PPPOE_TERMINATED = (1 << 5), ///< PPPoE Disconnected
|
|
||||||
|
|
||||||
#if _WIZCHIP_ != 5200
|
|
||||||
IK_DEST_UNREACH = (1 << 6), ///< Destination IP & Port Unreable, No use in W5200
|
|
||||||
#endif
|
|
||||||
|
|
||||||
IK_IP_CONFLICT = (1 << 7), ///< IP conflict occurred
|
|
||||||
|
|
||||||
IK_SOCK_0 = (1 << 8), ///< Socket 0 interrupt
|
|
||||||
IK_SOCK_1 = (1 << 9), ///< Socket 1 interrupt
|
|
||||||
IK_SOCK_2 = (1 << 10), ///< Socket 2 interrupt
|
|
||||||
IK_SOCK_3 = (1 << 11), ///< Socket 3 interrupt
|
|
||||||
#if _WIZCHIP_ > 5100
|
|
||||||
IK_SOCK_4 = (1 << 12), ///< Socket 4 interrupt, No use in 5100
|
|
||||||
IK_SOCK_5 = (1 << 13), ///< Socket 5 interrupt, No use in 5100
|
|
||||||
IK_SOCK_6 = (1 << 14), ///< Socket 6 interrupt, No use in 5100
|
|
||||||
IK_SOCK_7 = (1 << 15), ///< Socket 7 interrupt, No use in 5100
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ > 5100
|
|
||||||
IK_SOCK_ALL = (0xFF << 8) ///< All Socket interrpt
|
|
||||||
#else
|
|
||||||
IK_SOCK_ALL = (0x0F << 8) ///< All Socket interrpt
|
|
||||||
#endif
|
|
||||||
}intr_kind;
|
|
||||||
|
|
||||||
#define PHY_CONFBY_HW 0 ///< Configured PHY operation mode by HW pin
|
|
||||||
#define PHY_CONFBY_SW 1 ///< Configured PHY operation mode by SW register
|
|
||||||
#define PHY_MODE_MANUAL 0 ///< Configured PHY operation mode with user setting.
|
|
||||||
#define PHY_MODE_AUTONEGO 1 ///< Configured PHY operation mode with auto-negotiation
|
|
||||||
#define PHY_SPEED_10 0 ///< Link Speed 10
|
|
||||||
#define PHY_SPEED_100 1 ///< Link Speed 100
|
|
||||||
#define PHY_DUPLEX_HALF 0 ///< Link Half-Duplex
|
|
||||||
#define PHY_DUPLEX_FULL 1 ///< Link Full-Duplex
|
|
||||||
#define PHY_LINK_OFF 0 ///< Link Off
|
|
||||||
#define PHY_LINK_ON 1 ///< Link On
|
|
||||||
#define PHY_POWER_NORM 0 ///< PHY power normal mode
|
|
||||||
#define PHY_POWER_DOWN 1 ///< PHY power down mode
|
|
||||||
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* It configures PHY configuration when CW_SET PHYCONF or CW_GET_PHYCONF in W5500,
|
|
||||||
* and it indicates the real PHY status configured by HW or SW in all WIZCHIP. \n
|
|
||||||
* Valid only in W5500.
|
|
||||||
*/
|
|
||||||
typedef struct wiz_PhyConf_t
|
|
||||||
{
|
|
||||||
uint8_t by; ///< set by @ref PHY_CONFBY_HW or @ref PHY_CONFBY_SW
|
|
||||||
uint8_t mode; ///< set by @ref PHY_MODE_MANUAL or @ref PHY_MODE_AUTONEGO
|
|
||||||
uint8_t speed; ///< set by @ref PHY_SPEED_10 or @ref PHY_SPEED_100
|
|
||||||
uint8_t duplex; ///< set by @ref PHY_DUPLEX_HALF @ref PHY_DUPLEX_FULL
|
|
||||||
//uint8_t power; ///< set by @ref PHY_POWER_NORM or @ref PHY_POWER_DOWN
|
|
||||||
//uint8_t link; ///< Valid only in CW_GET_PHYSTATUS. set by @ref PHY_LINK_ON or PHY_DUPLEX_OFF
|
|
||||||
}wiz_PhyConf;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* It used in setting dhcp_mode of @ref wiz_NetInfo.
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
NETINFO_STATIC = 1, ///< Static IP configuration by manually.
|
|
||||||
NETINFO_DHCP ///< Dynamic IP configruation from a DHCP sever
|
|
||||||
}dhcp_mode;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* Network Information for WIZCHIP
|
|
||||||
*/
|
|
||||||
typedef struct wiz_NetInfo_t
|
|
||||||
{
|
|
||||||
uint8_t mac[6]; ///< Source Mac Address
|
|
||||||
uint8_t ip[4]; ///< Source IP Address
|
|
||||||
uint8_t sn[4]; ///< Subnet Mask
|
|
||||||
uint8_t gw[4]; ///< Gateway IP Address
|
|
||||||
uint8_t dns[4]; ///< DNS server IP Address
|
|
||||||
dhcp_mode dhcp; ///< 1 - Static, 2 - DHCP
|
|
||||||
}wiz_NetInfo;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* Network mode
|
|
||||||
*/
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
NM_FORCEARP = (1<<1), ///< Force to APP send whenever udp data is sent. Valid only in W5500
|
|
||||||
#endif
|
|
||||||
NM_WAKEONLAN = (1<<5), ///< Wake On Lan
|
|
||||||
NM_PINGBLOCK = (1<<4), ///< Block ping-request
|
|
||||||
NM_PPPOE = (1<<3), ///< PPPoE mode
|
|
||||||
}netmode_type;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup DATA_TYPE
|
|
||||||
* Used in CN_SET_TIMEOUT or CN_GET_TIMEOUT of @ref ctlwizchip() for timeout configruation.
|
|
||||||
*/
|
|
||||||
typedef struct wiz_NetTimeout_t
|
|
||||||
{
|
|
||||||
uint8_t retry_cnt; ///< retry count
|
|
||||||
uint16_t time_100us; ///< time unit 100us
|
|
||||||
}wiz_NetTimeout;
|
|
||||||
|
|
||||||
/**
|
|
||||||
*@brief Registers call back function for critical section of I/O functions such as
|
|
||||||
*\ref WIZCHIP_READ, @ref WIZCHIP_WRITE, @ref WIZCHIP_READ_BUF and @ref WIZCHIP_WRITE_BUF.
|
|
||||||
*@param cris_en : callback function for critical section enter.
|
|
||||||
*@param cris_ex : callback function for critical section exit.
|
|
||||||
*@todo Describe @ref WIZCHIP_CRITICAL_ENTER and @ref WIZCHIP_CRITICAL_EXIT marco or register your functions.
|
|
||||||
*@note If you do not describe or register, default functions(@ref wizchip_cris_enter & @ref wizchip_cris_exit) is called.
|
|
||||||
*/
|
|
||||||
void reg_wizchip_cris_cbfunc(void(*cris_en)(void), void(*cris_ex)(void));
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
*@brief Registers call back function for WIZCHIP select & deselect.
|
|
||||||
*@param cs_sel : callback function for WIZCHIP select
|
|
||||||
*@param cs_desel : callback fucntion for WIZCHIP deselect
|
|
||||||
*@todo Describe @ref wizchip_cs_select and @ref wizchip_cs_deselect function or register your functions.
|
|
||||||
*@note If you do not describe or register, null function is called.
|
|
||||||
*/
|
|
||||||
void reg_wizchip_cs_cbfunc(void(*cs_sel)(void), void(*cs_desel)(void));
|
|
||||||
|
|
||||||
/**
|
|
||||||
*@brief Registers call back function for bus interface.
|
|
||||||
*@param bus_rb : callback function to read byte data using system bus
|
|
||||||
*@param bus_wb : callback function to write byte data using system bus
|
|
||||||
*@todo Describe @ref wizchip_bus_readbyte and @ref wizchip_bus_writebyte function
|
|
||||||
*or register your functions.
|
|
||||||
*@note If you do not describe or register, null function is called.
|
|
||||||
*/
|
|
||||||
void reg_wizchip_bus_cbfunc(uint8_t (*bus_rb)(uint32_t addr), void (*bus_wb)(uint32_t addr, uint8_t wb));
|
|
||||||
|
|
||||||
/**
|
|
||||||
*@brief Registers call back function for SPI interface.
|
|
||||||
*@param spi_rb : callback function to read byte usig SPI
|
|
||||||
*@param spi_wb : callback function to write byte usig SPI
|
|
||||||
*@todo Describe \ref wizchip_spi_readbyte and \ref wizchip_spi_writebyte function
|
|
||||||
*or register your functions.
|
|
||||||
*@note If you do not describe or register, null function is called.
|
|
||||||
*/
|
|
||||||
void reg_wizchip_spi_cbfunc(void (*spi_rb)(uint8_t *, uint32_t), void (*spi_wb)(const uint8_t *, uint32_t));
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Controls to the WIZCHIP.
|
|
||||||
* @details Resets WIZCHIP & internal PHY, Configures PHY mode, Monitor PHY(Link,Speed,Half/Full/Auto),
|
|
||||||
* controls interrupt & mask and so on.
|
|
||||||
* @param cwtype : Decides to the control type
|
|
||||||
* @param arg : arg type is dependent on cwtype.
|
|
||||||
* @return 0 : Success \n
|
|
||||||
* -1 : Fail because of invalid \ref ctlwizchip_type or unsupported \ref ctlwizchip_type in WIZCHIP
|
|
||||||
*/
|
|
||||||
int8_t ctlwizchip(ctlwizchip_type cwtype, void* arg);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Controls to network.
|
|
||||||
* @details Controls to network environment, mode, timeout and so on.
|
|
||||||
* @param cntype : Input. Decides to the control type
|
|
||||||
* @param arg : Inout. arg type is dependent on cntype.
|
|
||||||
* @return -1 : Fail because of invalid \ref ctlnetwork_type or unsupported \ref ctlnetwork_type in WIZCHIP \n
|
|
||||||
* 0 : Success
|
|
||||||
*/
|
|
||||||
int8_t ctlnetwork(ctlnetwork_type cntype, void* arg);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* The following functions are implemented for internal use.
|
|
||||||
* but You can call these functions for code size reduction instead of ctlwizchip() and ctlnetwork().
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Reset WIZCHIP by softly.
|
|
||||||
*/
|
|
||||||
void wizchip_sw_reset(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Initializes WIZCHIP with socket buffer size
|
|
||||||
* @param txsize Socket tx buffer sizes. If null, initialized the default size 2KB.
|
|
||||||
* @param rxsize Socket rx buffer sizes. If null, initialized the default size 2KB.
|
|
||||||
* @return 0 : succcess \n
|
|
||||||
* -1 : fail. Invalid buffer size
|
|
||||||
*/
|
|
||||||
int8_t wizchip_init(uint8_t* txsize, uint8_t* rxsize);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Clear Interrupt of WIZCHIP.
|
|
||||||
* @param intr : @ref intr_kind value operated OR. It can type-cast to uint16_t.
|
|
||||||
*/
|
|
||||||
void wizchip_clrinterrupt(intr_kind intr);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get Interrupt of WIZCHIP.
|
|
||||||
* @return @ref intr_kind value operated OR. It can type-cast to uint16_t.
|
|
||||||
*/
|
|
||||||
intr_kind wizchip_getinterrupt(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Mask or Unmask Interrupt of WIZCHIP.
|
|
||||||
* @param intr : @ref intr_kind value operated OR. It can type-cast to uint16_t.
|
|
||||||
*/
|
|
||||||
void wizchip_setinterruptmask(intr_kind intr);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get Interrupt mask of WIZCHIP.
|
|
||||||
* @return : The operated OR vaule of @ref intr_kind. It can type-cast to uint16_t.
|
|
||||||
*/
|
|
||||||
intr_kind wizchip_getinterruptmask(void);
|
|
||||||
|
|
||||||
#if _WIZCHIP_ > 5100
|
|
||||||
int8_t wizphy_getphylink(void); ///< get the link status of phy in WIZCHIP. No use in W5100
|
|
||||||
int8_t wizphy_getphypmode(void); ///< get the power mode of PHY in WIZCHIP. No use in W5100
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5500
|
|
||||||
void wizphy_reset(void); ///< Reset phy. Vailid only in W5500
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Set the phy information for WIZCHIP without power mode
|
|
||||||
* @param phyconf : @ref wiz_PhyConf
|
|
||||||
*/
|
|
||||||
void wizphy_setphyconf(wiz_PhyConf* phyconf);
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get phy configuration information.
|
|
||||||
* @param phyconf : @ref wiz_PhyConf
|
|
||||||
*/
|
|
||||||
void wizphy_getphyconf(wiz_PhyConf* phyconf);
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get phy status.
|
|
||||||
* @param phyconf : @ref wiz_PhyConf
|
|
||||||
*/
|
|
||||||
void wizphy_getphystat(wiz_PhyConf* phyconf);
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief set the power mode of phy inside WIZCHIP. Refer to @ref PHYCFGR in W5500, @ref PHYSTATUS in W5200
|
|
||||||
* @param pmode Settig value of power down mode.
|
|
||||||
*/
|
|
||||||
int8_t wizphy_setphypmode(uint8_t pmode);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Set the network information for WIZCHIP
|
|
||||||
* @param pnetinfo : @ref wizNetInfo
|
|
||||||
*/
|
|
||||||
void wizchip_setnetinfo(wiz_NetInfo* pnetinfo);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get the network information for WIZCHIP
|
|
||||||
* @param pnetinfo : @ref wizNetInfo
|
|
||||||
*/
|
|
||||||
void wizchip_getnetinfo(wiz_NetInfo* pnetinfo);
|
|
||||||
|
|
||||||
#if _WIZCHIP_ == 5200 // for W5200 ARP errata
|
|
||||||
uint8_t *wizchip_getsubn(void);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Set the network mode such WOL, PPPoE, Ping Block, and etc.
|
|
||||||
* @param pnetinfo Value of network mode. Refer to @ref netmode_type.
|
|
||||||
*/
|
|
||||||
int8_t wizchip_setnetmode(netmode_type netmode);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get the network mode such WOL, PPPoE, Ping Block, and etc.
|
|
||||||
* @return Value of network mode. Refer to @ref netmode_type.
|
|
||||||
*/
|
|
||||||
netmode_type wizchip_getnetmode(void);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Set retry time value(@ref RTR) and retry count(@ref RCR).
|
|
||||||
* @details @ref RTR configures the retransmission timeout period and @ref RCR configures the number of time of retransmission.
|
|
||||||
* @param nettime @ref RTR value and @ref RCR value. Refer to @ref wiz_NetTimeout.
|
|
||||||
*/
|
|
||||||
void wizchip_settimeout(wiz_NetTimeout* nettime);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @ingroup extra_functions
|
|
||||||
* @brief Get retry time value(@ref RTR) and retry count(@ref RCR).
|
|
||||||
* @details @ref RTR configures the retransmission timeout period and @ref RCR configures the number of time of retransmission.
|
|
||||||
* @param nettime @ref RTR value and @ref RCR value. Refer to @ref wiz_NetTimeout.
|
|
||||||
*/
|
|
||||||
void wizchip_gettimeout(wiz_NetTimeout* nettime);
|
|
||||||
|
|
||||||
#endif // _WIZCHIP_CONF_H_
|
|
@ -1,975 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file dhcp.c
|
|
||||||
//! \brief DHCP APIs implement file.
|
|
||||||
//! \details Processig DHCP protocol as DISCOVER, OFFER, REQUEST, ACK, NACK and DECLINE.
|
|
||||||
//! \version 1.1.0
|
|
||||||
//! \date 2013/11/18
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2018/10/09> Modified by Nick Moore for CircuitPython
|
|
||||||
//! <2013/11/18> 1st Release
|
|
||||||
//! <2012/12/20> V1.1.0
|
|
||||||
//! 1. Optimize code
|
|
||||||
//! 2. Add reg_dhcp_cbfunc()
|
|
||||||
//! 3. Add DHCP_stop()
|
|
||||||
//! 4. Integrate check_DHCP_state() & DHCP_run() to DHCP_run()
|
|
||||||
//! 5. Don't care system endian
|
|
||||||
//! 6. Add comments
|
|
||||||
//! <2012/12/26> V1.1.1
|
|
||||||
//! 1. Modify variable declaration: dhcp_tick_1s is declared volatile for code optimization
|
|
||||||
//! \author Eric Jung & MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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 "Ethernet/socket.h"
|
|
||||||
//#include "Internet/DHCP/dhcp.h"
|
|
||||||
#include "../../ethernet/socket.h"
|
|
||||||
#include "dhcp.h"
|
|
||||||
|
|
||||||
/* If you want to display debug & processing message, Define _DHCP_DEBUG_ in dhcp.h */
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
#include <stdio.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* DHCP state machine. */
|
|
||||||
#define STATE_DHCP_INIT 0 ///< Initialize
|
|
||||||
#define STATE_DHCP_DISCOVER 1 ///< send DISCOVER and wait OFFER
|
|
||||||
#define STATE_DHCP_REQUEST 2 ///< send REQEUST and wait ACK or NACK
|
|
||||||
#define STATE_DHCP_LEASED 3 ///< ReceiveD ACK and IP leased
|
|
||||||
#define STATE_DHCP_REREQUEST 4 ///< send REQUEST for maintaining leased IP
|
|
||||||
#define STATE_DHCP_RELEASE 5 ///< No use
|
|
||||||
#define STATE_DHCP_STOP 6 ///< Stop processing DHCP
|
|
||||||
|
|
||||||
#define DHCP_FLAGSBROADCAST 0x8000 ///< The broadcast value of flags in @ref RIP_MSG
|
|
||||||
#define DHCP_FLAGSUNICAST 0x0000 ///< The unicast value of flags in @ref RIP_MSG
|
|
||||||
|
|
||||||
/* DHCP message OP code */
|
|
||||||
#define DHCP_BOOTREQUEST 1 ///< Request Message used in op of @ref RIP_MSG
|
|
||||||
#define DHCP_BOOTREPLY 2 ///< Reply Message used i op of @ref RIP_MSG
|
|
||||||
|
|
||||||
/* DHCP message type */
|
|
||||||
#define DHCP_DISCOVER 1 ///< DISCOVER message in OPT of @ref RIP_MSG
|
|
||||||
#define DHCP_OFFER 2 ///< OFFER message in OPT of @ref RIP_MSG
|
|
||||||
#define DHCP_REQUEST 3 ///< REQUEST message in OPT of @ref RIP_MSG
|
|
||||||
#define DHCP_DECLINE 4 ///< DECLINE message in OPT of @ref RIP_MSG
|
|
||||||
#define DHCP_ACK 5 ///< ACK message in OPT of @ref RIP_MSG
|
|
||||||
#define DHCP_NAK 6 ///< NACK message in OPT of @ref RIP_MSG
|
|
||||||
#define DHCP_RELEASE 7 ///< RELEASE message in OPT of @ref RIP_MSG. No use
|
|
||||||
#define DHCP_INFORM 8 ///< INFORM message in OPT of @ref RIP_MSG. No use
|
|
||||||
|
|
||||||
#define DHCP_HTYPE10MB 1 ///< Used in type of @ref RIP_MSG
|
|
||||||
#define DHCP_HTYPE100MB 2 ///< Used in type of @ref RIP_MSG
|
|
||||||
|
|
||||||
#define DHCP_HLENETHERNET 6 ///< Used in hlen of @ref RIP_MSG
|
|
||||||
#define DHCP_HOPS 0 ///< Used in hops of @ref RIP_MSG
|
|
||||||
#define DHCP_SECS 0 ///< Used in secs of @ref RIP_MSG
|
|
||||||
|
|
||||||
#define INFINITE_LEASETIME 0xffffffff ///< Infinite lease time
|
|
||||||
|
|
||||||
#define OPT_SIZE 312 /// Max OPT size of @ref RIP_MSG
|
|
||||||
#define RIP_MSG_SIZE (236+OPT_SIZE) /// Max size of @ref RIP_MSG
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief DHCP option and value (cf. RFC1533)
|
|
||||||
*/
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
padOption = 0,
|
|
||||||
subnetMask = 1,
|
|
||||||
timerOffset = 2,
|
|
||||||
routersOnSubnet = 3,
|
|
||||||
timeServer = 4,
|
|
||||||
nameServer = 5,
|
|
||||||
dns = 6,
|
|
||||||
logServer = 7,
|
|
||||||
cookieServer = 8,
|
|
||||||
lprServer = 9,
|
|
||||||
impressServer = 10,
|
|
||||||
resourceLocationServer = 11,
|
|
||||||
hostName = 12,
|
|
||||||
bootFileSize = 13,
|
|
||||||
meritDumpFile = 14,
|
|
||||||
domainName = 15,
|
|
||||||
swapServer = 16,
|
|
||||||
rootPath = 17,
|
|
||||||
extentionsPath = 18,
|
|
||||||
IPforwarding = 19,
|
|
||||||
nonLocalSourceRouting = 20,
|
|
||||||
policyFilter = 21,
|
|
||||||
maxDgramReasmSize = 22,
|
|
||||||
defaultIPTTL = 23,
|
|
||||||
pathMTUagingTimeout = 24,
|
|
||||||
pathMTUplateauTable = 25,
|
|
||||||
ifMTU = 26,
|
|
||||||
allSubnetsLocal = 27,
|
|
||||||
broadcastAddr = 28,
|
|
||||||
performMaskDiscovery = 29,
|
|
||||||
maskSupplier = 30,
|
|
||||||
performRouterDiscovery = 31,
|
|
||||||
routerSolicitationAddr = 32,
|
|
||||||
staticRoute = 33,
|
|
||||||
trailerEncapsulation = 34,
|
|
||||||
arpCacheTimeout = 35,
|
|
||||||
ethernetEncapsulation = 36,
|
|
||||||
tcpDefaultTTL = 37,
|
|
||||||
tcpKeepaliveInterval = 38,
|
|
||||||
tcpKeepaliveGarbage = 39,
|
|
||||||
nisDomainName = 40,
|
|
||||||
nisServers = 41,
|
|
||||||
ntpServers = 42,
|
|
||||||
vendorSpecificInfo = 43,
|
|
||||||
netBIOSnameServer = 44,
|
|
||||||
netBIOSdgramDistServer = 45,
|
|
||||||
netBIOSnodeType = 46,
|
|
||||||
netBIOSscope = 47,
|
|
||||||
xFontServer = 48,
|
|
||||||
xDisplayManager = 49,
|
|
||||||
dhcpRequestedIPaddr = 50,
|
|
||||||
dhcpIPaddrLeaseTime = 51,
|
|
||||||
dhcpOptionOverload = 52,
|
|
||||||
dhcpMessageType = 53,
|
|
||||||
dhcpServerIdentifier = 54,
|
|
||||||
dhcpParamRequest = 55,
|
|
||||||
dhcpMsg = 56,
|
|
||||||
dhcpMaxMsgSize = 57,
|
|
||||||
dhcpT1value = 58,
|
|
||||||
dhcpT2value = 59,
|
|
||||||
dhcpClassIdentifier = 60,
|
|
||||||
dhcpClientIdentifier = 61,
|
|
||||||
endOption = 255
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief DHCP message format
|
|
||||||
*/
|
|
||||||
typedef struct {
|
|
||||||
uint8_t op; ///< @ref DHCP_BOOTREQUEST or @ref DHCP_BOOTREPLY
|
|
||||||
uint8_t htype; ///< @ref DHCP_HTYPE10MB or @ref DHCP_HTYPE100MB
|
|
||||||
uint8_t hlen; ///< @ref DHCP_HLENETHERNET
|
|
||||||
uint8_t hops; ///< @ref DHCP_HOPS
|
|
||||||
uint32_t xid; ///< @ref DHCP_XID This increase one every DHCP transaction.
|
|
||||||
uint16_t secs; ///< @ref DHCP_SECS
|
|
||||||
uint16_t flags; ///< @ref DHCP_FLAGSBROADCAST or @ref DHCP_FLAGSUNICAST
|
|
||||||
uint8_t ciaddr[4]; ///< @ref Request IP to DHCP sever
|
|
||||||
uint8_t yiaddr[4]; ///< @ref Offered IP from DHCP server
|
|
||||||
uint8_t siaddr[4]; ///< No use
|
|
||||||
uint8_t giaddr[4]; ///< No use
|
|
||||||
uint8_t chaddr[16]; ///< DHCP client 6bytes MAC address. Others is filled to zero
|
|
||||||
uint8_t sname[64]; ///< No use
|
|
||||||
uint8_t file[128]; ///< No use
|
|
||||||
uint8_t OPT[OPT_SIZE]; ///< Option
|
|
||||||
} RIP_MSG;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
uint8_t DHCP_SOCKET; // Socket number for DHCP
|
|
||||||
|
|
||||||
uint8_t DHCP_SIP[4]; // DHCP Server IP address
|
|
||||||
|
|
||||||
// Network information from DHCP Server
|
|
||||||
uint8_t OLD_allocated_ip[4] = {0, }; // Previous IP address
|
|
||||||
uint8_t DHCP_allocated_ip[4] = {0, }; // IP address from DHCP
|
|
||||||
uint8_t DHCP_allocated_gw[4] = {0, }; // Gateway address from DHCP
|
|
||||||
uint8_t DHCP_allocated_sn[4] = {0, }; // Subnet mask from DHCP
|
|
||||||
uint8_t DHCP_allocated_dns[4] = {0, }; // DNS address from DHCP
|
|
||||||
|
|
||||||
|
|
||||||
int8_t dhcp_state = STATE_DHCP_INIT; // DHCP state
|
|
||||||
int8_t dhcp_retry_count = 0;
|
|
||||||
|
|
||||||
uint32_t dhcp_lease_time = INFINITE_LEASETIME;
|
|
||||||
volatile uint32_t dhcp_tick_1s = 0; // unit 1 second
|
|
||||||
uint32_t dhcp_tick_next = DHCP_WAIT_TIME ;
|
|
||||||
|
|
||||||
uint32_t DHCP_XID; // Any number
|
|
||||||
|
|
||||||
RIP_MSG* pDHCPMSG; // Buffer pointer for DHCP processing
|
|
||||||
|
|
||||||
uint8_t HOST_NAME[] = DCHP_HOST_NAME;
|
|
||||||
|
|
||||||
uint8_t DHCP_CHADDR[6]; // DHCP Client MAC address.
|
|
||||||
|
|
||||||
/* The default callback function */
|
|
||||||
void default_ip_assign(void);
|
|
||||||
void default_ip_update(void);
|
|
||||||
void default_ip_conflict(void);
|
|
||||||
|
|
||||||
/* Callback handler */
|
|
||||||
void (*dhcp_ip_assign)(void) = default_ip_assign; /* handler to be called when the IP address from DHCP server is first assigned */
|
|
||||||
void (*dhcp_ip_update)(void) = default_ip_update; /* handler to be called when the IP address from DHCP server is updated */
|
|
||||||
void (*dhcp_ip_conflict)(void) = default_ip_conflict; /* handler to be called when the IP address from DHCP server is conflict */
|
|
||||||
|
|
||||||
void reg_dhcp_cbfunc(void(*ip_assign)(void), void(*ip_update)(void), void(*ip_conflict)(void));
|
|
||||||
|
|
||||||
|
|
||||||
/* send DISCOVER message to DHCP server */
|
|
||||||
void send_DHCP_DISCOVER(void);
|
|
||||||
|
|
||||||
/* send REQEUST message to DHCP server */
|
|
||||||
void send_DHCP_REQUEST(void);
|
|
||||||
|
|
||||||
/* send DECLINE message to DHCP server */
|
|
||||||
void send_DHCP_DECLINE(void);
|
|
||||||
|
|
||||||
/* IP conflict check by sending ARP-request to leased IP and wait ARP-response. */
|
|
||||||
int8_t check_DHCP_leasedIP(void);
|
|
||||||
|
|
||||||
/* check the timeout in DHCP process */
|
|
||||||
uint8_t check_DHCP_timeout(void);
|
|
||||||
|
|
||||||
/* Intialize to timeout process. */
|
|
||||||
void reset_DHCP_timeout(void);
|
|
||||||
|
|
||||||
/* Parse message as OFFER and ACK and NACK from DHCP server.*/
|
|
||||||
int8_t parseDHCPCMSG(void);
|
|
||||||
|
|
||||||
/* The default handler of ip assign first */
|
|
||||||
void default_ip_assign(void)
|
|
||||||
{
|
|
||||||
setSIPR(DHCP_allocated_ip);
|
|
||||||
setSUBR(DHCP_allocated_sn);
|
|
||||||
setGAR (DHCP_allocated_gw);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* The default handler of ip changed */
|
|
||||||
void default_ip_update(void)
|
|
||||||
{
|
|
||||||
/* WIZchip Software Reset */
|
|
||||||
setMR(MR_RST);
|
|
||||||
getMR(); // for delay
|
|
||||||
default_ip_assign();
|
|
||||||
setSHAR(DHCP_CHADDR);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* The default handler of ip changed */
|
|
||||||
void default_ip_conflict(void)
|
|
||||||
{
|
|
||||||
// WIZchip Software Reset
|
|
||||||
setMR(MR_RST);
|
|
||||||
getMR(); // for delay
|
|
||||||
setSHAR(DHCP_CHADDR);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* register the call back func. */
|
|
||||||
void reg_dhcp_cbfunc(void(*ip_assign)(void), void(*ip_update)(void), void(*ip_conflict)(void))
|
|
||||||
{
|
|
||||||
dhcp_ip_assign = default_ip_assign;
|
|
||||||
dhcp_ip_update = default_ip_update;
|
|
||||||
dhcp_ip_conflict = default_ip_conflict;
|
|
||||||
if(ip_assign) dhcp_ip_assign = ip_assign;
|
|
||||||
if(ip_update) dhcp_ip_update = ip_update;
|
|
||||||
if(ip_conflict) dhcp_ip_conflict = ip_conflict;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* make the common DHCP message */
|
|
||||||
void makeDHCPMSG(void)
|
|
||||||
{
|
|
||||||
uint8_t bk_mac[6];
|
|
||||||
uint8_t* ptmp;
|
|
||||||
uint8_t i;
|
|
||||||
getSHAR(bk_mac);
|
|
||||||
pDHCPMSG->op = DHCP_BOOTREQUEST;
|
|
||||||
pDHCPMSG->htype = DHCP_HTYPE10MB;
|
|
||||||
pDHCPMSG->hlen = DHCP_HLENETHERNET;
|
|
||||||
pDHCPMSG->hops = DHCP_HOPS;
|
|
||||||
ptmp = (uint8_t*)(&pDHCPMSG->xid);
|
|
||||||
*(ptmp+0) = (uint8_t)((DHCP_XID & 0xFF000000) >> 24);
|
|
||||||
*(ptmp+1) = (uint8_t)((DHCP_XID & 0x00FF0000) >> 16);
|
|
||||||
*(ptmp+2) = (uint8_t)((DHCP_XID & 0x0000FF00) >> 8);
|
|
||||||
*(ptmp+3) = (uint8_t)((DHCP_XID & 0x000000FF) >> 0);
|
|
||||||
pDHCPMSG->secs = DHCP_SECS;
|
|
||||||
ptmp = (uint8_t*)(&pDHCPMSG->flags);
|
|
||||||
*(ptmp+0) = (uint8_t)((DHCP_FLAGSBROADCAST & 0xFF00) >> 8);
|
|
||||||
*(ptmp+1) = (uint8_t)((DHCP_FLAGSBROADCAST & 0x00FF) >> 0);
|
|
||||||
|
|
||||||
pDHCPMSG->ciaddr[0] = 0;
|
|
||||||
pDHCPMSG->ciaddr[1] = 0;
|
|
||||||
pDHCPMSG->ciaddr[2] = 0;
|
|
||||||
pDHCPMSG->ciaddr[3] = 0;
|
|
||||||
|
|
||||||
pDHCPMSG->yiaddr[0] = 0;
|
|
||||||
pDHCPMSG->yiaddr[1] = 0;
|
|
||||||
pDHCPMSG->yiaddr[2] = 0;
|
|
||||||
pDHCPMSG->yiaddr[3] = 0;
|
|
||||||
|
|
||||||
pDHCPMSG->siaddr[0] = 0;
|
|
||||||
pDHCPMSG->siaddr[1] = 0;
|
|
||||||
pDHCPMSG->siaddr[2] = 0;
|
|
||||||
pDHCPMSG->siaddr[3] = 0;
|
|
||||||
|
|
||||||
pDHCPMSG->giaddr[0] = 0;
|
|
||||||
pDHCPMSG->giaddr[1] = 0;
|
|
||||||
pDHCPMSG->giaddr[2] = 0;
|
|
||||||
pDHCPMSG->giaddr[3] = 0;
|
|
||||||
|
|
||||||
pDHCPMSG->chaddr[0] = DHCP_CHADDR[0];
|
|
||||||
pDHCPMSG->chaddr[1] = DHCP_CHADDR[1];
|
|
||||||
pDHCPMSG->chaddr[2] = DHCP_CHADDR[2];
|
|
||||||
pDHCPMSG->chaddr[3] = DHCP_CHADDR[3];
|
|
||||||
pDHCPMSG->chaddr[4] = DHCP_CHADDR[4];
|
|
||||||
pDHCPMSG->chaddr[5] = DHCP_CHADDR[5];
|
|
||||||
|
|
||||||
for (i = 6; i < 16; i++) pDHCPMSG->chaddr[i] = 0;
|
|
||||||
for (i = 0; i < 64; i++) pDHCPMSG->sname[i] = 0;
|
|
||||||
for (i = 0; i < 128; i++) pDHCPMSG->file[i] = 0;
|
|
||||||
|
|
||||||
// MAGIC_COOKIE
|
|
||||||
pDHCPMSG->OPT[0] = (uint8_t)((MAGIC_COOKIE & 0xFF000000) >> 24);
|
|
||||||
pDHCPMSG->OPT[1] = (uint8_t)((MAGIC_COOKIE & 0x00FF0000) >> 16);
|
|
||||||
pDHCPMSG->OPT[2] = (uint8_t)((MAGIC_COOKIE & 0x0000FF00) >> 8);
|
|
||||||
pDHCPMSG->OPT[3] = (uint8_t) (MAGIC_COOKIE & 0x000000FF) >> 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* SEND DHCP DISCOVER */
|
|
||||||
void send_DHCP_DISCOVER(void)
|
|
||||||
{
|
|
||||||
uint16_t i;
|
|
||||||
uint8_t ip[4];
|
|
||||||
uint16_t k = 0;
|
|
||||||
|
|
||||||
makeDHCPMSG();
|
|
||||||
|
|
||||||
k = 4; // because MAGIC_COOKIE already made by makeDHCPMSG()
|
|
||||||
|
|
||||||
// Option Request Param
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpMessageType;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x01;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_DISCOVER;
|
|
||||||
|
|
||||||
// Client identifier
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpClientIdentifier;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x07;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x01;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[3];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[4];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[5];
|
|
||||||
|
|
||||||
// host name
|
|
||||||
pDHCPMSG->OPT[k++] = hostName;
|
|
||||||
pDHCPMSG->OPT[k++] = 0; // fill zero length of hostname
|
|
||||||
for(i = 0 ; HOST_NAME[i] != 0; i++)
|
|
||||||
pDHCPMSG->OPT[k++] = HOST_NAME[i];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[3];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[4];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[5];
|
|
||||||
pDHCPMSG->OPT[k - (i+3+1)] = i+3; // length of hostname
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpParamRequest;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x06; // length of request
|
|
||||||
pDHCPMSG->OPT[k++] = subnetMask;
|
|
||||||
pDHCPMSG->OPT[k++] = routersOnSubnet;
|
|
||||||
pDHCPMSG->OPT[k++] = dns;
|
|
||||||
pDHCPMSG->OPT[k++] = domainName;
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpT1value;
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpT2value;
|
|
||||||
pDHCPMSG->OPT[k++] = endOption;
|
|
||||||
|
|
||||||
for (i = k; i < OPT_SIZE; i++) pDHCPMSG->OPT[i] = 0;
|
|
||||||
|
|
||||||
// send broadcasting packet
|
|
||||||
ip[0] = 255;
|
|
||||||
ip[1] = 255;
|
|
||||||
ip[2] = 255;
|
|
||||||
ip[3] = 255;
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Send DHCP_DISCOVER\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
WIZCHIP_EXPORT(sendto)(DHCP_SOCKET, (uint8_t *)pDHCPMSG, RIP_MSG_SIZE, ip, DHCP_SERVER_PORT);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* SEND DHCP REQUEST */
|
|
||||||
void send_DHCP_REQUEST(void)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
uint8_t ip[4];
|
|
||||||
uint16_t k = 0;
|
|
||||||
|
|
||||||
makeDHCPMSG();
|
|
||||||
|
|
||||||
if(dhcp_state == STATE_DHCP_LEASED || dhcp_state == STATE_DHCP_REREQUEST)
|
|
||||||
{
|
|
||||||
*((uint8_t*)(&pDHCPMSG->flags)) = ((DHCP_FLAGSUNICAST & 0xFF00)>> 8);
|
|
||||||
*((uint8_t*)(&pDHCPMSG->flags)+1) = (DHCP_FLAGSUNICAST & 0x00FF);
|
|
||||||
pDHCPMSG->ciaddr[0] = DHCP_allocated_ip[0];
|
|
||||||
pDHCPMSG->ciaddr[1] = DHCP_allocated_ip[1];
|
|
||||||
pDHCPMSG->ciaddr[2] = DHCP_allocated_ip[2];
|
|
||||||
pDHCPMSG->ciaddr[3] = DHCP_allocated_ip[3];
|
|
||||||
ip[0] = DHCP_SIP[0];
|
|
||||||
ip[1] = DHCP_SIP[1];
|
|
||||||
ip[2] = DHCP_SIP[2];
|
|
||||||
ip[3] = DHCP_SIP[3];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ip[0] = 255;
|
|
||||||
ip[1] = 255;
|
|
||||||
ip[2] = 255;
|
|
||||||
ip[3] = 255;
|
|
||||||
}
|
|
||||||
|
|
||||||
k = 4; // because MAGIC_COOKIE already made by makeDHCPMSG()
|
|
||||||
|
|
||||||
// Option Request Param.
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpMessageType;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x01;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_REQUEST;
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpClientIdentifier;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x07;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x01;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[3];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[4];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[5];
|
|
||||||
|
|
||||||
if(ip[3] == 255) // if(dchp_state == STATE_DHCP_LEASED || dchp_state == DHCP_REREQUEST_STATE)
|
|
||||||
{
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpRequestedIPaddr;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x04;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[3];
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpServerIdentifier;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x04;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[3];
|
|
||||||
}
|
|
||||||
|
|
||||||
// host name
|
|
||||||
pDHCPMSG->OPT[k++] = hostName;
|
|
||||||
pDHCPMSG->OPT[k++] = 0; // length of hostname
|
|
||||||
for(i = 0 ; HOST_NAME[i] != 0; i++)
|
|
||||||
pDHCPMSG->OPT[k++] = HOST_NAME[i];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[3];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[4];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[5];
|
|
||||||
pDHCPMSG->OPT[k - (i+3+1)] = i+3; // length of hostname
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpParamRequest;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x08;
|
|
||||||
pDHCPMSG->OPT[k++] = subnetMask;
|
|
||||||
pDHCPMSG->OPT[k++] = routersOnSubnet;
|
|
||||||
pDHCPMSG->OPT[k++] = dns;
|
|
||||||
pDHCPMSG->OPT[k++] = domainName;
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpT1value;
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpT2value;
|
|
||||||
pDHCPMSG->OPT[k++] = performRouterDiscovery;
|
|
||||||
pDHCPMSG->OPT[k++] = staticRoute;
|
|
||||||
pDHCPMSG->OPT[k++] = endOption;
|
|
||||||
|
|
||||||
for (i = k; i < OPT_SIZE; i++) pDHCPMSG->OPT[i] = 0;
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Send DHCP_REQUEST\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
WIZCHIP_EXPORT(sendto)(DHCP_SOCKET, (uint8_t *)pDHCPMSG, RIP_MSG_SIZE, ip, DHCP_SERVER_PORT);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/* SEND DHCP DHCPDECLINE */
|
|
||||||
void send_DHCP_DECLINE(void)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
uint8_t ip[4];
|
|
||||||
uint16_t k = 0;
|
|
||||||
|
|
||||||
makeDHCPMSG();
|
|
||||||
|
|
||||||
k = 4; // because MAGIC_COOKIE already made by makeDHCPMSG()
|
|
||||||
|
|
||||||
*((uint8_t*)(&pDHCPMSG->flags)) = ((DHCP_FLAGSUNICAST & 0xFF00)>> 8);
|
|
||||||
*((uint8_t*)(&pDHCPMSG->flags)+1) = (DHCP_FLAGSUNICAST & 0x00FF);
|
|
||||||
|
|
||||||
// Option Request Param.
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpMessageType;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x01;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_DECLINE;
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpClientIdentifier;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x07;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x01;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[3];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[4];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_CHADDR[5];
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpRequestedIPaddr;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x04;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_allocated_ip[3];
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = dhcpServerIdentifier;
|
|
||||||
pDHCPMSG->OPT[k++] = 0x04;
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[0];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[1];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[2];
|
|
||||||
pDHCPMSG->OPT[k++] = DHCP_SIP[3];
|
|
||||||
|
|
||||||
pDHCPMSG->OPT[k++] = endOption;
|
|
||||||
|
|
||||||
for (i = k; i < OPT_SIZE; i++) pDHCPMSG->OPT[i] = 0;
|
|
||||||
|
|
||||||
//send broadcasting packet
|
|
||||||
ip[0] = 0xFF;
|
|
||||||
ip[1] = 0xFF;
|
|
||||||
ip[2] = 0xFF;
|
|
||||||
ip[3] = 0xFF;
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("\r\n> Send DHCP_DECLINE\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
WIZCHIP_EXPORT(sendto)(DHCP_SOCKET, (uint8_t *)pDHCPMSG, RIP_MSG_SIZE, ip, DHCP_SERVER_PORT);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* PARSE REPLY pDHCPMSG */
|
|
||||||
int8_t parseDHCPMSG(void)
|
|
||||||
{
|
|
||||||
uint8_t svr_addr[6];
|
|
||||||
uint16_t svr_port;
|
|
||||||
uint16_t len;
|
|
||||||
|
|
||||||
uint8_t * p;
|
|
||||||
uint8_t * e;
|
|
||||||
uint8_t type = 0;
|
|
||||||
uint8_t opt_len;
|
|
||||||
|
|
||||||
if((len = getSn_RX_RSR(DHCP_SOCKET)) > 0)
|
|
||||||
{
|
|
||||||
len = WIZCHIP_EXPORT(recvfrom)(DHCP_SOCKET, (uint8_t *)pDHCPMSG, len, svr_addr, &svr_port);
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("DHCP message : %d.%d.%d.%d(%d) %d received. \r\n",svr_addr[0],svr_addr[1],svr_addr[2], svr_addr[3],svr_port, len);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
else return 0;
|
|
||||||
if (svr_port == DHCP_SERVER_PORT) {
|
|
||||||
// compare mac address
|
|
||||||
if ( (pDHCPMSG->chaddr[0] != DHCP_CHADDR[0]) || (pDHCPMSG->chaddr[1] != DHCP_CHADDR[1]) ||
|
|
||||||
(pDHCPMSG->chaddr[2] != DHCP_CHADDR[2]) || (pDHCPMSG->chaddr[3] != DHCP_CHADDR[3]) ||
|
|
||||||
(pDHCPMSG->chaddr[4] != DHCP_CHADDR[4]) || (pDHCPMSG->chaddr[5] != DHCP_CHADDR[5]) )
|
|
||||||
return 0;
|
|
||||||
type = 0;
|
|
||||||
p = (uint8_t *)(&pDHCPMSG->op);
|
|
||||||
p = p + 240; // 240 = sizeof(RIP_MSG) + MAGIC_COOKIE size in RIP_MSG.opt - sizeof(RIP_MSG.opt)
|
|
||||||
e = p + (len - 240);
|
|
||||||
|
|
||||||
while ( p < e ) {
|
|
||||||
|
|
||||||
switch ( *p ) {
|
|
||||||
|
|
||||||
case endOption :
|
|
||||||
p = e; // for break while(p < e)
|
|
||||||
break;
|
|
||||||
case padOption :
|
|
||||||
p++;
|
|
||||||
break;
|
|
||||||
case dhcpMessageType :
|
|
||||||
p++;
|
|
||||||
p++;
|
|
||||||
type = *p++;
|
|
||||||
break;
|
|
||||||
case subnetMask :
|
|
||||||
p++;
|
|
||||||
p++;
|
|
||||||
DHCP_allocated_sn[0] = *p++;
|
|
||||||
DHCP_allocated_sn[1] = *p++;
|
|
||||||
DHCP_allocated_sn[2] = *p++;
|
|
||||||
DHCP_allocated_sn[3] = *p++;
|
|
||||||
break;
|
|
||||||
case routersOnSubnet :
|
|
||||||
p++;
|
|
||||||
opt_len = *p++;
|
|
||||||
DHCP_allocated_gw[0] = *p++;
|
|
||||||
DHCP_allocated_gw[1] = *p++;
|
|
||||||
DHCP_allocated_gw[2] = *p++;
|
|
||||||
DHCP_allocated_gw[3] = *p++;
|
|
||||||
p = p + (opt_len - 4);
|
|
||||||
break;
|
|
||||||
case dns :
|
|
||||||
p++;
|
|
||||||
opt_len = *p++;
|
|
||||||
DHCP_allocated_dns[0] = *p++;
|
|
||||||
DHCP_allocated_dns[1] = *p++;
|
|
||||||
DHCP_allocated_dns[2] = *p++;
|
|
||||||
DHCP_allocated_dns[3] = *p++;
|
|
||||||
p = p + (opt_len - 4);
|
|
||||||
break;
|
|
||||||
case dhcpIPaddrLeaseTime :
|
|
||||||
p++;
|
|
||||||
opt_len = *p++;
|
|
||||||
dhcp_lease_time = *p++;
|
|
||||||
dhcp_lease_time = (dhcp_lease_time << 8) + *p++;
|
|
||||||
dhcp_lease_time = (dhcp_lease_time << 8) + *p++;
|
|
||||||
dhcp_lease_time = (dhcp_lease_time << 8) + *p++;
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
dhcp_lease_time = 10;
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
case dhcpServerIdentifier :
|
|
||||||
p++;
|
|
||||||
opt_len = *p++;
|
|
||||||
DHCP_SIP[0] = *p++;
|
|
||||||
DHCP_SIP[1] = *p++;
|
|
||||||
DHCP_SIP[2] = *p++;
|
|
||||||
DHCP_SIP[3] = *p++;
|
|
||||||
break;
|
|
||||||
default :
|
|
||||||
p++;
|
|
||||||
opt_len = *p++;
|
|
||||||
p += opt_len;
|
|
||||||
break;
|
|
||||||
} // switch
|
|
||||||
} // while
|
|
||||||
} // if
|
|
||||||
return type;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t DHCP_run(void)
|
|
||||||
{
|
|
||||||
uint8_t type;
|
|
||||||
uint8_t ret;
|
|
||||||
|
|
||||||
if(dhcp_state == STATE_DHCP_STOP) return DHCP_STOPPED;
|
|
||||||
|
|
||||||
if(getSn_SR(DHCP_SOCKET) != SOCK_UDP)
|
|
||||||
WIZCHIP_EXPORT(socket)(DHCP_SOCKET, Sn_MR_UDP, DHCP_CLIENT_PORT, 0x00);
|
|
||||||
|
|
||||||
ret = DHCP_RUNNING;
|
|
||||||
type = parseDHCPMSG();
|
|
||||||
|
|
||||||
switch ( dhcp_state ) {
|
|
||||||
case STATE_DHCP_INIT :
|
|
||||||
DHCP_allocated_ip[0] = 0;
|
|
||||||
DHCP_allocated_ip[1] = 0;
|
|
||||||
DHCP_allocated_ip[2] = 0;
|
|
||||||
DHCP_allocated_ip[3] = 0;
|
|
||||||
send_DHCP_DISCOVER();
|
|
||||||
dhcp_state = STATE_DHCP_DISCOVER;
|
|
||||||
break;
|
|
||||||
case STATE_DHCP_DISCOVER :
|
|
||||||
if (type == DHCP_OFFER){
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Receive DHCP_OFFER\r\n");
|
|
||||||
#endif
|
|
||||||
DHCP_allocated_ip[0] = pDHCPMSG->yiaddr[0];
|
|
||||||
DHCP_allocated_ip[1] = pDHCPMSG->yiaddr[1];
|
|
||||||
DHCP_allocated_ip[2] = pDHCPMSG->yiaddr[2];
|
|
||||||
DHCP_allocated_ip[3] = pDHCPMSG->yiaddr[3];
|
|
||||||
|
|
||||||
send_DHCP_REQUEST();
|
|
||||||
dhcp_state = STATE_DHCP_REQUEST;
|
|
||||||
} else ret = check_DHCP_timeout();
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STATE_DHCP_REQUEST :
|
|
||||||
if (type == DHCP_ACK) {
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Receive DHCP_ACK\r\n");
|
|
||||||
#endif
|
|
||||||
if (check_DHCP_leasedIP()) {
|
|
||||||
// Network info assignment from DHCP
|
|
||||||
dhcp_ip_assign();
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
|
|
||||||
dhcp_state = STATE_DHCP_LEASED;
|
|
||||||
} else {
|
|
||||||
// IP address conflict occurred
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
dhcp_ip_conflict();
|
|
||||||
dhcp_state = STATE_DHCP_INIT;
|
|
||||||
}
|
|
||||||
} else if (type == DHCP_NAK) {
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Receive DHCP_NACK\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
|
|
||||||
dhcp_state = STATE_DHCP_DISCOVER;
|
|
||||||
} else ret = check_DHCP_timeout();
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STATE_DHCP_LEASED :
|
|
||||||
ret = DHCP_IP_LEASED;
|
|
||||||
if ((dhcp_lease_time != INFINITE_LEASETIME) && ((dhcp_lease_time/2) < dhcp_tick_1s)) {
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Maintains the IP address \r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
type = 0;
|
|
||||||
OLD_allocated_ip[0] = DHCP_allocated_ip[0];
|
|
||||||
OLD_allocated_ip[1] = DHCP_allocated_ip[1];
|
|
||||||
OLD_allocated_ip[2] = DHCP_allocated_ip[2];
|
|
||||||
OLD_allocated_ip[3] = DHCP_allocated_ip[3];
|
|
||||||
|
|
||||||
DHCP_XID++;
|
|
||||||
|
|
||||||
send_DHCP_REQUEST();
|
|
||||||
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
|
|
||||||
dhcp_state = STATE_DHCP_REREQUEST;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STATE_DHCP_REREQUEST :
|
|
||||||
ret = DHCP_IP_LEASED;
|
|
||||||
if (type == DHCP_ACK) {
|
|
||||||
dhcp_retry_count = 0;
|
|
||||||
if (OLD_allocated_ip[0] != DHCP_allocated_ip[0] ||
|
|
||||||
OLD_allocated_ip[1] != DHCP_allocated_ip[1] ||
|
|
||||||
OLD_allocated_ip[2] != DHCP_allocated_ip[2] ||
|
|
||||||
OLD_allocated_ip[3] != DHCP_allocated_ip[3])
|
|
||||||
{
|
|
||||||
ret = DHCP_IP_CHANGED;
|
|
||||||
dhcp_ip_update();
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf(">IP changed.\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
}
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
else printf(">IP is continued.\r\n");
|
|
||||||
#endif
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
dhcp_state = STATE_DHCP_LEASED;
|
|
||||||
} else if (type == DHCP_NAK) {
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("> Receive DHCP_NACK, Failed to maintain ip\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
|
|
||||||
dhcp_state = STATE_DHCP_DISCOVER;
|
|
||||||
} else ret = check_DHCP_timeout();
|
|
||||||
break;
|
|
||||||
default :
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DHCP_stop(void)
|
|
||||||
{
|
|
||||||
WIZCHIP_EXPORT(close)(DHCP_SOCKET);
|
|
||||||
dhcp_state = STATE_DHCP_STOP;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t check_DHCP_timeout(void)
|
|
||||||
{
|
|
||||||
uint8_t ret = DHCP_RUNNING;
|
|
||||||
|
|
||||||
if (dhcp_retry_count < MAX_DHCP_RETRY) {
|
|
||||||
if (dhcp_tick_next < dhcp_tick_1s) {
|
|
||||||
|
|
||||||
switch ( dhcp_state ) {
|
|
||||||
case STATE_DHCP_DISCOVER :
|
|
||||||
// printf("<<timeout>> state : STATE_DHCP_DISCOVER\r\n");
|
|
||||||
send_DHCP_DISCOVER();
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STATE_DHCP_REQUEST :
|
|
||||||
// printf("<<timeout>> state : STATE_DHCP_REQUEST\r\n");
|
|
||||||
|
|
||||||
send_DHCP_REQUEST();
|
|
||||||
break;
|
|
||||||
|
|
||||||
case STATE_DHCP_REREQUEST :
|
|
||||||
// printf("<<timeout>> state : STATE_DHCP_REREQUEST\r\n");
|
|
||||||
|
|
||||||
send_DHCP_REQUEST();
|
|
||||||
break;
|
|
||||||
|
|
||||||
default :
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
dhcp_tick_1s = 0;
|
|
||||||
dhcp_tick_next = dhcp_tick_1s + DHCP_WAIT_TIME;
|
|
||||||
dhcp_retry_count++;
|
|
||||||
}
|
|
||||||
} else { // timeout occurred
|
|
||||||
|
|
||||||
switch(dhcp_state) {
|
|
||||||
case STATE_DHCP_DISCOVER:
|
|
||||||
dhcp_state = STATE_DHCP_INIT;
|
|
||||||
ret = DHCP_FAILED;
|
|
||||||
break;
|
|
||||||
case STATE_DHCP_REQUEST:
|
|
||||||
case STATE_DHCP_REREQUEST:
|
|
||||||
send_DHCP_DISCOVER();
|
|
||||||
dhcp_state = STATE_DHCP_DISCOVER;
|
|
||||||
break;
|
|
||||||
default :
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int8_t check_DHCP_leasedIP(void)
|
|
||||||
{
|
|
||||||
uint8_t tmp;
|
|
||||||
int32_t ret;
|
|
||||||
|
|
||||||
//WIZchip RCR value changed for ARP Timeout count control
|
|
||||||
tmp = getRCR();
|
|
||||||
setRCR(0x03);
|
|
||||||
|
|
||||||
// IP conflict detection : ARP request - ARP reply
|
|
||||||
// Broadcasting ARP Request for check the IP conflict using UDP sendto() function
|
|
||||||
ret = WIZCHIP_EXPORT(sendto)(DHCP_SOCKET, (uint8_t *)"CHECK_IP_CONFLICT", 17, DHCP_allocated_ip, 5000);
|
|
||||||
|
|
||||||
// RCR value restore
|
|
||||||
setRCR(tmp);
|
|
||||||
|
|
||||||
if(ret == SOCKERR_TIMEOUT) {
|
|
||||||
// UDP send Timeout occurred : allocated IP address is unique, DHCP Success
|
|
||||||
|
|
||||||
#ifdef _DHCP_DEBUG_
|
|
||||||
printf("\r\n> Check leased IP - OK\r\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
} else {
|
|
||||||
// Received ARP reply or etc : IP address conflict occur, DHCP Failed
|
|
||||||
send_DHCP_DECLINE();
|
|
||||||
|
|
||||||
ret = dhcp_tick_1s;
|
|
||||||
while((dhcp_tick_1s - ret) < 2) ; // wait for 1s over; wait to complete to send DECLINE message;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void DHCP_init(uint8_t s, DHCP_INIT_BUFFER_TYPE* buf)
|
|
||||||
{
|
|
||||||
uint8_t zeroip[4] = {0,0,0,0};
|
|
||||||
getSHAR(DHCP_CHADDR);
|
|
||||||
if((DHCP_CHADDR[0] | DHCP_CHADDR[1] | DHCP_CHADDR[2] | DHCP_CHADDR[3] | DHCP_CHADDR[4] | DHCP_CHADDR[5]) == 0x00)
|
|
||||||
{
|
|
||||||
// assign temporary mac address, you should be set SHAR before call this function.
|
|
||||||
DHCP_CHADDR[0] = 0x00;
|
|
||||||
DHCP_CHADDR[1] = 0x08;
|
|
||||||
DHCP_CHADDR[2] = 0xdc;
|
|
||||||
DHCP_CHADDR[3] = 0x00;
|
|
||||||
DHCP_CHADDR[4] = 0x00;
|
|
||||||
DHCP_CHADDR[5] = 0x00;
|
|
||||||
setSHAR(DHCP_CHADDR);
|
|
||||||
}
|
|
||||||
|
|
||||||
DHCP_SOCKET = s; // SOCK_DHCP
|
|
||||||
pDHCPMSG = (RIP_MSG*)buf;
|
|
||||||
DHCP_XID = 0x12345678;
|
|
||||||
|
|
||||||
// WIZchip Netinfo Clear
|
|
||||||
setSIPR(zeroip);
|
|
||||||
setSIPR(zeroip);
|
|
||||||
setGAR(zeroip);
|
|
||||||
|
|
||||||
reset_DHCP_timeout();
|
|
||||||
dhcp_state = STATE_DHCP_INIT;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Rset the DHCP timeout count and retry count. */
|
|
||||||
void reset_DHCP_timeout(void)
|
|
||||||
{
|
|
||||||
dhcp_tick_1s = 0;
|
|
||||||
dhcp_tick_next = DHCP_WAIT_TIME;
|
|
||||||
dhcp_retry_count = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DHCP_time_handler(void)
|
|
||||||
{
|
|
||||||
dhcp_tick_1s++;
|
|
||||||
}
|
|
||||||
|
|
||||||
void getIPfromDHCP(uint8_t* ip)
|
|
||||||
{
|
|
||||||
ip[0] = DHCP_allocated_ip[0];
|
|
||||||
ip[1] = DHCP_allocated_ip[1];
|
|
||||||
ip[2] = DHCP_allocated_ip[2];
|
|
||||||
ip[3] = DHCP_allocated_ip[3];
|
|
||||||
}
|
|
||||||
|
|
||||||
void getGWfromDHCP(uint8_t* ip)
|
|
||||||
{
|
|
||||||
ip[0] =DHCP_allocated_gw[0];
|
|
||||||
ip[1] =DHCP_allocated_gw[1];
|
|
||||||
ip[2] =DHCP_allocated_gw[2];
|
|
||||||
ip[3] =DHCP_allocated_gw[3];
|
|
||||||
}
|
|
||||||
|
|
||||||
void getSNfromDHCP(uint8_t* ip)
|
|
||||||
{
|
|
||||||
ip[0] = DHCP_allocated_sn[0];
|
|
||||||
ip[1] = DHCP_allocated_sn[1];
|
|
||||||
ip[2] = DHCP_allocated_sn[2];
|
|
||||||
ip[3] = DHCP_allocated_sn[3];
|
|
||||||
}
|
|
||||||
|
|
||||||
void getDNSfromDHCP(uint8_t* ip)
|
|
||||||
{
|
|
||||||
ip[0] = DHCP_allocated_dns[0];
|
|
||||||
ip[1] = DHCP_allocated_dns[1];
|
|
||||||
ip[2] = DHCP_allocated_dns[2];
|
|
||||||
ip[3] = DHCP_allocated_dns[3];
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t getDHCPLeasetime(void)
|
|
||||||
{
|
|
||||||
return dhcp_lease_time;
|
|
||||||
}
|
|
@ -1,152 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file dhcp.h
|
|
||||||
//! \brief DHCP APIs Header file.
|
|
||||||
//! \details Processig DHCP protocol as DISCOVER, OFFER, REQUEST, ACK, NACK and DECLINE.
|
|
||||||
//! \version 1.1.0
|
|
||||||
//! \date 2013/11/18
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2013/11/18> 1st Release
|
|
||||||
//! <2012/12/20> V1.1.0
|
|
||||||
//! 1. Move unreferenced DEFINE to dhcp.c
|
|
||||||
//! <2012/12/26> V1.1.1
|
|
||||||
//! \author Eric Jung & MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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.
|
|
||||||
//
|
|
||||||
//*****************************************************************************
|
|
||||||
#ifndef _DHCP_H_
|
|
||||||
#define _DHCP_H_
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief
|
|
||||||
* @details If you want to display debug & processing message, Define _DHCP_DEBUG_
|
|
||||||
* @note If defined, it depends on <stdio.h>
|
|
||||||
*/
|
|
||||||
|
|
||||||
//#define _DHCP_DEBUG_
|
|
||||||
|
|
||||||
/* Retry to processing DHCP */
|
|
||||||
#define MAX_DHCP_RETRY 2 ///< Maximum retry count
|
|
||||||
#define DHCP_WAIT_TIME 3 ///< Wait Time 3s (was 10s)
|
|
||||||
|
|
||||||
/* UDP port numbers for DHCP */
|
|
||||||
#define DHCP_SERVER_PORT 67 ///< DHCP server port number
|
|
||||||
#define DHCP_CLIENT_PORT 68 ///< DHCP client port number
|
|
||||||
|
|
||||||
#define MAGIC_COOKIE 0x63825363 ///< Any number. You can be modified it any number
|
|
||||||
|
|
||||||
#define DCHP_HOST_NAME "WIZnet\0"
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief return value of @ref DHCP_run()
|
|
||||||
*/
|
|
||||||
enum
|
|
||||||
{
|
|
||||||
DHCP_FAILED = 0, ///< Processing Fail
|
|
||||||
DHCP_RUNNING, ///< Processing DHCP protocol
|
|
||||||
DHCP_IP_ASSIGN, ///< First Occupy IP from DHPC server (if cbfunc == null, act as default default_ip_assign)
|
|
||||||
DHCP_IP_CHANGED, ///< Change IP address by new IP address from DHCP (if cbfunc == null, act as default default_ip_update)
|
|
||||||
DHCP_IP_LEASED, ///< Stand by
|
|
||||||
DHCP_STOPPED ///< Stop processing DHCP protocol
|
|
||||||
};
|
|
||||||
|
|
||||||
#define DHCP_INIT_BUFFER_TYPE uint32_t
|
|
||||||
#define DHCP_INIT_BUFFER_SIZE (137)
|
|
||||||
/*
|
|
||||||
* @brief DHCP client initialization (outside of the main loop)
|
|
||||||
* @param s - socket number
|
|
||||||
* @param buf - buffer for processing DHCP message
|
|
||||||
*/
|
|
||||||
void DHCP_init(uint8_t s, DHCP_INIT_BUFFER_TYPE* buf);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief DHCP 1s Tick Timer handler
|
|
||||||
* @note SHOULD BE register to your system 1s Tick timer handler
|
|
||||||
*/
|
|
||||||
void DHCP_time_handler(void);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief Register call back function
|
|
||||||
* @param ip_assign - callback func when IP is assigned from DHCP server first
|
|
||||||
* @param ip_update - callback func when IP is changed
|
|
||||||
* @prarm ip_conflict - callback func when the assigned IP is conflict with others.
|
|
||||||
*/
|
|
||||||
void reg_dhcp_cbfunc(void(*ip_assign)(void), void(*ip_update)(void), void(*ip_conflict)(void));
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief DHCP client in the main loop
|
|
||||||
* @return The value is as the follow \n
|
|
||||||
* @ref DHCP_FAILED \n
|
|
||||||
* @ref DHCP_RUNNING \n
|
|
||||||
* @ref DHCP_IP_ASSIGN \n
|
|
||||||
* @ref DHCP_IP_CHANGED \n
|
|
||||||
* @ref DHCP_IP_LEASED \n
|
|
||||||
* @ref DHCP_STOPPED \n
|
|
||||||
*
|
|
||||||
* @note This function is always called by you main task.
|
|
||||||
*/
|
|
||||||
uint8_t DHCP_run(void);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief Stop DHCP processing
|
|
||||||
* @note If you want to restart. call DHCP_init() and DHCP_run()
|
|
||||||
*/
|
|
||||||
void DHCP_stop(void);
|
|
||||||
|
|
||||||
/* Get Network information assigned from DHCP server */
|
|
||||||
/*
|
|
||||||
* @brief Get IP address
|
|
||||||
* @param ip - IP address to be returned
|
|
||||||
*/
|
|
||||||
void getIPfromDHCP(uint8_t* ip);
|
|
||||||
/*
|
|
||||||
* @brief Get Gateway address
|
|
||||||
* @param ip - Gateway address to be returned
|
|
||||||
*/
|
|
||||||
void getGWfromDHCP(uint8_t* ip);
|
|
||||||
/*
|
|
||||||
* @brief Get Subnet mask value
|
|
||||||
* @param ip - Subnet mask to be returned
|
|
||||||
*/
|
|
||||||
void getSNfromDHCP(uint8_t* ip);
|
|
||||||
/*
|
|
||||||
* @brief Get DNS address
|
|
||||||
* @param ip - DNS address to be returned
|
|
||||||
*/
|
|
||||||
void getDNSfromDHCP(uint8_t* ip);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief Get the leased time by DHCP sever
|
|
||||||
* @return unit 1s
|
|
||||||
*/
|
|
||||||
uint32_t getDHCPLeasetime(void);
|
|
||||||
|
|
||||||
#endif /* _DHCP_H_ */
|
|
@ -1,572 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file dns.c
|
|
||||||
//! \brief DNS APIs Implement file.
|
|
||||||
//! \details Send DNS query & Receive DNS reponse. \n
|
|
||||||
//! It depends on stdlib.h & string.h in ansi-c library
|
|
||||||
//! \version 1.1.0
|
|
||||||
//! \date 2013/11/18
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! <2013/12/20> V1.1.0
|
|
||||||
//! 1. Remove secondary DNS server in DNS_run
|
|
||||||
//! If 1st DNS_run failed, call DNS_run with 2nd DNS again
|
|
||||||
//! 2. DNS_timerHandler -> DNS_time_handler
|
|
||||||
//! 3. Remove the unused define
|
|
||||||
//! 4. Integrated dns.h dns.c & dns_parse.h dns_parse.c into dns.h & dns.c
|
|
||||||
//! <2013/12/20> V1.1.0
|
|
||||||
//! <2018/10/04> Modified HAL_GetTick for use with CircuitPython by Nick Moore
|
|
||||||
//!
|
|
||||||
//! \author Eric Jung & MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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 <string.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include "supervisor/shared/tick.h"
|
|
||||||
|
|
||||||
//#include "Ethernet/socket.h"
|
|
||||||
//#include "Internet/DNS/dns.h"
|
|
||||||
#include "../../ethernet/socket.h"
|
|
||||||
#include "dns.h"
|
|
||||||
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
#include <stdio.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define INITRTT 2000L /* Initial smoothed response time */
|
|
||||||
#define MAXCNAME (MAX_DOMAIN_NAME + (MAX_DOMAIN_NAME>>1)) /* Maximum amount of cname recursion */
|
|
||||||
|
|
||||||
#define TYPE_A 1 /* Host address */
|
|
||||||
#define TYPE_NS 2 /* Name server */
|
|
||||||
#define TYPE_MD 3 /* Mail destination (obsolete) */
|
|
||||||
#define TYPE_MF 4 /* Mail forwarder (obsolete) */
|
|
||||||
#define TYPE_CNAME 5 /* Canonical name */
|
|
||||||
#define TYPE_SOA 6 /* Start of Authority */
|
|
||||||
#define TYPE_MB 7 /* Mailbox name (experimental) */
|
|
||||||
#define TYPE_MG 8 /* Mail group member (experimental) */
|
|
||||||
#define TYPE_MR 9 /* Mail rename name (experimental) */
|
|
||||||
#define TYPE_NULL 10 /* Null (experimental) */
|
|
||||||
#define TYPE_WKS 11 /* Well-known sockets */
|
|
||||||
#define TYPE_PTR 12 /* Pointer record */
|
|
||||||
#define TYPE_HINFO 13 /* Host information */
|
|
||||||
#define TYPE_MINFO 14 /* Mailbox information (experimental)*/
|
|
||||||
#define TYPE_MX 15 /* Mail exchanger */
|
|
||||||
#define TYPE_TXT 16 /* Text strings */
|
|
||||||
#define TYPE_ANY 255 /* Matches any type */
|
|
||||||
|
|
||||||
#define CLASS_IN 1 /* The ARPA Internet */
|
|
||||||
|
|
||||||
/* Round trip timing parameters */
|
|
||||||
#define AGAIN 8 /* Average RTT gain = 1/8 */
|
|
||||||
#define LAGAIN 3 /* Log2(AGAIN) */
|
|
||||||
#define DGAIN 4 /* Mean deviation gain = 1/4 */
|
|
||||||
#define LDGAIN 2 /* log2(DGAIN) */
|
|
||||||
|
|
||||||
/* Header for all domain messages */
|
|
||||||
struct dhdr
|
|
||||||
{
|
|
||||||
uint16_t id; /* Identification */
|
|
||||||
uint8_t qr; /* Query/Response */
|
|
||||||
#define QUERY 0
|
|
||||||
#define RESPONSE 1
|
|
||||||
uint8_t opcode;
|
|
||||||
#define IQUERY 1
|
|
||||||
uint8_t aa; /* Authoratative answer */
|
|
||||||
uint8_t tc; /* Truncation */
|
|
||||||
uint8_t rd; /* Recursion desired */
|
|
||||||
uint8_t ra; /* Recursion available */
|
|
||||||
uint8_t rcode; /* Response code */
|
|
||||||
#define NO_ERROR 0
|
|
||||||
#define FORMAT_ERROR 1
|
|
||||||
#define SERVER_FAIL 2
|
|
||||||
#define NAME_ERROR 3
|
|
||||||
#define NOT_IMPL 4
|
|
||||||
#define REFUSED 5
|
|
||||||
uint16_t qdcount; /* Question count */
|
|
||||||
uint16_t ancount; /* Answer count */
|
|
||||||
uint16_t nscount; /* Authority (name server) count */
|
|
||||||
uint16_t arcount; /* Additional record count */
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
uint8_t* pDNSMSG; // DNS message buffer
|
|
||||||
uint8_t DNS_SOCKET; // SOCKET number for DNS
|
|
||||||
uint16_t DNS_MSGID; // DNS message ID
|
|
||||||
|
|
||||||
|
|
||||||
uint32_t HAL_GetTick(void) {
|
|
||||||
return supervisor_ticks_ms32();
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t hal_sys_tick;
|
|
||||||
|
|
||||||
/* converts uint16_t from network buffer to a host byte order integer. */
|
|
||||||
uint16_t get16(uint8_t * s)
|
|
||||||
{
|
|
||||||
uint16_t i;
|
|
||||||
i = *s++ << 8;
|
|
||||||
i = i + *s;
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* copies uint16_t to the network buffer with network byte order. */
|
|
||||||
uint8_t * put16(uint8_t * s, uint16_t i)
|
|
||||||
{
|
|
||||||
*s++ = i >> 8;
|
|
||||||
*s++ = i;
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* CONVERT A DOMAIN NAME TO THE HUMAN-READABLE FORM
|
|
||||||
*
|
|
||||||
* Description : This function converts a compressed domain name to the human-readable form
|
|
||||||
* Arguments : msg - is a pointer to the reply message
|
|
||||||
* compressed - is a pointer to the domain name in reply message.
|
|
||||||
* buf - is a pointer to the buffer for the human-readable form name.
|
|
||||||
* len - is the MAX. size of buffer.
|
|
||||||
* Returns : the length of compressed message
|
|
||||||
*/
|
|
||||||
int parse_name(uint8_t * msg, uint8_t * compressed, char * buf, int16_t len)
|
|
||||||
{
|
|
||||||
uint16_t slen; /* Length of current segment */
|
|
||||||
uint8_t * cp;
|
|
||||||
int clen = 0; /* Total length of compressed name */
|
|
||||||
int indirect = 0; /* Set if indirection encountered */
|
|
||||||
int nseg = 0; /* Total number of segments in name */
|
|
||||||
|
|
||||||
cp = compressed;
|
|
||||||
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
slen = *cp++; /* Length of this segment */
|
|
||||||
|
|
||||||
if (!indirect) clen++;
|
|
||||||
|
|
||||||
if ((slen & 0xc0) == 0xc0)
|
|
||||||
{
|
|
||||||
if (!indirect)
|
|
||||||
clen++;
|
|
||||||
indirect = 1;
|
|
||||||
/* Follow indirection */
|
|
||||||
cp = &msg[((slen & 0x3f)<<8) + *cp];
|
|
||||||
slen = *cp++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (slen == 0) /* zero length == all done */
|
|
||||||
break;
|
|
||||||
|
|
||||||
len -= slen + 1;
|
|
||||||
|
|
||||||
if (len < 0) return -1;
|
|
||||||
|
|
||||||
if (!indirect) clen += slen;
|
|
||||||
|
|
||||||
while (slen-- != 0) *buf++ = (char)*cp++;
|
|
||||||
*buf++ = '.';
|
|
||||||
nseg++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nseg == 0)
|
|
||||||
{
|
|
||||||
/* Root name; represent as single dot */
|
|
||||||
*buf++ = '.';
|
|
||||||
len--;
|
|
||||||
}
|
|
||||||
|
|
||||||
*buf++ = '\0';
|
|
||||||
len--;
|
|
||||||
|
|
||||||
return clen; /* Length of compressed message */
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* PARSE QUESTION SECTION
|
|
||||||
*
|
|
||||||
* Description : This function parses the question record of the reply message.
|
|
||||||
* Arguments : msg - is a pointer to the reply message
|
|
||||||
* cp - is a pointer to the question record.
|
|
||||||
* Returns : a pointer the to next record.
|
|
||||||
*/
|
|
||||||
uint8_t * dns_question(uint8_t * msg, uint8_t * cp)
|
|
||||||
{
|
|
||||||
int len;
|
|
||||||
char name[MAXCNAME];
|
|
||||||
|
|
||||||
len = parse_name(msg, cp, name, MAXCNAME);
|
|
||||||
|
|
||||||
|
|
||||||
if (len == -1) return 0;
|
|
||||||
|
|
||||||
cp += len;
|
|
||||||
cp += 2; /* type */
|
|
||||||
cp += 2; /* class */
|
|
||||||
|
|
||||||
return cp;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* PARSE ANSER SECTION
|
|
||||||
*
|
|
||||||
* Description : This function parses the answer record of the reply message.
|
|
||||||
* Arguments : msg - is a pointer to the reply message
|
|
||||||
* cp - is a pointer to the answer record.
|
|
||||||
* Returns : a pointer the to next record.
|
|
||||||
*/
|
|
||||||
uint8_t * dns_answer(uint8_t * msg, uint8_t * cp, uint8_t * ip_from_dns)
|
|
||||||
{
|
|
||||||
int len, type;
|
|
||||||
char name[MAXCNAME];
|
|
||||||
|
|
||||||
len = parse_name(msg, cp, name, MAXCNAME);
|
|
||||||
|
|
||||||
if (len == -1) return 0;
|
|
||||||
|
|
||||||
cp += len;
|
|
||||||
type = get16(cp);
|
|
||||||
cp += 2; /* type */
|
|
||||||
cp += 2; /* class */
|
|
||||||
cp += 4; /* ttl */
|
|
||||||
cp += 2; /* len */
|
|
||||||
|
|
||||||
|
|
||||||
switch (type)
|
|
||||||
{
|
|
||||||
case TYPE_A:
|
|
||||||
/* Just read the address directly into the structure */
|
|
||||||
ip_from_dns[0] = *cp++;
|
|
||||||
ip_from_dns[1] = *cp++;
|
|
||||||
ip_from_dns[2] = *cp++;
|
|
||||||
ip_from_dns[3] = *cp++;
|
|
||||||
break;
|
|
||||||
case TYPE_CNAME:
|
|
||||||
case TYPE_MB:
|
|
||||||
case TYPE_MG:
|
|
||||||
case TYPE_MR:
|
|
||||||
case TYPE_NS:
|
|
||||||
case TYPE_PTR:
|
|
||||||
/* These types all consist of a single domain name */
|
|
||||||
/* convert it to ASCII format */
|
|
||||||
len = parse_name(msg, cp, name, MAXCNAME);
|
|
||||||
if (len == -1) return 0;
|
|
||||||
|
|
||||||
cp += len;
|
|
||||||
break;
|
|
||||||
case TYPE_HINFO:
|
|
||||||
len = *cp++;
|
|
||||||
cp += len;
|
|
||||||
|
|
||||||
len = *cp++;
|
|
||||||
cp += len;
|
|
||||||
break;
|
|
||||||
case TYPE_MX:
|
|
||||||
cp += 2;
|
|
||||||
/* Get domain name of exchanger */
|
|
||||||
len = parse_name(msg, cp, name, MAXCNAME);
|
|
||||||
if (len == -1) return 0;
|
|
||||||
|
|
||||||
cp += len;
|
|
||||||
break;
|
|
||||||
case TYPE_SOA:
|
|
||||||
/* Get domain name of name server */
|
|
||||||
len = parse_name(msg, cp, name, MAXCNAME);
|
|
||||||
if (len == -1) return 0;
|
|
||||||
|
|
||||||
cp += len;
|
|
||||||
|
|
||||||
/* Get domain name of responsible person */
|
|
||||||
len = parse_name(msg, cp, name, MAXCNAME);
|
|
||||||
if (len == -1) return 0;
|
|
||||||
|
|
||||||
cp += len;
|
|
||||||
|
|
||||||
cp += 4;
|
|
||||||
cp += 4;
|
|
||||||
cp += 4;
|
|
||||||
cp += 4;
|
|
||||||
cp += 4;
|
|
||||||
break;
|
|
||||||
case TYPE_TXT:
|
|
||||||
/* Just stash */
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
/* Ignore */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return cp;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* PARSE THE DNS REPLY
|
|
||||||
*
|
|
||||||
* Description : This function parses the reply message from DNS server.
|
|
||||||
* Arguments : dhdr - is a pointer to the header for DNS message
|
|
||||||
* buf - is a pointer to the reply message.
|
|
||||||
* len - is the size of reply message.
|
|
||||||
* Returns : -1 - Domain name length is too big
|
|
||||||
* 0 - Fail (Timeout or parse error)
|
|
||||||
* 1 - Success,
|
|
||||||
*/
|
|
||||||
int8_t parseDNSMSG(struct dhdr * pdhdr, uint8_t * pbuf, uint8_t * ip_from_dns)
|
|
||||||
{
|
|
||||||
uint16_t tmp;
|
|
||||||
uint16_t i;
|
|
||||||
uint8_t * msg;
|
|
||||||
uint8_t * cp;
|
|
||||||
|
|
||||||
msg = pbuf;
|
|
||||||
memset(pdhdr, 0, sizeof(*pdhdr));
|
|
||||||
|
|
||||||
pdhdr->id = get16(&msg[0]);
|
|
||||||
tmp = get16(&msg[2]);
|
|
||||||
if (tmp & 0x8000) pdhdr->qr = 1;
|
|
||||||
|
|
||||||
pdhdr->opcode = (tmp >> 11) & 0xf;
|
|
||||||
|
|
||||||
if (tmp & 0x0400) pdhdr->aa = 1;
|
|
||||||
if (tmp & 0x0200) pdhdr->tc = 1;
|
|
||||||
if (tmp & 0x0100) pdhdr->rd = 1;
|
|
||||||
if (tmp & 0x0080) pdhdr->ra = 1;
|
|
||||||
|
|
||||||
pdhdr->rcode = tmp & 0xf;
|
|
||||||
pdhdr->qdcount = get16(&msg[4]);
|
|
||||||
pdhdr->ancount = get16(&msg[6]);
|
|
||||||
pdhdr->nscount = get16(&msg[8]);
|
|
||||||
pdhdr->arcount = get16(&msg[10]);
|
|
||||||
|
|
||||||
|
|
||||||
/* Now parse the variable length sections */
|
|
||||||
cp = &msg[12];
|
|
||||||
|
|
||||||
/* Question section */
|
|
||||||
for (i = 0; i < pdhdr->qdcount; i++)
|
|
||||||
{
|
|
||||||
cp = dns_question(msg, cp);
|
|
||||||
if(!cp)
|
|
||||||
{
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
printf("MAX_DOMAIN_NAME is too small, it should be redefined in dns.h\r\n");
|
|
||||||
#endif
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Answer section */
|
|
||||||
for (i = 0; i < pdhdr->ancount; i++)
|
|
||||||
{
|
|
||||||
cp = dns_answer(msg, cp, ip_from_dns);
|
|
||||||
if(!cp)
|
|
||||||
{
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
printf("MAX_DOMAIN_NAME is too small, it should be redefined in dns.h\r\n");
|
|
||||||
#endif
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Name server (authority) section */
|
|
||||||
for (i = 0; i < pdhdr->nscount; i++)
|
|
||||||
{
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Additional section */
|
|
||||||
for (i = 0; i < pdhdr->arcount; i++)
|
|
||||||
{
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(pdhdr->rcode == 0) return 1; // No error
|
|
||||||
else return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* MAKE DNS QUERY MESSAGE
|
|
||||||
*
|
|
||||||
* Description : This function makes DNS query message.
|
|
||||||
* Arguments : op - Recursion desired
|
|
||||||
* name - is a pointer to the domain name.
|
|
||||||
* buf - is a pointer to the buffer for DNS message.
|
|
||||||
* len - is the MAX. size of buffer.
|
|
||||||
* Returns : the pointer to the DNS message.
|
|
||||||
*/
|
|
||||||
int16_t dns_makequery(uint16_t op, char * name, uint8_t * buf, uint16_t len)
|
|
||||||
{
|
|
||||||
uint8_t *cp;
|
|
||||||
char *cp1;
|
|
||||||
char sname[MAXCNAME];
|
|
||||||
char *dname;
|
|
||||||
uint16_t p;
|
|
||||||
uint16_t dlen;
|
|
||||||
|
|
||||||
cp = buf;
|
|
||||||
|
|
||||||
DNS_MSGID++;
|
|
||||||
cp = put16(cp, DNS_MSGID);
|
|
||||||
p = (op << 11) | 0x0100; /* Recursion desired */
|
|
||||||
cp = put16(cp, p);
|
|
||||||
cp = put16(cp, 1);
|
|
||||||
cp = put16(cp, 0);
|
|
||||||
cp = put16(cp, 0);
|
|
||||||
cp = put16(cp, 0);
|
|
||||||
|
|
||||||
strcpy(sname, name);
|
|
||||||
dname = sname;
|
|
||||||
dlen = strlen(dname);
|
|
||||||
for (;;)
|
|
||||||
{
|
|
||||||
/* Look for next dot */
|
|
||||||
cp1 = strchr(dname, '.');
|
|
||||||
|
|
||||||
if (cp1 != NULL) len = cp1 - dname; /* More to come */
|
|
||||||
else len = dlen; /* Last component */
|
|
||||||
|
|
||||||
*cp++ = len; /* Write length of component */
|
|
||||||
if (len == 0) break;
|
|
||||||
|
|
||||||
/* Copy component up to (but not including) dot */
|
|
||||||
memcpy(cp, dname, len);
|
|
||||||
cp += len;
|
|
||||||
if (cp1 == NULL)
|
|
||||||
{
|
|
||||||
*cp++ = 0; /* Last one; write null and finish */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
dname += len+1;
|
|
||||||
dlen -= len+1;
|
|
||||||
}
|
|
||||||
|
|
||||||
cp = put16(cp, 0x0001); /* type */
|
|
||||||
cp = put16(cp, 0x0001); /* class */
|
|
||||||
|
|
||||||
return ((int16_t)((uint32_t)(cp) - (uint32_t)(buf)));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* CHECK DNS TIMEOUT
|
|
||||||
*
|
|
||||||
* Description : This function check the DNS timeout
|
|
||||||
* Arguments : None.
|
|
||||||
* Returns : -1 - timeout occurred, 0 - timer over, but no timeout, 1 - no timer over, no timeout occur
|
|
||||||
* Note : timeout : retry count and timer both over.
|
|
||||||
*/
|
|
||||||
|
|
||||||
int8_t check_DNS_timeout(void)
|
|
||||||
{
|
|
||||||
static uint8_t retry_count;
|
|
||||||
|
|
||||||
uint32_t tick = HAL_GetTick();
|
|
||||||
if(tick - hal_sys_tick >= DNS_WAIT_TIME * 1000)
|
|
||||||
{
|
|
||||||
hal_sys_tick = tick;
|
|
||||||
if(retry_count >= MAX_DNS_RETRY) {
|
|
||||||
retry_count = 0;
|
|
||||||
return -1; // timeout occurred
|
|
||||||
}
|
|
||||||
retry_count++;
|
|
||||||
return 0; // timer over, but no timeout
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1; // no timer over, no timeout occur
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* DNS CLIENT INIT */
|
|
||||||
void DNS_init(uint8_t s, uint8_t * buf)
|
|
||||||
{
|
|
||||||
DNS_SOCKET = s; // SOCK_DNS
|
|
||||||
pDNSMSG = buf; // User's shared buffer
|
|
||||||
DNS_MSGID = DNS_MSG_ID;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* DNS CLIENT RUN */
|
|
||||||
int8_t DNS_run(uint8_t * dns_ip, uint8_t * name, uint8_t * ip_from_dns)
|
|
||||||
{
|
|
||||||
int8_t ret;
|
|
||||||
struct dhdr dhp;
|
|
||||||
uint8_t ip[4];
|
|
||||||
uint16_t len, port;
|
|
||||||
int8_t ret_check_timeout;
|
|
||||||
|
|
||||||
hal_sys_tick = HAL_GetTick();
|
|
||||||
|
|
||||||
// Socket open
|
|
||||||
WIZCHIP_EXPORT(socket)(DNS_SOCKET, Sn_MR_UDP, 0, 0);
|
|
||||||
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
printf("> DNS Query to DNS Server : %d.%d.%d.%d\r\n", dns_ip[0], dns_ip[1], dns_ip[2], dns_ip[3]);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
len = dns_makequery(0, (char *)name, pDNSMSG, MAX_DNS_BUF_SIZE);
|
|
||||||
WIZCHIP_EXPORT(sendto)(DNS_SOCKET, pDNSMSG, len, dns_ip, IPPORT_DOMAIN);
|
|
||||||
|
|
||||||
while (1)
|
|
||||||
{
|
|
||||||
if ((len = getSn_RX_RSR(DNS_SOCKET)) > 0)
|
|
||||||
{
|
|
||||||
if (len > MAX_DNS_BUF_SIZE) len = MAX_DNS_BUF_SIZE;
|
|
||||||
len = WIZCHIP_EXPORT(recvfrom)(DNS_SOCKET, pDNSMSG, len, ip, &port);
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
printf("> Receive DNS message from %d.%d.%d.%d(%d). len = %d\r\n", ip[0], ip[1], ip[2], ip[3],port,len);
|
|
||||||
#endif
|
|
||||||
ret = parseDNSMSG(&dhp, pDNSMSG, ip_from_dns);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
// Check Timeout
|
|
||||||
ret_check_timeout = check_DNS_timeout();
|
|
||||||
if (ret_check_timeout < 0) {
|
|
||||||
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
printf("> DNS Server is not responding : %d.%d.%d.%d\r\n", dns_ip[0], dns_ip[1], dns_ip[2], dns_ip[3]);
|
|
||||||
#endif
|
|
||||||
return 0; // timeout occurred
|
|
||||||
}
|
|
||||||
else if (ret_check_timeout == 0) {
|
|
||||||
|
|
||||||
#ifdef _DNS_DEBUG_
|
|
||||||
printf("> DNS Timeout\r\n");
|
|
||||||
#endif
|
|
||||||
WIZCHIP_EXPORT(sendto)(DNS_SOCKET, pDNSMSG, len, dns_ip, IPPORT_DOMAIN);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
WIZCHIP_EXPORT(close)(DNS_SOCKET);
|
|
||||||
// Return value
|
|
||||||
// 0 > : failed / 1 - success
|
|
||||||
return ret;
|
|
||||||
}
|
|
@ -1,96 +0,0 @@
|
|||||||
//*****************************************************************************
|
|
||||||
//
|
|
||||||
//! \file dns.h
|
|
||||||
//! \brief DNS APIs Header file.
|
|
||||||
//! \details Send DNS query & Receive DNS reponse.
|
|
||||||
//! \version 1.1.0
|
|
||||||
//! \date 2013/11/18
|
|
||||||
//! \par Revision history
|
|
||||||
//! <2013/10/21> 1st Release
|
|
||||||
//! <2013/12/20> V1.1.0
|
|
||||||
//! 1. Remove secondary DNS server in DNS_run
|
|
||||||
//! If 1st DNS_run failed, call DNS_run with 2nd DNS again
|
|
||||||
//! 2. DNS_timerHandler -> DNS_time_handler
|
|
||||||
//! 3. Move the no reference define to dns.c
|
|
||||||
//! 4. Integrated dns.h dns.c & dns_parse.h dns_parse.c into dns.h & dns.c
|
|
||||||
//! <2013/12/20> V1.1.0
|
|
||||||
//!
|
|
||||||
//! \author Eric Jung & MidnightCow
|
|
||||||
//! \copyright
|
|
||||||
//!
|
|
||||||
//! Copyright (c) 2013, WIZnet Co., LTD.
|
|
||||||
//! 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 the <ORGANIZATION> 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 OWNER 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.
|
|
||||||
//
|
|
||||||
//*****************************************************************************
|
|
||||||
|
|
||||||
#ifndef _DNS_H_
|
|
||||||
#define _DNS_H_
|
|
||||||
|
|
||||||
#include <stdint.h>
|
|
||||||
/*
|
|
||||||
* @brief Define it for Debug & Monitor DNS processing.
|
|
||||||
* @note If defined, it depends on <stdio.h>
|
|
||||||
*/
|
|
||||||
|
|
||||||
//#define _DNS_DEBUG_
|
|
||||||
|
|
||||||
#define MAX_DNS_BUF_SIZE 256 ///< maximum size of DNS buffer. */
|
|
||||||
/*
|
|
||||||
* @brief Maximum length of your queried Domain name
|
|
||||||
* @todo SHOULD BE defined it equal as or greater than your Domain name length + null character(1)
|
|
||||||
* @note SHOULD BE careful to stack overflow because it is allocated 1.5 times as MAX_DOMAIN_NAME in stack.
|
|
||||||
*/
|
|
||||||
#define MAX_DOMAIN_NAME 32 // for example "www.google.com"
|
|
||||||
|
|
||||||
#define MAX_DNS_RETRY 2 ///< Requery Count
|
|
||||||
#define DNS_WAIT_TIME 4 ///< Wait response time. unit 1s.
|
|
||||||
|
|
||||||
#define IPPORT_DOMAIN 53 ///< DNS server port number
|
|
||||||
|
|
||||||
#define DNS_MSG_ID 0x1122 ///< ID for DNS message. You can be modified it any number
|
|
||||||
/*
|
|
||||||
* @brief DNS process initialize
|
|
||||||
* @param s : Socket number for DNS
|
|
||||||
* @param buf : Buffer for DNS message
|
|
||||||
*/
|
|
||||||
void DNS_init(uint8_t s, uint8_t * buf);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @brief DNS process
|
|
||||||
* @details Send DNS query and receive DNS response
|
|
||||||
* @param dns_ip : DNS server ip address
|
|
||||||
* @param name : Domain name to be queried
|
|
||||||
* @param ip_from_dns : IP address from DNS server
|
|
||||||
* @return -1 : failed. @ref MAX_DOMIN_NAME is too small \n
|
|
||||||
* 0 : failed (Timeout or Parse error)\n
|
|
||||||
* 1 : success
|
|
||||||
* @note This function blocks until success or fail. max time = @ref MAX_DNS_RETRY * @ref DNS_WAIT_TIME
|
|
||||||
*/
|
|
||||||
int8_t DNS_run(uint8_t * dns_ip, uint8_t * name, uint8_t * ip_from_dns);
|
|
||||||
|
|
||||||
#endif /* _DNS_H_ */
|
|
1
examples/natmod/.gitignore
vendored
Normal file
1
examples/natmod/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
*.mpy
|
37
examples/natmod/btree/Makefile
Normal file
37
examples/natmod/btree/Makefile
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module (different to built-in btree so it can coexist)
|
||||||
|
MOD = btree_$(ARCH)
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = btree_c.c btree_py.py
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
BTREE_DIR = $(MPY_DIR)/lib/berkeley-db-1.xx
|
||||||
|
BTREE_DEFS = -D__DBINTERFACE_PRIVATE=1 -Dmpool_error="(void)" -Dabort=abort_ "-Dvirt_fd_t=void*" $(BTREE_DEFS_EXTRA)
|
||||||
|
CFLAGS += -I$(BTREE_DIR)/PORT/include
|
||||||
|
CFLAGS += -Wno-old-style-definition -Wno-sign-compare -Wno-unused-parameter $(BTREE_DEFS)
|
||||||
|
|
||||||
|
SRC += $(addprefix $(realpath $(BTREE_DIR))/,\
|
||||||
|
btree/bt_close.c \
|
||||||
|
btree/bt_conv.c \
|
||||||
|
btree/bt_delete.c \
|
||||||
|
btree/bt_get.c \
|
||||||
|
btree/bt_open.c \
|
||||||
|
btree/bt_overflow.c \
|
||||||
|
btree/bt_page.c \
|
||||||
|
btree/bt_put.c \
|
||||||
|
btree/bt_search.c \
|
||||||
|
btree/bt_seq.c \
|
||||||
|
btree/bt_split.c \
|
||||||
|
btree/bt_utils.c \
|
||||||
|
mpool/mpool.c \
|
||||||
|
)
|
||||||
|
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
||||||
|
|
||||||
|
# btree needs gnu99 defined
|
||||||
|
CFLAGS += -std=gnu99
|
147
examples/natmod/btree/btree_c.c
Normal file
147
examples/natmod/btree/btree_c.c
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
#define MICROPY_PY_BTREE (1)
|
||||||
|
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#if !defined(__linux__)
|
||||||
|
void *memcpy(void *dst, const void *src, size_t n) {
|
||||||
|
return mp_fun_table.memmove_(dst, src, n);
|
||||||
|
}
|
||||||
|
void *memset(void *s, int c, size_t n) {
|
||||||
|
return mp_fun_table.memset_(s, c, n);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void *memmove(void *dest, const void *src, size_t n) {
|
||||||
|
return mp_fun_table.memmove_(dest, src, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *malloc(size_t n) {
|
||||||
|
void *ptr = m_malloc(n, false);
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
void *realloc(void *ptr, size_t n) {
|
||||||
|
mp_printf(&mp_plat_print, "UNDEF %d\n", __LINE__);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
void *calloc(size_t n, size_t m) {
|
||||||
|
void *ptr = m_malloc(n * m, false);
|
||||||
|
// memory already cleared by conservative GC
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void free(void *ptr) {
|
||||||
|
m_free(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void abort_(void) {
|
||||||
|
nlr_raise(mp_obj_new_exception(mp_load_global(MP_QSTR_RuntimeError)));
|
||||||
|
}
|
||||||
|
|
||||||
|
int native_errno;
|
||||||
|
#if defined(__linux__)
|
||||||
|
int *__errno_location (void)
|
||||||
|
#else
|
||||||
|
int *__errno (void)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
return &native_errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t mp_stream_posix_write(void *stream, const void *buf, size_t len) {
|
||||||
|
mp_obj_base_t* o = stream;
|
||||||
|
const mp_stream_p_t *stream_p = o->type->protocol;
|
||||||
|
mp_uint_t out_sz = stream_p->write(MP_OBJ_FROM_PTR(stream), buf, len, &native_errno);
|
||||||
|
if (out_sz == MP_STREAM_ERROR) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
return out_sz;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t mp_stream_posix_read(void *stream, void *buf, size_t len) {
|
||||||
|
mp_obj_base_t* o = stream;
|
||||||
|
const mp_stream_p_t *stream_p = o->type->protocol;
|
||||||
|
mp_uint_t out_sz = stream_p->read(MP_OBJ_FROM_PTR(stream), buf, len, &native_errno);
|
||||||
|
if (out_sz == MP_STREAM_ERROR) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
return out_sz;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
off_t mp_stream_posix_lseek(void *stream, off_t offset, int whence) {
|
||||||
|
const mp_obj_base_t* o = stream;
|
||||||
|
const mp_stream_p_t *stream_p = o->type->protocol;
|
||||||
|
struct mp_stream_seek_t seek_s;
|
||||||
|
seek_s.offset = offset;
|
||||||
|
seek_s.whence = whence;
|
||||||
|
mp_uint_t res = stream_p->ioctl(MP_OBJ_FROM_PTR(stream), MP_STREAM_SEEK, (mp_uint_t)(uintptr_t)&seek_s, &native_errno);
|
||||||
|
if (res == MP_STREAM_ERROR) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return seek_s.offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mp_stream_posix_fsync(void *stream) {
|
||||||
|
mp_obj_base_t* o = stream;
|
||||||
|
const mp_stream_p_t *stream_p = o->type->protocol;
|
||||||
|
mp_uint_t res = stream_p->ioctl(MP_OBJ_FROM_PTR(stream), MP_STREAM_FLUSH, 0, &native_errno);
|
||||||
|
if (res == MP_STREAM_ERROR) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
mp_obj_type_t btree_type;
|
||||||
|
|
||||||
|
#include "extmod/modbtree.c"
|
||||||
|
|
||||||
|
mp_map_elem_t btree_locals_dict_table[8];
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(btree_locals_dict, btree_locals_dict_table);
|
||||||
|
|
||||||
|
STATIC mp_obj_t btree_open(size_t n_args, const mp_obj_t *args) {
|
||||||
|
// Make sure we got a stream object
|
||||||
|
mp_get_stream_raise(args[0], MP_STREAM_OP_READ | MP_STREAM_OP_WRITE | MP_STREAM_OP_IOCTL);
|
||||||
|
|
||||||
|
BTREEINFO openinfo = {0};
|
||||||
|
openinfo.flags = mp_obj_get_int(args[1]);
|
||||||
|
openinfo.cachesize = mp_obj_get_int(args[2]);
|
||||||
|
openinfo.psize = mp_obj_get_int(args[3]);
|
||||||
|
openinfo.minkeypage = mp_obj_get_int(args[4]);
|
||||||
|
DB *db = __bt_open(MP_OBJ_TO_PTR(args[0]), &btree_stream_fvtable, &openinfo, 0);
|
||||||
|
if (db == NULL) {
|
||||||
|
mp_raise_OSError(native_errno);
|
||||||
|
}
|
||||||
|
|
||||||
|
return MP_OBJ_FROM_PTR(btree_new(db, args[0]));
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(btree_open_obj, 5, 5, btree_open);
|
||||||
|
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
btree_type.base.type = (void*)&mp_fun_table.type_type;
|
||||||
|
btree_type.name = MP_QSTR_btree;
|
||||||
|
btree_type.print = btree_print;
|
||||||
|
btree_type.getiter = btree_getiter;
|
||||||
|
btree_type.iternext = btree_iternext;
|
||||||
|
btree_type.binary_op = btree_binary_op;
|
||||||
|
btree_type.subscr = btree_subscr;
|
||||||
|
btree_locals_dict_table[0] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_close), MP_OBJ_FROM_PTR(&btree_close_obj) };
|
||||||
|
btree_locals_dict_table[1] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_flush), MP_OBJ_FROM_PTR(&btree_flush_obj) };
|
||||||
|
btree_locals_dict_table[2] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_get), MP_OBJ_FROM_PTR(&btree_get_obj) };
|
||||||
|
btree_locals_dict_table[3] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_put), MP_OBJ_FROM_PTR(&btree_put_obj) };
|
||||||
|
btree_locals_dict_table[4] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_seq), MP_OBJ_FROM_PTR(&btree_seq_obj) };
|
||||||
|
btree_locals_dict_table[5] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_keys), MP_OBJ_FROM_PTR(&btree_keys_obj) };
|
||||||
|
btree_locals_dict_table[6] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_values), MP_OBJ_FROM_PTR(&btree_values_obj) };
|
||||||
|
btree_locals_dict_table[7] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_items), MP_OBJ_FROM_PTR(&btree_items_obj) };
|
||||||
|
btree_type.locals_dict = (void*)&btree_locals_dict;
|
||||||
|
|
||||||
|
mp_store_global(MP_QSTR__open, MP_OBJ_FROM_PTR(&btree_open_obj));
|
||||||
|
mp_store_global(MP_QSTR_INCL, MP_OBJ_NEW_SMALL_INT(FLAG_END_KEY_INCL));
|
||||||
|
mp_store_global(MP_QSTR_DESC, MP_OBJ_NEW_SMALL_INT(FLAG_DESC));
|
||||||
|
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
3
examples/natmod/btree/btree_py.py
Normal file
3
examples/natmod/btree/btree_py.py
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
# Implemented in Python to support keyword arguments
|
||||||
|
def open(stream, *, flags=0, cachesize=0, pagesize=0, minkeypage=0):
|
||||||
|
return _open(stream, flags, cachesize, pagesize, minkeypage)
|
14
examples/natmod/features0/Makefile
Normal file
14
examples/natmod/features0/Makefile
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module
|
||||||
|
MOD = features0
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = features0.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
# Include to get the rules for compiling and linking the module
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
40
examples/natmod/features0/features0.c
Normal file
40
examples/natmod/features0/features0.c
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
/* This example demonstrates the following features in a native module:
|
||||||
|
- defining a simple function exposed to Python
|
||||||
|
- defining a local, helper C function
|
||||||
|
- getting and creating integer objects
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Include the header file to get access to the MicroPython API
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
// Helper function to compute factorial
|
||||||
|
STATIC mp_int_t factorial_helper(mp_int_t x) {
|
||||||
|
if (x == 0) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return x * factorial_helper(x - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is the function which will be called from Python, as factorial(x)
|
||||||
|
STATIC mp_obj_t factorial(mp_obj_t x_obj) {
|
||||||
|
// Extract the integer from the MicroPython input object
|
||||||
|
mp_int_t x = mp_obj_get_int(x_obj);
|
||||||
|
// Calculate the factorial
|
||||||
|
mp_int_t result = factorial_helper(x);
|
||||||
|
// Convert the result to a MicroPython integer object and return it
|
||||||
|
return mp_obj_new_int(result);
|
||||||
|
}
|
||||||
|
// Define a Python reference to the function above
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_1(factorial_obj, factorial);
|
||||||
|
|
||||||
|
// This is the entry point and is called when the module is imported
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
// This must be first, it sets up the globals dict and other things
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
// Make the function available in the module's namespace
|
||||||
|
mp_store_global(MP_QSTR_factorial, MP_OBJ_FROM_PTR(&factorial_obj));
|
||||||
|
|
||||||
|
// This must be last, it restores the globals dict
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
14
examples/natmod/features1/Makefile
Normal file
14
examples/natmod/features1/Makefile
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module
|
||||||
|
MOD = features1
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = features1.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
# Include to get the rules for compiling and linking the module
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
106
examples/natmod/features1/features1.c
Normal file
106
examples/natmod/features1/features1.c
Normal file
@ -0,0 +1,106 @@
|
|||||||
|
/* This example demonstrates the following features in a native module:
|
||||||
|
- defining simple functions exposed to Python
|
||||||
|
- defining local, helper C functions
|
||||||
|
- defining constant integers and strings exposed to Python
|
||||||
|
- getting and creating integer objects
|
||||||
|
- creating Python lists
|
||||||
|
- raising exceptions
|
||||||
|
- allocating memory
|
||||||
|
- BSS and constant data (rodata)
|
||||||
|
- relocated pointers in rodata
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Include the header file to get access to the MicroPython API
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
// BSS (zero) data
|
||||||
|
uint16_t data16[4];
|
||||||
|
|
||||||
|
// Constant data (rodata)
|
||||||
|
const uint8_t table8[] = { 0, 1, 1, 2, 3, 5, 8, 13 };
|
||||||
|
const uint16_t table16[] = { 0x1000, 0x2000 };
|
||||||
|
|
||||||
|
// Constant data pointing to BSS/constant data
|
||||||
|
uint16_t *const table_ptr16a[] = { &data16[0], &data16[1], &data16[2], &data16[3] };
|
||||||
|
const uint16_t *const table_ptr16b[] = { &table16[0], &table16[1] };
|
||||||
|
|
||||||
|
// A simple function that adds its 2 arguments (must be integers)
|
||||||
|
STATIC mp_obj_t add(mp_obj_t x_in, mp_obj_t y_in) {
|
||||||
|
mp_int_t x = mp_obj_get_int(x_in);
|
||||||
|
mp_int_t y = mp_obj_get_int(y_in);
|
||||||
|
return mp_obj_new_int(x + y);
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(add_obj, add);
|
||||||
|
|
||||||
|
// A local helper function (not exposed to Python)
|
||||||
|
STATIC mp_int_t fibonacci_helper(mp_int_t x) {
|
||||||
|
if (x < MP_ARRAY_SIZE(table8)) {
|
||||||
|
return table8[x];
|
||||||
|
} else {
|
||||||
|
return fibonacci_helper(x - 1) + fibonacci_helper(x - 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// A function which computes Fibonacci numbers
|
||||||
|
STATIC mp_obj_t fibonacci(mp_obj_t x_in) {
|
||||||
|
mp_int_t x = mp_obj_get_int(x_in);
|
||||||
|
if (x < 0) {
|
||||||
|
mp_raise_ValueError(MP_ERROR_TEXT("can't compute negative Fibonacci number"));
|
||||||
|
}
|
||||||
|
return mp_obj_new_int(fibonacci_helper(x));
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_1(fibonacci_obj, fibonacci);
|
||||||
|
|
||||||
|
// A function that accesses the BSS data
|
||||||
|
STATIC mp_obj_t access(size_t n_args, const mp_obj_t *args) {
|
||||||
|
if (n_args == 0) {
|
||||||
|
// Create a list holding all items from data16
|
||||||
|
mp_obj_list_t *lst = MP_OBJ_TO_PTR(mp_obj_new_list(MP_ARRAY_SIZE(data16), NULL));
|
||||||
|
for (int i = 0; i < MP_ARRAY_SIZE(data16); ++i) {
|
||||||
|
lst->items[i] = mp_obj_new_int(data16[i]);
|
||||||
|
}
|
||||||
|
return MP_OBJ_FROM_PTR(lst);
|
||||||
|
} else if (n_args == 1) {
|
||||||
|
// Get one item from data16
|
||||||
|
mp_int_t idx = mp_obj_get_int(args[0]) & 3;
|
||||||
|
return mp_obj_new_int(data16[idx]);
|
||||||
|
} else {
|
||||||
|
// Set one item in data16 (via table_ptr16a)
|
||||||
|
mp_int_t idx = mp_obj_get_int(args[0]) & 3;
|
||||||
|
*table_ptr16a[idx] = mp_obj_get_int(args[1]);
|
||||||
|
return mp_const_none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(access_obj, 0, 2, access);
|
||||||
|
|
||||||
|
// A function that allocates memory and creates a bytearray
|
||||||
|
STATIC mp_obj_t make_array(void) {
|
||||||
|
uint16_t *ptr = m_new(uint16_t, MP_ARRAY_SIZE(table_ptr16b));
|
||||||
|
for (int i = 0; i < MP_ARRAY_SIZE(table_ptr16b); ++i) {
|
||||||
|
ptr[i] = *table_ptr16b[i];
|
||||||
|
}
|
||||||
|
return mp_obj_new_bytearray_by_ref(sizeof(uint16_t) * MP_ARRAY_SIZE(table_ptr16b), ptr);
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_0(make_array_obj, make_array);
|
||||||
|
|
||||||
|
// This is the entry point and is called when the module is imported
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
// This must be first, it sets up the globals dict and other things
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
// Messages can be printed as usualy
|
||||||
|
mp_printf(&mp_plat_print, "initialising module self=%p\n", self);
|
||||||
|
|
||||||
|
// Make the functions available in the module's namespace
|
||||||
|
mp_store_global(MP_QSTR_add, MP_OBJ_FROM_PTR(&add_obj));
|
||||||
|
mp_store_global(MP_QSTR_fibonacci, MP_OBJ_FROM_PTR(&fibonacci_obj));
|
||||||
|
mp_store_global(MP_QSTR_access, MP_OBJ_FROM_PTR(&access_obj));
|
||||||
|
mp_store_global(MP_QSTR_make_array, MP_OBJ_FROM_PTR(&make_array_obj));
|
||||||
|
|
||||||
|
// Add some constants to the module's namespace
|
||||||
|
mp_store_global(MP_QSTR_VAL, MP_OBJ_NEW_SMALL_INT(42));
|
||||||
|
mp_store_global(MP_QSTR_MSG, MP_OBJ_NEW_QSTR(MP_QSTR_HELLO_MICROPYTHON));
|
||||||
|
|
||||||
|
// This must be last, it restores the globals dict
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
14
examples/natmod/features2/Makefile
Normal file
14
examples/natmod/features2/Makefile
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module
|
||||||
|
MOD = features2
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = main.c prod.c test.py
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
# Include to get the rules for compiling and linking the module
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
83
examples/natmod/features2/main.c
Normal file
83
examples/natmod/features2/main.c
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
/* This example demonstrates the following features in a native module:
|
||||||
|
- using floats
|
||||||
|
- defining additional code in Python (see test.py)
|
||||||
|
- have extra C code in a separate file (see prod.c)
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Include the header file to get access to the MicroPython API
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
// Include the header for auxiliary C code for this module
|
||||||
|
#include "prod.h"
|
||||||
|
|
||||||
|
// Automatically detect if this module should include double-precision code.
|
||||||
|
// If double precision is supported by the target architecture then it can
|
||||||
|
// be used in native module regardless of what float setting the target
|
||||||
|
// MicroPython runtime uses (being none, float or double).
|
||||||
|
#if defined(__i386__) || defined(__x86_64__) || (defined(__ARM_FP) && (__ARM_FP & 8))
|
||||||
|
#define USE_DOUBLE 1
|
||||||
|
#else
|
||||||
|
#define USE_DOUBLE 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// A function that uses the default float type configured for the current target
|
||||||
|
// This default can be overridden by specifying MICROPY_FLOAT_IMPL at the make level
|
||||||
|
STATIC mp_obj_t add(mp_obj_t x, mp_obj_t y) {
|
||||||
|
return mp_obj_new_float(mp_obj_get_float(x) + mp_obj_get_float(y));
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(add_obj, add);
|
||||||
|
|
||||||
|
// A function that explicitly uses single precision floats
|
||||||
|
STATIC mp_obj_t add_f(mp_obj_t x, mp_obj_t y) {
|
||||||
|
return mp_obj_new_float_from_f(mp_obj_get_float_to_f(x) + mp_obj_get_float_to_f(y));
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(add_f_obj, add_f);
|
||||||
|
|
||||||
|
#if USE_DOUBLE
|
||||||
|
// A function that explicitly uses double precision floats
|
||||||
|
STATIC mp_obj_t add_d(mp_obj_t x, mp_obj_t y) {
|
||||||
|
return mp_obj_new_float_from_d(mp_obj_get_float_to_d(x) + mp_obj_get_float_to_d(y));
|
||||||
|
}
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(add_d_obj, add_d);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// A function that computes the product of floats in an array.
|
||||||
|
// This function uses the most general C argument interface, which is more difficult
|
||||||
|
// to use but has access to the globals dict of the module via self->globals.
|
||||||
|
STATIC mp_obj_t productf(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
// Check number of arguments is valid
|
||||||
|
mp_arg_check_num_mp(n_args, n_kw, 1, 1, false);
|
||||||
|
|
||||||
|
// Extract buffer pointer and verify typecode
|
||||||
|
mp_buffer_info_t bufinfo;
|
||||||
|
mp_get_buffer_raise(args[0], &bufinfo, MP_BUFFER_RW);
|
||||||
|
if (bufinfo.typecode != 'f') {
|
||||||
|
mp_raise_ValueError(MP_ERROR_TEXT("expecting float array"));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compute product, store result back in first element of array
|
||||||
|
float *ptr = bufinfo.buf;
|
||||||
|
float prod = prod_array(bufinfo.len / sizeof(*ptr), ptr);
|
||||||
|
ptr[0] = prod;
|
||||||
|
|
||||||
|
return mp_const_none;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is the entry point and is called when the module is imported
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
// This must be first, it sets up the globals dict and other things
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
// Make the functions available in the module's namespace
|
||||||
|
mp_store_global(MP_QSTR_add, MP_OBJ_FROM_PTR(&add_obj));
|
||||||
|
mp_store_global(MP_QSTR_add_f, MP_OBJ_FROM_PTR(&add_f_obj));
|
||||||
|
#if USE_DOUBLE
|
||||||
|
mp_store_global(MP_QSTR_add_d, MP_OBJ_FROM_PTR(&add_d_obj));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// The productf function uses the most general C argument interface
|
||||||
|
mp_store_global(MP_QSTR_productf, MP_DYNRUNTIME_MAKE_FUNCTION(productf));
|
||||||
|
|
||||||
|
// This must be last, it restores the globals dict
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
9
examples/natmod/features2/prod.c
Normal file
9
examples/natmod/features2/prod.c
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
#include "prod.h"
|
||||||
|
|
||||||
|
float prod_array(int n, float *ar) {
|
||||||
|
float ans = 1;
|
||||||
|
for (int i = 0; i < n; ++i) {
|
||||||
|
ans *= ar[i];
|
||||||
|
}
|
||||||
|
return ans;
|
||||||
|
}
|
1
examples/natmod/features2/prod.h
Normal file
1
examples/natmod/features2/prod.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
float prod_array(int n, float *ar);
|
29
examples/natmod/features2/test.py
Normal file
29
examples/natmod/features2/test.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# This Python code will be merged with the C code in main.c
|
||||||
|
|
||||||
|
import array
|
||||||
|
|
||||||
|
|
||||||
|
def isclose(a, b):
|
||||||
|
return abs(a - b) < 1e-3
|
||||||
|
|
||||||
|
|
||||||
|
def test():
|
||||||
|
tests = [
|
||||||
|
isclose(add(0.1, 0.2), 0.3),
|
||||||
|
isclose(add_f(0.1, 0.2), 0.3),
|
||||||
|
]
|
||||||
|
|
||||||
|
ar = array.array("f", [1, 2, 3.5])
|
||||||
|
productf(ar)
|
||||||
|
tests.append(isclose(ar[0], 7))
|
||||||
|
|
||||||
|
if "add_d" in globals():
|
||||||
|
tests.append(isclose(add_d(0.1, 0.2), 0.3))
|
||||||
|
|
||||||
|
print(tests)
|
||||||
|
|
||||||
|
if not all(tests):
|
||||||
|
raise SystemExit(1)
|
||||||
|
|
||||||
|
|
||||||
|
test()
|
13
examples/natmod/framebuf/Makefile
Normal file
13
examples/natmod/framebuf/Makefile
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module (different to built-in framebuf so it can coexist)
|
||||||
|
MOD = framebuf_$(ARCH)
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = framebuf.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
49
examples/natmod/framebuf/framebuf.c
Normal file
49
examples/natmod/framebuf/framebuf.c
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
#define MICROPY_PY_FRAMEBUF (1)
|
||||||
|
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
#if !defined(__linux__)
|
||||||
|
void *memset(void *s, int c, size_t n) {
|
||||||
|
return mp_fun_table.memset_(s, c, n);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
mp_obj_type_t mp_type_framebuf;
|
||||||
|
|
||||||
|
#include "extmod/modframebuf.c"
|
||||||
|
|
||||||
|
mp_map_elem_t framebuf_locals_dict_table[10];
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(framebuf_locals_dict, framebuf_locals_dict_table);
|
||||||
|
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
mp_type_framebuf.base.type = (void*)&mp_type_type;
|
||||||
|
mp_type_framebuf.name = MP_QSTR_FrameBuffer;
|
||||||
|
mp_type_framebuf.make_new = framebuf_make_new;
|
||||||
|
mp_type_framebuf.buffer_p.get_buffer = framebuf_get_buffer;
|
||||||
|
framebuf_locals_dict_table[0] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_fill), MP_OBJ_FROM_PTR(&framebuf_fill_obj) };
|
||||||
|
framebuf_locals_dict_table[1] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_fill_rect), MP_OBJ_FROM_PTR(&framebuf_fill_rect_obj) };
|
||||||
|
framebuf_locals_dict_table[2] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_pixel), MP_OBJ_FROM_PTR(&framebuf_pixel_obj) };
|
||||||
|
framebuf_locals_dict_table[3] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_hline), MP_OBJ_FROM_PTR(&framebuf_hline_obj) };
|
||||||
|
framebuf_locals_dict_table[4] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_vline), MP_OBJ_FROM_PTR(&framebuf_vline_obj) };
|
||||||
|
framebuf_locals_dict_table[5] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_rect), MP_OBJ_FROM_PTR(&framebuf_rect_obj) };
|
||||||
|
framebuf_locals_dict_table[6] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_line), MP_OBJ_FROM_PTR(&framebuf_line_obj) };
|
||||||
|
framebuf_locals_dict_table[7] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_blit), MP_OBJ_FROM_PTR(&framebuf_blit_obj) };
|
||||||
|
framebuf_locals_dict_table[8] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_scroll), MP_OBJ_FROM_PTR(&framebuf_scroll_obj) };
|
||||||
|
framebuf_locals_dict_table[9] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_text), MP_OBJ_FROM_PTR(&framebuf_text_obj) };
|
||||||
|
mp_type_framebuf.locals_dict = (void*)&framebuf_locals_dict;
|
||||||
|
|
||||||
|
mp_store_global(MP_QSTR_FrameBuffer, MP_OBJ_FROM_PTR(&mp_type_framebuf));
|
||||||
|
mp_store_global(MP_QSTR_FrameBuffer1, MP_OBJ_FROM_PTR(&legacy_framebuffer1_obj));
|
||||||
|
mp_store_global(MP_QSTR_MVLSB, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_MVLSB));
|
||||||
|
mp_store_global(MP_QSTR_MONO_VLSB, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_MVLSB));
|
||||||
|
mp_store_global(MP_QSTR_RGB565, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_RGB565));
|
||||||
|
mp_store_global(MP_QSTR_GS2_HMSB, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_GS2_HMSB));
|
||||||
|
mp_store_global(MP_QSTR_GS4_HMSB, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_GS4_HMSB));
|
||||||
|
mp_store_global(MP_QSTR_GS8, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_GS8));
|
||||||
|
mp_store_global(MP_QSTR_MONO_HLSB, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_MHLSB));
|
||||||
|
mp_store_global(MP_QSTR_MONO_HMSB, MP_OBJ_NEW_SMALL_INT(FRAMEBUF_MHMSB));
|
||||||
|
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
13
examples/natmod/uheapq/Makefile
Normal file
13
examples/natmod/uheapq/Makefile
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module (different to built-in uheapq so it can coexist)
|
||||||
|
MOD = uheapq_$(ARCH)
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = uheapq.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
16
examples/natmod/uheapq/uheapq.c
Normal file
16
examples/natmod/uheapq/uheapq.c
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#define MICROPY_PY_UHEAPQ (1)
|
||||||
|
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
#include "extmod/moduheapq.c"
|
||||||
|
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
mp_store_global(MP_QSTR___name__, MP_OBJ_NEW_QSTR(MP_QSTR_uheapq));
|
||||||
|
mp_store_global(MP_QSTR_heappush, MP_OBJ_FROM_PTR(&mod_uheapq_heappush_obj));
|
||||||
|
mp_store_global(MP_QSTR_heappop, MP_OBJ_FROM_PTR(&mod_uheapq_heappop_obj));
|
||||||
|
mp_store_global(MP_QSTR_heapify, MP_OBJ_FROM_PTR(&mod_uheapq_heapify_obj));
|
||||||
|
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
13
examples/natmod/urandom/Makefile
Normal file
13
examples/natmod/urandom/Makefile
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module (different to built-in urandom so it can coexist)
|
||||||
|
MOD = urandom_$(ARCH)
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = urandom.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
33
examples/natmod/urandom/urandom.c
Normal file
33
examples/natmod/urandom/urandom.c
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#define MICROPY_PY_URANDOM (1)
|
||||||
|
#define MICROPY_PY_URANDOM_EXTRA_FUNCS (1)
|
||||||
|
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
// Dynamic native modules don't support a data section so these must go in the BSS
|
||||||
|
uint32_t yasmarang_pad, yasmarang_n, yasmarang_d;
|
||||||
|
uint8_t yasmarang_dat;
|
||||||
|
|
||||||
|
#include "extmod/modurandom.c"
|
||||||
|
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
yasmarang_pad = 0xeda4baba;
|
||||||
|
yasmarang_n = 69;
|
||||||
|
yasmarang_d = 233;
|
||||||
|
|
||||||
|
mp_store_global(MP_QSTR___name__, MP_OBJ_NEW_QSTR(MP_QSTR_urandom));
|
||||||
|
mp_store_global(MP_QSTR_getrandbits, MP_OBJ_FROM_PTR(&mod_urandom_getrandbits_obj));
|
||||||
|
mp_store_global(MP_QSTR_seed, MP_OBJ_FROM_PTR(&mod_urandom_seed_obj));
|
||||||
|
#if MICROPY_PY_URANDOM_EXTRA_FUNCS
|
||||||
|
mp_store_global(MP_QSTR_randrange, MP_OBJ_FROM_PTR(&mod_urandom_randrange_obj));
|
||||||
|
mp_store_global(MP_QSTR_randint, MP_OBJ_FROM_PTR(&mod_urandom_randint_obj));
|
||||||
|
mp_store_global(MP_QSTR_choice, MP_OBJ_FROM_PTR(&mod_urandom_choice_obj));
|
||||||
|
#if MICROPY_PY_BUILTINS_FLOAT
|
||||||
|
mp_store_global(MP_QSTR_random, MP_OBJ_FROM_PTR(&mod_urandom_random_obj));
|
||||||
|
mp_store_global(MP_QSTR_uniform, MP_OBJ_FROM_PTR(&mod_urandom_uniform_obj));
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
13
examples/natmod/ure/Makefile
Normal file
13
examples/natmod/ure/Makefile
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module (different to built-in ure so it can coexist)
|
||||||
|
MOD = ure_$(ARCH)
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = ure.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
78
examples/natmod/ure/ure.c
Normal file
78
examples/natmod/ure/ure.c
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
#define MICROPY_STACK_CHECK (1)
|
||||||
|
#define MICROPY_PY_URE (1)
|
||||||
|
#define MICROPY_PY_URE_MATCH_GROUPS (1)
|
||||||
|
#define MICROPY_PY_URE_MATCH_SPAN_START_END (1)
|
||||||
|
#define MICROPY_PY_URE_SUB (0) // requires vstr interface
|
||||||
|
|
||||||
|
#include <alloca.h>
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
#define STACK_LIMIT (2048)
|
||||||
|
|
||||||
|
const char *stack_top;
|
||||||
|
|
||||||
|
void mp_stack_check(void) {
|
||||||
|
// Assumes descending stack on target
|
||||||
|
volatile char dummy;
|
||||||
|
if (stack_top - &dummy >= STACK_LIMIT) {
|
||||||
|
mp_raise_msg(&mp_type_RuntimeError, MP_ERROR_TEXT("maximum recursion depth exceeded"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if !defined(__linux__)
|
||||||
|
void *memcpy(void *dst, const void *src, size_t n) {
|
||||||
|
return mp_fun_table.memmove_(dst, src, n);
|
||||||
|
}
|
||||||
|
void *memset(void *s, int c, size_t n) {
|
||||||
|
return mp_fun_table.memset_(s, c, n);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void *memmove(void *dest, const void *src, size_t n) {
|
||||||
|
return mp_fun_table.memmove_(dest, src, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
mp_obj_type_t match_type;
|
||||||
|
mp_obj_type_t re_type;
|
||||||
|
|
||||||
|
#include "extmod/modure.c"
|
||||||
|
|
||||||
|
mp_map_elem_t match_locals_dict_table[5];
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(match_locals_dict, match_locals_dict_table);
|
||||||
|
|
||||||
|
mp_map_elem_t re_locals_dict_table[3];
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(re_locals_dict, re_locals_dict_table);
|
||||||
|
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
char dummy;
|
||||||
|
stack_top = &dummy;
|
||||||
|
|
||||||
|
// Because MP_QSTR_start/end/split are static, xtensa and xtensawin will make a small data section
|
||||||
|
// to copy in this key/value pair if they are specified as a struct, so assign them separately.
|
||||||
|
|
||||||
|
match_type.base.type = (void*)&mp_fun_table.type_type;
|
||||||
|
match_type.name = MP_QSTR_match;
|
||||||
|
match_type.print = match_print;
|
||||||
|
match_locals_dict_table[0] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_group), MP_OBJ_FROM_PTR(&match_group_obj) };
|
||||||
|
match_locals_dict_table[1] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_groups), MP_OBJ_FROM_PTR(&match_groups_obj) };
|
||||||
|
match_locals_dict_table[2] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_span), MP_OBJ_FROM_PTR(&match_span_obj) };
|
||||||
|
match_locals_dict_table[3] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_start), MP_OBJ_FROM_PTR(&match_start_obj) };
|
||||||
|
match_locals_dict_table[4] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_end), MP_OBJ_FROM_PTR(&match_end_obj) };
|
||||||
|
match_type.locals_dict = (void*)&match_locals_dict;
|
||||||
|
|
||||||
|
re_type.base.type = (void*)&mp_fun_table.type_type;
|
||||||
|
re_type.name = MP_QSTR_ure;
|
||||||
|
re_type.print = re_print;
|
||||||
|
re_locals_dict_table[0] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_match), MP_OBJ_FROM_PTR(&re_match_obj) };
|
||||||
|
re_locals_dict_table[1] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_search), MP_OBJ_FROM_PTR(&re_search_obj) };
|
||||||
|
re_locals_dict_table[2] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_split), MP_OBJ_FROM_PTR(&re_split_obj) };
|
||||||
|
re_type.locals_dict = (void*)&re_locals_dict;
|
||||||
|
|
||||||
|
mp_store_global(MP_QSTR_compile, MP_OBJ_FROM_PTR(&mod_re_compile_obj));
|
||||||
|
mp_store_global(MP_QSTR_match, MP_OBJ_FROM_PTR(&re_match_obj));
|
||||||
|
mp_store_global(MP_QSTR_search, MP_OBJ_FROM_PTR(&re_search_obj));
|
||||||
|
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
13
examples/natmod/uzlib/Makefile
Normal file
13
examples/natmod/uzlib/Makefile
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# Location of top-level MicroPython directory
|
||||||
|
MPY_DIR = ../../..
|
||||||
|
|
||||||
|
# Name of module (different to built-in uzlib so it can coexist)
|
||||||
|
MOD = uzlib_$(ARCH)
|
||||||
|
|
||||||
|
# Source files (.c or .py)
|
||||||
|
SRC = uzlib.c
|
||||||
|
|
||||||
|
# Architecture to build for (x86, x64, armv7m, xtensa, xtensawin)
|
||||||
|
ARCH = x64
|
||||||
|
|
||||||
|
include $(MPY_DIR)/py/dynruntime.mk
|
35
examples/natmod/uzlib/uzlib.c
Normal file
35
examples/natmod/uzlib/uzlib.c
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
#define MICROPY_PY_UZLIB (1)
|
||||||
|
|
||||||
|
#include "py/dynruntime.h"
|
||||||
|
|
||||||
|
#if !defined(__linux__)
|
||||||
|
void *memset(void *s, int c, size_t n) {
|
||||||
|
return mp_fun_table.memset_(s, c, n);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
mp_obj_type_t decompio_type;
|
||||||
|
|
||||||
|
#include "extmod/moduzlib.c"
|
||||||
|
|
||||||
|
mp_map_elem_t decompio_locals_dict_table[3];
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(decompio_locals_dict, decompio_locals_dict_table);
|
||||||
|
|
||||||
|
mp_obj_t mpy_init(mp_obj_fun_bc_t *self, size_t n_args, size_t n_kw, mp_obj_t *args) {
|
||||||
|
MP_DYNRUNTIME_INIT_ENTRY
|
||||||
|
|
||||||
|
decompio_type.base.type = mp_fun_table.type_type;
|
||||||
|
decompio_type.name = MP_QSTR_DecompIO;
|
||||||
|
decompio_type.make_new = decompio_make_new;
|
||||||
|
decompio_type.protocol = &decompio_stream_p;
|
||||||
|
decompio_locals_dict_table[0] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_read), MP_OBJ_FROM_PTR(&mp_stream_read_obj) };
|
||||||
|
decompio_locals_dict_table[1] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_readinto), MP_OBJ_FROM_PTR(&mp_stream_readinto_obj) };
|
||||||
|
decompio_locals_dict_table[2] = (mp_map_elem_t){ MP_OBJ_NEW_QSTR(MP_QSTR_readline), MP_OBJ_FROM_PTR(&mp_stream_unbuffered_readline_obj) };
|
||||||
|
decompio_type.locals_dict = (void*)&decompio_locals_dict;
|
||||||
|
|
||||||
|
mp_store_global(MP_QSTR___name__, MP_OBJ_NEW_QSTR(MP_QSTR_uzlib));
|
||||||
|
mp_store_global(MP_QSTR_decompress, MP_OBJ_FROM_PTR(&mod_uzlib_decompress_obj));
|
||||||
|
mp_store_global(MP_QSTR_DecompIO, MP_OBJ_FROM_PTR(&decompio_type));
|
||||||
|
|
||||||
|
MP_DYNRUNTIME_INIT_EXIT
|
||||||
|
}
|
34
examples/usercmodule/cexample/examplemodule.c
Normal file
34
examples/usercmodule/cexample/examplemodule.c
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
// Include MicroPython API.
|
||||||
|
#include "py/runtime.h"
|
||||||
|
|
||||||
|
// This is the function which will be called from Python as cexample.add_ints(a, b).
|
||||||
|
STATIC mp_obj_t example_add_ints(mp_obj_t a_obj, mp_obj_t b_obj) {
|
||||||
|
// Extract the ints from the micropython input objects.
|
||||||
|
int a = mp_obj_get_int(a_obj);
|
||||||
|
int b = mp_obj_get_int(b_obj);
|
||||||
|
|
||||||
|
// Calculate the addition and convert to MicroPython object.
|
||||||
|
return mp_obj_new_int(a + b);
|
||||||
|
}
|
||||||
|
// Define a Python reference to the function above.
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(example_add_ints_obj, example_add_ints);
|
||||||
|
|
||||||
|
// Define all properties of the module.
|
||||||
|
// Table entries are key/value pairs of the attribute name (a string)
|
||||||
|
// and the MicroPython object reference.
|
||||||
|
// All identifiers and strings are written as MP_QSTR_xxx and will be
|
||||||
|
// optimized to word-sized integers by the build system (interned strings).
|
||||||
|
STATIC const mp_rom_map_elem_t example_module_globals_table[] = {
|
||||||
|
{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_cexample) },
|
||||||
|
{ MP_ROM_QSTR(MP_QSTR_add_ints), MP_ROM_PTR(&example_add_ints_obj) },
|
||||||
|
};
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(example_module_globals, example_module_globals_table);
|
||||||
|
|
||||||
|
// Define module object.
|
||||||
|
const mp_obj_module_t example_user_cmodule = {
|
||||||
|
.base = { &mp_type_module },
|
||||||
|
.globals = (mp_obj_dict_t *)&example_module_globals,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Register the module to make it available in Python.
|
||||||
|
MP_REGISTER_MODULE(MP_QSTR_cexample, example_user_cmodule, MODULE_CEXAMPLE_ENABLED);
|
9
examples/usercmodule/cexample/micropython.mk
Normal file
9
examples/usercmodule/cexample/micropython.mk
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
EXAMPLE_MOD_DIR := $(USERMOD_DIR)
|
||||||
|
|
||||||
|
# Add all C files to SRC_USERMOD.
|
||||||
|
SRC_USERMOD += $(EXAMPLE_MOD_DIR)/examplemodule.c
|
||||||
|
|
||||||
|
# We can add our module folder to include paths if needed
|
||||||
|
# This is not actually needed in this example.
|
||||||
|
CFLAGS_USERMOD += -I$(EXAMPLE_MOD_DIR)
|
||||||
|
CEXAMPLE_MOD_DIR := $(USERMOD_DIR)
|
17
examples/usercmodule/cppexample/example.cpp
Normal file
17
examples/usercmodule/cppexample/example.cpp
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
extern "C" {
|
||||||
|
#include <examplemodule.h>
|
||||||
|
|
||||||
|
// Here we implement the function using C++ code, but since it's
|
||||||
|
// declaration has to be compatible with C everything goes in extern "C" scope.
|
||||||
|
mp_obj_t cppfunc(mp_obj_t a_obj, mp_obj_t b_obj) {
|
||||||
|
// Prove we have (at least) C++11 features.
|
||||||
|
const auto a = mp_obj_get_int(a_obj);
|
||||||
|
const auto b = mp_obj_get_int(b_obj);
|
||||||
|
const auto sum = [&]() {
|
||||||
|
return mp_obj_new_int(a + b);
|
||||||
|
} ();
|
||||||
|
// Prove we're being scanned for QSTRs.
|
||||||
|
mp_obj_t tup[] = {sum, MP_ROM_QSTR(MP_QSTR_hellocpp)};
|
||||||
|
return mp_obj_new_tuple(2, tup);
|
||||||
|
}
|
||||||
|
}
|
25
examples/usercmodule/cppexample/examplemodule.c
Normal file
25
examples/usercmodule/cppexample/examplemodule.c
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#include <examplemodule.h>
|
||||||
|
|
||||||
|
// Define a Python reference to the function we'll make available.
|
||||||
|
// See example.cpp for the definition.
|
||||||
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(cppfunc_obj, cppfunc);
|
||||||
|
|
||||||
|
// Define all properties of the module.
|
||||||
|
// Table entries are key/value pairs of the attribute name (a string)
|
||||||
|
// and the MicroPython object reference.
|
||||||
|
// All identifiers and strings are written as MP_QSTR_xxx and will be
|
||||||
|
// optimized to word-sized integers by the build system (interned strings).
|
||||||
|
STATIC const mp_rom_map_elem_t cppexample_module_globals_table[] = {
|
||||||
|
{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_cppexample) },
|
||||||
|
{ MP_ROM_QSTR(MP_QSTR_cppfunc), MP_ROM_PTR(&cppfunc_obj) },
|
||||||
|
};
|
||||||
|
STATIC MP_DEFINE_CONST_DICT(cppexample_module_globals, cppexample_module_globals_table);
|
||||||
|
|
||||||
|
// Define module object.
|
||||||
|
const mp_obj_module_t cppexample_user_cmodule = {
|
||||||
|
.base = { &mp_type_module },
|
||||||
|
.globals = (mp_obj_dict_t *)&cppexample_module_globals,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Register the module to make it available in Python.
|
||||||
|
MP_REGISTER_MODULE(MP_QSTR_cppexample, cppexample_user_cmodule, MODULE_CPPEXAMPLE_ENABLED);
|
5
examples/usercmodule/cppexample/examplemodule.h
Normal file
5
examples/usercmodule/cppexample/examplemodule.h
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
// Include MicroPython API.
|
||||||
|
#include "py/runtime.h"
|
||||||
|
|
||||||
|
// Declare the function we'll make available in Python as cppexample.cppfunc().
|
||||||
|
extern mp_obj_t cppfunc(mp_obj_t a_obj, mp_obj_t b_obj);
|
12
examples/usercmodule/cppexample/micropython.mk
Normal file
12
examples/usercmodule/cppexample/micropython.mk
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
CPPEXAMPLE_MOD_DIR := $(USERMOD_DIR)
|
||||||
|
|
||||||
|
# Add our source files to the respective variables.
|
||||||
|
SRC_USERMOD += $(CPPEXAMPLE_MOD_DIR)/examplemodule.c
|
||||||
|
SRC_USERMOD_CXX += $(CPPEXAMPLE_MOD_DIR)/example.cpp
|
||||||
|
|
||||||
|
# Add our module directory to the include path.
|
||||||
|
CFLAGS_USERMOD += -I$(CPPEXAMPLE_MOD_DIR)
|
||||||
|
CXXFLAGS_USERMOD += -I$(CPPEXAMPLE_MOD_DIR)
|
||||||
|
|
||||||
|
# We use C++ features so have to link against the standard library.
|
||||||
|
LDFLAGS_USERMOD += -lstdc++
|
117
extmod/axtls-include/config.h
Normal file
117
extmod/axtls-include/config.h
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
/*
|
||||||
|
* Automatically generated header file: don't edit
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define HAVE_DOT_CONFIG 1
|
||||||
|
#define CONFIG_PLATFORM_LINUX 1
|
||||||
|
#undef CONFIG_PLATFORM_CYGWIN
|
||||||
|
#undef CONFIG_PLATFORM_WIN32
|
||||||
|
|
||||||
|
/*
|
||||||
|
* General Configuration
|
||||||
|
*/
|
||||||
|
#define PREFIX "/usr/local"
|
||||||
|
#undef CONFIG_DEBUG
|
||||||
|
#undef CONFIG_STRIP_UNWANTED_SECTIONS
|
||||||
|
#undef CONFIG_VISUAL_STUDIO_7_0
|
||||||
|
#undef CONFIG_VISUAL_STUDIO_8_0
|
||||||
|
#undef CONFIG_VISUAL_STUDIO_10_0
|
||||||
|
#define CONFIG_VISUAL_STUDIO_7_0_BASE ""
|
||||||
|
#define CONFIG_VISUAL_STUDIO_8_0_BASE ""
|
||||||
|
#define CONFIG_VISUAL_STUDIO_10_0_BASE ""
|
||||||
|
#define CONFIG_EXTRA_CFLAGS_OPTIONS ""
|
||||||
|
#define CONFIG_EXTRA_LDFLAGS_OPTIONS ""
|
||||||
|
|
||||||
|
/*
|
||||||
|
* SSL Library
|
||||||
|
*/
|
||||||
|
#undef CONFIG_SSL_SERVER_ONLY
|
||||||
|
#undef CONFIG_SSL_CERT_VERIFICATION
|
||||||
|
#undef CONFIG_SSL_FULL_MODE
|
||||||
|
#define CONFIG_SSL_SKELETON_MODE 1
|
||||||
|
#define CONFIG_SSL_ENABLE_SERVER 1
|
||||||
|
#define CONFIG_SSL_ENABLE_CLIENT 1
|
||||||
|
#undef CONFIG_SSL_DIAGNOSTICS
|
||||||
|
#define CONFIG_SSL_PROT_LOW 1
|
||||||
|
#undef CONFIG_SSL_PROT_MEDIUM
|
||||||
|
#undef CONFIG_SSL_PROT_HIGH
|
||||||
|
#define CONFIG_SSL_AES 1
|
||||||
|
#define CONFIG_SSL_USE_DEFAULT_KEY 1
|
||||||
|
#define CONFIG_SSL_PRIVATE_KEY_LOCATION ""
|
||||||
|
#define CONFIG_SSL_PRIVATE_KEY_PASSWORD ""
|
||||||
|
#define CONFIG_SSL_X509_CERT_LOCATION ""
|
||||||
|
#undef CONFIG_SSL_GENERATE_X509_CERT
|
||||||
|
#define CONFIG_SSL_X509_COMMON_NAME ""
|
||||||
|
#define CONFIG_SSL_X509_ORGANIZATION_NAME ""
|
||||||
|
#define CONFIG_SSL_X509_ORGANIZATION_UNIT_NAME ""
|
||||||
|
#undef CONFIG_SSL_HAS_PEM
|
||||||
|
#undef CONFIG_SSL_USE_PKCS12
|
||||||
|
#define CONFIG_SSL_EXPIRY_TIME
|
||||||
|
#define CONFIG_X509_MAX_CA_CERTS 0
|
||||||
|
#define CONFIG_SSL_MAX_CERTS 3
|
||||||
|
#undef CONFIG_SSL_CTX_MUTEXING
|
||||||
|
#undef CONFIG_USE_DEV_URANDOM
|
||||||
|
#undef CONFIG_WIN32_USE_CRYPTO_LIB
|
||||||
|
#undef CONFIG_OPENSSL_COMPATIBLE
|
||||||
|
#undef CONFIG_PERFORMANCE_TESTING
|
||||||
|
#undef CONFIG_SSL_TEST
|
||||||
|
#undef CONFIG_AXTLSWRAP
|
||||||
|
#undef CONFIG_AXHTTPD
|
||||||
|
#undef CONFIG_HTTP_STATIC_BUILD
|
||||||
|
#define CONFIG_HTTP_PORT
|
||||||
|
#define CONFIG_HTTP_HTTPS_PORT
|
||||||
|
#define CONFIG_HTTP_SESSION_CACHE_SIZE
|
||||||
|
#define CONFIG_HTTP_WEBROOT ""
|
||||||
|
#define CONFIG_HTTP_TIMEOUT
|
||||||
|
#undef CONFIG_HTTP_HAS_CGI
|
||||||
|
#define CONFIG_HTTP_CGI_EXTENSIONS ""
|
||||||
|
#undef CONFIG_HTTP_ENABLE_LUA
|
||||||
|
#define CONFIG_HTTP_LUA_PREFIX ""
|
||||||
|
#undef CONFIG_HTTP_BUILD_LUA
|
||||||
|
#define CONFIG_HTTP_CGI_LAUNCHER ""
|
||||||
|
#undef CONFIG_HTTP_DIRECTORIES
|
||||||
|
#undef CONFIG_HTTP_HAS_AUTHORIZATION
|
||||||
|
#undef CONFIG_HTTP_HAS_IPV6
|
||||||
|
#undef CONFIG_HTTP_ENABLE_DIFFERENT_USER
|
||||||
|
#define CONFIG_HTTP_USER ""
|
||||||
|
#undef CONFIG_HTTP_VERBOSE
|
||||||
|
#undef CONFIG_HTTP_IS_DAEMON
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Language Bindings
|
||||||
|
*/
|
||||||
|
#undef CONFIG_BINDINGS
|
||||||
|
#undef CONFIG_CSHARP_BINDINGS
|
||||||
|
#undef CONFIG_VBNET_BINDINGS
|
||||||
|
#define CONFIG_DOT_NET_FRAMEWORK_BASE ""
|
||||||
|
#undef CONFIG_JAVA_BINDINGS
|
||||||
|
#define CONFIG_JAVA_HOME ""
|
||||||
|
#undef CONFIG_PERL_BINDINGS
|
||||||
|
#define CONFIG_PERL_CORE ""
|
||||||
|
#define CONFIG_PERL_LIB ""
|
||||||
|
#undef CONFIG_LUA_BINDINGS
|
||||||
|
#define CONFIG_LUA_CORE ""
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Samples
|
||||||
|
*/
|
||||||
|
#undef CONFIG_SAMPLES
|
||||||
|
#undef CONFIG_C_SAMPLES
|
||||||
|
#undef CONFIG_CSHARP_SAMPLES
|
||||||
|
#undef CONFIG_VBNET_SAMPLES
|
||||||
|
#undef CONFIG_JAVA_SAMPLES
|
||||||
|
#undef CONFIG_PERL_SAMPLES
|
||||||
|
#undef CONFIG_LUA_SAMPLES
|
||||||
|
#undef CONFIG_BIGINT_CLASSICAL
|
||||||
|
#undef CONFIG_BIGINT_MONTGOMERY
|
||||||
|
#undef CONFIG_BIGINT_BARRETT
|
||||||
|
#undef CONFIG_BIGINT_CRT
|
||||||
|
#undef CONFIG_BIGINT_KARATSUBA
|
||||||
|
#define MUL_KARATSUBA_THRESH
|
||||||
|
#define SQU_KARATSUBA_THRESH
|
||||||
|
#undef CONFIG_BIGINT_SLIDING_WINDOW
|
||||||
|
#undef CONFIG_BIGINT_SQUARE
|
||||||
|
#undef CONFIG_BIGINT_CHECK_ON
|
||||||
|
#undef CONFIG_INTEGER_32BIT
|
||||||
|
#undef CONFIG_INTEGER_16BIT
|
||||||
|
#undef CONFIG_INTEGER_8BIT
|
1
extmod/axtls-include/version.h
Normal file
1
extmod/axtls-include/version.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#define AXTLS_VERSION "(no version)"
|
@ -1,7 +1,8 @@
|
|||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
|
* Source: https://github.com/B-Con/crypto-algorithms
|
||||||
* Filename: sha256.c
|
* Filename: sha256.c
|
||||||
* Author: Brad Conte (brad AT bradconte.com)
|
* Author: Brad Conte (brad AT bradconte.com)
|
||||||
* Copyright:
|
* Copyright: This code is released into the public domain.
|
||||||
* Disclaimer: This code is presented "as is" without any guarantees.
|
* Disclaimer: This code is presented "as is" without any guarantees.
|
||||||
* Details: Implementation of the SHA-256 hashing algorithm.
|
* Details: Implementation of the SHA-256 hashing algorithm.
|
||||||
SHA-256 is one of the three algorithms in the SHA2
|
SHA-256 is one of the three algorithms in the SHA2
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
/*********************************************************************
|
/*********************************************************************
|
||||||
|
* Source: https://github.com/B-Con/crypto-algorithms
|
||||||
* Filename: sha256.h
|
* Filename: sha256.h
|
||||||
* Author: Brad Conte (brad AT bradconte.com)
|
* Author: Brad Conte (brad AT bradconte.com)
|
||||||
* Copyright:
|
* Copyright: This code is released into the public domain.
|
||||||
* Disclaimer: This code is presented "as is" without any guarantees.
|
* Disclaimer: This code is presented "as is" without any guarantees.
|
||||||
* Details: Defines the API for the corresponding SHA1 implementation.
|
* Details: Defines the API for the corresponding SHA1 implementation.
|
||||||
*********************************************************************/
|
*********************************************************************/
|
||||||
|
93
extmod/extmod.cmake
Normal file
93
extmod/extmod.cmake
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
# CMake fragment for MicroPython extmod component
|
||||||
|
|
||||||
|
set(MICROPY_EXTMOD_DIR "${MICROPY_DIR}/extmod")
|
||||||
|
set(MICROPY_OOFATFS_DIR "${MICROPY_DIR}/lib/oofatfs")
|
||||||
|
|
||||||
|
set(MICROPY_SOURCE_EXTMOD
|
||||||
|
${MICROPY_DIR}/lib/embed/abort_.c
|
||||||
|
${MICROPY_DIR}/lib/utils/printf.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/machine_i2c.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/machine_mem.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/machine_pulse.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/machine_signal.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/machine_spi.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modbluetooth.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modbtree.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modframebuf.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modonewire.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moduasyncio.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modubinascii.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moducryptolib.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moductypes.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moduhashlib.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moduheapq.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modujson.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modurandom.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modure.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moduselect.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modussl_axtls.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modussl_mbedtls.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modutimeq.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moduwebsocket.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/moduzlib.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/modwebrepl.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/uos_dupterm.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/utime_mphal.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_blockdev.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_fat.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_fat_diskio.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_fat_file.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_lfs.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_posix.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_posix_file.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/vfs_reader.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/virtpin.c
|
||||||
|
${MICROPY_EXTMOD_DIR}/nimble/modbluetooth_nimble.c
|
||||||
|
)
|
||||||
|
|
||||||
|
# Library for btree module and associated code
|
||||||
|
|
||||||
|
set(MICROPY_LIB_BERKELEY_DIR "${MICROPY_DIR}/lib/berkeley-db-1.xx")
|
||||||
|
|
||||||
|
if(EXISTS "${MICROPY_LIB_BERKELEY_DIR}/btree/bt_close.c")
|
||||||
|
add_library(micropy_extmod_btree OBJECT
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_close.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_conv.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_debug.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_delete.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_get.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_open.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_overflow.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_page.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_put.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_search.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_seq.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_split.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/btree/bt_utils.c
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/mpool/mpool.c
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(micropy_extmod_btree PRIVATE
|
||||||
|
${MICROPY_LIB_BERKELEY_DIR}/PORT/include
|
||||||
|
)
|
||||||
|
|
||||||
|
target_compile_definitions(micropy_extmod_btree PRIVATE
|
||||||
|
__DBINTERFACE_PRIVATE=1
|
||||||
|
mpool_error=printf
|
||||||
|
abort=abort_
|
||||||
|
"virt_fd_t=void*"
|
||||||
|
)
|
||||||
|
|
||||||
|
# The include directories and compile definitions below are needed to build
|
||||||
|
# modbtree.c and should be added to the main MicroPython target.
|
||||||
|
|
||||||
|
list(APPEND MICROPY_INC_CORE
|
||||||
|
"${MICROPY_LIB_BERKELEY_DIR}/PORT/include"
|
||||||
|
)
|
||||||
|
|
||||||
|
list(APPEND MICROPY_DEF_CORE
|
||||||
|
__DBINTERFACE_PRIVATE=1
|
||||||
|
"virt_fd_t=void*"
|
||||||
|
)
|
||||||
|
endif()
|
234
extmod/extmod.mk
Normal file
234
extmod/extmod.mk
Normal file
@ -0,0 +1,234 @@
|
|||||||
|
# This makefile fragment provides rules to build 3rd-party components for extmod modules
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# VFS FAT FS
|
||||||
|
|
||||||
|
OOFATFS_DIR = lib/oofatfs
|
||||||
|
|
||||||
|
# this sets the config file for FatFs
|
||||||
|
CFLAGS_MOD += -DFFCONF_H=\"$(OOFATFS_DIR)/ffconf.h\"
|
||||||
|
|
||||||
|
ifeq ($(MICROPY_VFS_FAT),1)
|
||||||
|
CFLAGS_MOD += -DMICROPY_VFS_FAT=1
|
||||||
|
SRC_MOD += $(addprefix $(OOFATFS_DIR)/,\
|
||||||
|
ff.c \
|
||||||
|
ffunicode.c \
|
||||||
|
)
|
||||||
|
endif
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# VFS littlefs
|
||||||
|
|
||||||
|
LITTLEFS_DIR = lib/littlefs
|
||||||
|
|
||||||
|
ifeq ($(MICROPY_VFS_LFS1),1)
|
||||||
|
CFLAGS_MOD += -DMICROPY_VFS_LFS1=1
|
||||||
|
CFLAGS_MOD += -DLFS1_NO_MALLOC -DLFS1_NO_DEBUG -DLFS1_NO_WARN -DLFS1_NO_ERROR -DLFS1_NO_ASSERT
|
||||||
|
SRC_MOD += $(addprefix $(LITTLEFS_DIR)/,\
|
||||||
|
lfs1.c \
|
||||||
|
lfs1_util.c \
|
||||||
|
)
|
||||||
|
else
|
||||||
|
CFLAGS_MOD += -DMICROPY_VFS_LFS1=0
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(MICROPY_VFS_LFS2),1)
|
||||||
|
CFLAGS_MOD += -DMICROPY_VFS_LFS2=1
|
||||||
|
CFLAGS_MOD += -DLFS2_NO_MALLOC -DLFS2_NO_DEBUG -DLFS2_NO_WARN -DLFS2_NO_ERROR -DLFS2_NO_ASSERT
|
||||||
|
SRC_MOD += $(addprefix $(LITTLEFS_DIR)/,\
|
||||||
|
lfs2.c \
|
||||||
|
lfs2_util.c \
|
||||||
|
)
|
||||||
|
else
|
||||||
|
CFLAGS_MOD += -DMICROPY_VFS_LFS2=0
|
||||||
|
|
||||||
|
$(BUILD)/$(LITTLEFS_DIR)/lfs2.o: CFLAGS += -Wno-missing-field-initializers
|
||||||
|
endif
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# ussl
|
||||||
|
|
||||||
|
ifeq ($(MICROPY_PY_USSL),1)
|
||||||
|
CFLAGS_MOD += -DMICROPY_PY_USSL=1
|
||||||
|
ifeq ($(MICROPY_SSL_AXTLS),1)
|
||||||
|
CFLAGS_MOD += -DMICROPY_SSL_AXTLS=1 -I$(TOP)/lib/axtls/ssl -I$(TOP)/lib/axtls/crypto -I$(TOP)/extmod/axtls-include
|
||||||
|
AXTLS_DIR = lib/axtls
|
||||||
|
$(BUILD)/$(AXTLS_DIR)/%.o: CFLAGS += -Wno-all -Wno-unused-parameter -Wno-uninitialized -Wno-sign-compare -Wno-old-style-definition -Dmp_stream_errno=errno $(AXTLS_DEFS_EXTRA)
|
||||||
|
SRC_MOD += $(addprefix $(AXTLS_DIR)/,\
|
||||||
|
ssl/asn1.c \
|
||||||
|
ssl/loader.c \
|
||||||
|
ssl/tls1.c \
|
||||||
|
ssl/tls1_svr.c \
|
||||||
|
ssl/tls1_clnt.c \
|
||||||
|
ssl/x509.c \
|
||||||
|
crypto/aes.c \
|
||||||
|
crypto/bigint.c \
|
||||||
|
crypto/crypto_misc.c \
|
||||||
|
crypto/hmac.c \
|
||||||
|
crypto/md5.c \
|
||||||
|
crypto/rsa.c \
|
||||||
|
crypto/sha1.c \
|
||||||
|
)
|
||||||
|
else ifeq ($(MICROPY_SSL_MBEDTLS),1)
|
||||||
|
MBEDTLS_DIR = lib/mbedtls
|
||||||
|
CFLAGS_MOD += -DMICROPY_SSL_MBEDTLS=1 -I$(TOP)/$(MBEDTLS_DIR)/include
|
||||||
|
SRC_MOD += $(addprefix $(MBEDTLS_DIR)/library/,\
|
||||||
|
aes.c \
|
||||||
|
aesni.c \
|
||||||
|
arc4.c \
|
||||||
|
asn1parse.c \
|
||||||
|
asn1write.c \
|
||||||
|
base64.c \
|
||||||
|
bignum.c \
|
||||||
|
blowfish.c \
|
||||||
|
camellia.c \
|
||||||
|
ccm.c \
|
||||||
|
certs.c \
|
||||||
|
chacha20.c \
|
||||||
|
chachapoly.c \
|
||||||
|
cipher.c \
|
||||||
|
cipher_wrap.c \
|
||||||
|
cmac.c \
|
||||||
|
ctr_drbg.c \
|
||||||
|
debug.c \
|
||||||
|
des.c \
|
||||||
|
dhm.c \
|
||||||
|
ecdh.c \
|
||||||
|
ecdsa.c \
|
||||||
|
ecjpake.c \
|
||||||
|
ecp.c \
|
||||||
|
ecp_curves.c \
|
||||||
|
entropy.c \
|
||||||
|
entropy_poll.c \
|
||||||
|
error.c \
|
||||||
|
gcm.c \
|
||||||
|
havege.c \
|
||||||
|
hmac_drbg.c \
|
||||||
|
md2.c \
|
||||||
|
md4.c \
|
||||||
|
md5.c \
|
||||||
|
md.c \
|
||||||
|
md_wrap.c \
|
||||||
|
oid.c \
|
||||||
|
padlock.c \
|
||||||
|
pem.c \
|
||||||
|
pk.c \
|
||||||
|
pkcs11.c \
|
||||||
|
pkcs12.c \
|
||||||
|
pkcs5.c \
|
||||||
|
pkparse.c \
|
||||||
|
pk_wrap.c \
|
||||||
|
pkwrite.c \
|
||||||
|
platform.c \
|
||||||
|
platform_util.c \
|
||||||
|
poly1305.c \
|
||||||
|
ripemd160.c \
|
||||||
|
rsa.c \
|
||||||
|
rsa_internal.c \
|
||||||
|
sha1.c \
|
||||||
|
sha256.c \
|
||||||
|
sha512.c \
|
||||||
|
ssl_cache.c \
|
||||||
|
ssl_ciphersuites.c \
|
||||||
|
ssl_cli.c \
|
||||||
|
ssl_cookie.c \
|
||||||
|
ssl_srv.c \
|
||||||
|
ssl_ticket.c \
|
||||||
|
ssl_tls.c \
|
||||||
|
timing.c \
|
||||||
|
x509.c \
|
||||||
|
x509_create.c \
|
||||||
|
x509_crl.c \
|
||||||
|
x509_crt.c \
|
||||||
|
x509_csr.c \
|
||||||
|
x509write_crt.c \
|
||||||
|
x509write_csr.c \
|
||||||
|
xtea.c \
|
||||||
|
)
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# lwip
|
||||||
|
|
||||||
|
ifeq ($(MICROPY_PY_LWIP),1)
|
||||||
|
# A port should add an include path where lwipopts.h can be found (eg extmod/lwip-include)
|
||||||
|
LWIP_DIR = lib/lwip/src
|
||||||
|
INC += -I$(TOP)/$(LWIP_DIR)/include
|
||||||
|
CFLAGS_MOD += -DMICROPY_PY_LWIP=1
|
||||||
|
$(BUILD)/$(LWIP_DIR)/core/ipv4/dhcp.o: CFLAGS_MOD += -Wno-address
|
||||||
|
SRC_MOD += extmod/modlwip.c lib/netutils/netutils.c
|
||||||
|
SRC_MOD += $(addprefix $(LWIP_DIR)/,\
|
||||||
|
apps/mdns/mdns.c \
|
||||||
|
core/def.c \
|
||||||
|
core/dns.c \
|
||||||
|
core/inet_chksum.c \
|
||||||
|
core/init.c \
|
||||||
|
core/ip.c \
|
||||||
|
core/mem.c \
|
||||||
|
core/memp.c \
|
||||||
|
core/netif.c \
|
||||||
|
core/pbuf.c \
|
||||||
|
core/raw.c \
|
||||||
|
core/stats.c \
|
||||||
|
core/sys.c \
|
||||||
|
core/tcp.c \
|
||||||
|
core/tcp_in.c \
|
||||||
|
core/tcp_out.c \
|
||||||
|
core/timeouts.c \
|
||||||
|
core/udp.c \
|
||||||
|
core/ipv4/autoip.c \
|
||||||
|
core/ipv4/dhcp.c \
|
||||||
|
core/ipv4/etharp.c \
|
||||||
|
core/ipv4/icmp.c \
|
||||||
|
core/ipv4/igmp.c \
|
||||||
|
core/ipv4/ip4_addr.c \
|
||||||
|
core/ipv4/ip4.c \
|
||||||
|
core/ipv4/ip4_frag.c \
|
||||||
|
core/ipv6/dhcp6.c \
|
||||||
|
core/ipv6/ethip6.c \
|
||||||
|
core/ipv6/icmp6.c \
|
||||||
|
core/ipv6/inet6.c \
|
||||||
|
core/ipv6/ip6_addr.c \
|
||||||
|
core/ipv6/ip6.c \
|
||||||
|
core/ipv6/ip6_frag.c \
|
||||||
|
core/ipv6/mld6.c \
|
||||||
|
core/ipv6/nd6.c \
|
||||||
|
netif/ethernet.c \
|
||||||
|
)
|
||||||
|
ifeq ($(MICROPY_PY_LWIP_SLIP),1)
|
||||||
|
CFLAGS_MOD += -DMICROPY_PY_LWIP_SLIP=1
|
||||||
|
SRC_MOD += $(LWIP_DIR)/netif/slipif.c
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# btree
|
||||||
|
|
||||||
|
ifeq ($(MICROPY_PY_BTREE),1)
|
||||||
|
BTREE_DIR = lib/berkeley-db-1.xx
|
||||||
|
BTREE_DEFS = -D__DBINTERFACE_PRIVATE=1 -Dmpool_error=printf -Dabort=abort_ "-Dvirt_fd_t=void*" $(BTREE_DEFS_EXTRA)
|
||||||
|
INC += -I$(TOP)/$(BTREE_DIR)/PORT/include
|
||||||
|
SRC_MOD += extmod/modbtree.c
|
||||||
|
SRC_MOD += $(addprefix $(BTREE_DIR)/,\
|
||||||
|
btree/bt_close.c \
|
||||||
|
btree/bt_conv.c \
|
||||||
|
btree/bt_debug.c \
|
||||||
|
btree/bt_delete.c \
|
||||||
|
btree/bt_get.c \
|
||||||
|
btree/bt_open.c \
|
||||||
|
btree/bt_overflow.c \
|
||||||
|
btree/bt_page.c \
|
||||||
|
btree/bt_put.c \
|
||||||
|
btree/bt_search.c \
|
||||||
|
btree/bt_seq.c \
|
||||||
|
btree/bt_split.c \
|
||||||
|
btree/bt_utils.c \
|
||||||
|
mpool/mpool.c \
|
||||||
|
)
|
||||||
|
CFLAGS_MOD += -DMICROPY_PY_BTREE=1
|
||||||
|
# we need to suppress certain warnings to get berkeley-db to compile cleanly
|
||||||
|
# and we have separate BTREE_DEFS so the definitions don't interfere with other source code
|
||||||
|
$(BUILD)/$(BTREE_DIR)/%.o: CFLAGS += -Wno-old-style-definition -Wno-sign-compare -Wno-unused-parameter $(BTREE_DEFS)
|
||||||
|
$(BUILD)/extmod/modbtree.o: CFLAGS += $(BTREE_DEFS)
|
||||||
|
endif
|
@ -1,94 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
// SPDX-FileCopyrightText: Copyright (c) 2013, 2014 Damien P. George
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#include "py/runtime.h"
|
|
||||||
#include "extmod/machine_mem.h"
|
|
||||||
|
|
||||||
#if MICROPY_PY_MACHINE
|
|
||||||
|
|
||||||
// If you wish to override the functions for mapping the machine_mem read/write
|
|
||||||
// address, then add a #define for MICROPY_MACHINE_MEM_GET_READ_ADDR and/or
|
|
||||||
// MICROPY_MACHINE_MEM_GET_WRITE_ADDR in your mpconfigport.h. Since the
|
|
||||||
// prototypes are identical, it is allowable for both of the macros to evaluate
|
|
||||||
// the to same function.
|
|
||||||
//
|
|
||||||
// It is expected that the modmachine.c file for a given port will provide the
|
|
||||||
// implementations, if the default implementation isn't used.
|
|
||||||
|
|
||||||
#if !defined(MICROPY_MACHINE_MEM_GET_READ_ADDR) || !defined(MICROPY_MACHINE_MEM_GET_WRITE_ADDR)
|
|
||||||
STATIC uintptr_t machine_mem_get_addr(mp_obj_t addr_o, uint align) {
|
|
||||||
uintptr_t addr = mp_obj_int_get_truncated(addr_o);
|
|
||||||
if ((addr & (align - 1)) != 0) {
|
|
||||||
mp_raise_ValueError_varg(translate("address %08x is not aligned to %d bytes"), addr, align);
|
|
||||||
}
|
|
||||||
return addr;
|
|
||||||
}
|
|
||||||
#if !defined(MICROPY_MACHINE_MEM_GET_READ_ADDR)
|
|
||||||
#define MICROPY_MACHINE_MEM_GET_READ_ADDR machine_mem_get_addr
|
|
||||||
#endif
|
|
||||||
#if !defined(MICROPY_MACHINE_MEM_GET_WRITE_ADDR)
|
|
||||||
#define MICROPY_MACHINE_MEM_GET_WRITE_ADDR machine_mem_get_addr
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
STATIC void machine_mem_print(const mp_print_t *print, mp_obj_t self_in, mp_print_kind_t kind) {
|
|
||||||
(void)kind;
|
|
||||||
machine_mem_obj_t *self = MP_OBJ_TO_PTR(self_in);
|
|
||||||
mp_printf(print, "<%u-bit memory>", 8 * self->elem_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC mp_obj_t machine_mem_subscr(mp_obj_t self_in, mp_obj_t index, mp_obj_t value) {
|
|
||||||
// TODO support slice index to read/write multiple values at once
|
|
||||||
machine_mem_obj_t *self = MP_OBJ_TO_PTR(self_in);
|
|
||||||
if (value == MP_OBJ_NULL) {
|
|
||||||
// delete
|
|
||||||
return MP_OBJ_NULL; // op not supported
|
|
||||||
} else if (value == MP_OBJ_SENTINEL) {
|
|
||||||
// load
|
|
||||||
uintptr_t addr = MICROPY_MACHINE_MEM_GET_READ_ADDR(index, self->elem_size);
|
|
||||||
uint32_t val;
|
|
||||||
switch (self->elem_size) {
|
|
||||||
case 1:
|
|
||||||
val = (*(uint8_t *)addr);
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
val = (*(uint16_t *)addr);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
val = (*(uint32_t *)addr);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return mp_obj_new_int(val);
|
|
||||||
} else {
|
|
||||||
// store
|
|
||||||
uintptr_t addr = MICROPY_MACHINE_MEM_GET_WRITE_ADDR(index, self->elem_size);
|
|
||||||
uint32_t val = mp_obj_get_int_truncated(value);
|
|
||||||
switch (self->elem_size) {
|
|
||||||
case 1:
|
|
||||||
(*(uint8_t *)addr) = val;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
(*(uint16_t *)addr) = val;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
(*(uint32_t *)addr) = val;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return mp_const_none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const mp_obj_type_t machine_mem_type = {
|
|
||||||
{ &mp_type_type },
|
|
||||||
.name = MP_QSTR_mem,
|
|
||||||
.print = machine_mem_print,
|
|
||||||
.subscr = machine_mem_subscr,
|
|
||||||
};
|
|
||||||
|
|
||||||
const machine_mem_obj_t machine_mem8_obj = {{&machine_mem_type}, 1};
|
|
||||||
const machine_mem_obj_t machine_mem16_obj = {{&machine_mem_type}, 2};
|
|
||||||
const machine_mem_obj_t machine_mem32_obj = {{&machine_mem_type}, 4};
|
|
||||||
|
|
||||||
#endif // MICROPY_PY_MACHINE
|
|
@ -1,29 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
// SPDX-FileCopyrightText: Copyright (c) 2015 Damien P. George
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#ifndef MICROPY_INCLUDED_EXTMOD_MACHINE_MEM_H
|
|
||||||
#define MICROPY_INCLUDED_EXTMOD_MACHINE_MEM_H
|
|
||||||
|
|
||||||
#include "py/obj.h"
|
|
||||||
|
|
||||||
typedef struct _machine_mem_obj_t {
|
|
||||||
mp_obj_base_t base;
|
|
||||||
unsigned elem_size; // in bytes
|
|
||||||
} machine_mem_obj_t;
|
|
||||||
|
|
||||||
extern const mp_obj_type_t machine_mem_type;
|
|
||||||
|
|
||||||
extern const machine_mem_obj_t machine_mem8_obj;
|
|
||||||
extern const machine_mem_obj_t machine_mem16_obj;
|
|
||||||
extern const machine_mem_obj_t machine_mem32_obj;
|
|
||||||
|
|
||||||
#if defined(MICROPY_MACHINE_MEM_GET_READ_ADDR)
|
|
||||||
uintptr_t MICROPY_MACHINE_MEM_GET_READ_ADDR(mp_obj_t addr_o, uint align);
|
|
||||||
#endif
|
|
||||||
#if defined(MICROPY_MACHINE_MEM_GET_WRITE_ADDR)
|
|
||||||
uintptr_t MICROPY_MACHINE_MEM_GET_WRITE_ADDR(mp_obj_t addr_o, uint align);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif // MICROPY_INCLUDED_EXTMOD_MACHINE_MEM_H
|
|
@ -1,67 +0,0 @@
|
|||||||
// Copyright (c) 2016 Paul Sokolovsky
|
|
||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#include "py/mpconfig.h"
|
|
||||||
#if MICROPY_PY_MACHINE
|
|
||||||
|
|
||||||
#include "py/obj.h"
|
|
||||||
#include "py/runtime.h"
|
|
||||||
#include "extmod/virtpin.h"
|
|
||||||
#include "extmod/machine_pinbase.h"
|
|
||||||
|
|
||||||
// PinBase class
|
|
||||||
|
|
||||||
// As this is abstract class, its instance is null.
|
|
||||||
// But there should be an instance, as the rest of instance code
|
|
||||||
// expects that there will be concrete object for inheritance.
|
|
||||||
typedef struct _mp_pinbase_t {
|
|
||||||
mp_obj_base_t base;
|
|
||||||
} mp_pinbase_t;
|
|
||||||
|
|
||||||
STATIC const mp_pinbase_t pinbase_singleton = {
|
|
||||||
.base = { &machine_pinbase_type },
|
|
||||||
};
|
|
||||||
|
|
||||||
STATIC mp_obj_t pinbase_make_new(const mp_obj_type_t *type, size_t n_args, const mp_obj_t *args, mp_map_t *kw_args) {
|
|
||||||
(void)type;
|
|
||||||
(void)n_args;
|
|
||||||
(void)args;
|
|
||||||
(void)kw_args;
|
|
||||||
return MP_OBJ_FROM_PTR(&pinbase_singleton);
|
|
||||||
}
|
|
||||||
|
|
||||||
mp_uint_t pinbase_ioctl(mp_obj_t obj, mp_uint_t request, uintptr_t arg, int *errcode);
|
|
||||||
mp_uint_t pinbase_ioctl(mp_obj_t obj, mp_uint_t request, uintptr_t arg, int *errcode) {
|
|
||||||
(void)errcode;
|
|
||||||
switch (request) {
|
|
||||||
case MP_PIN_READ: {
|
|
||||||
mp_obj_t dest[2];
|
|
||||||
mp_load_method(obj, MP_QSTR_value, dest);
|
|
||||||
return mp_obj_get_int(mp_call_method_n_kw(0, 0, dest));
|
|
||||||
}
|
|
||||||
case MP_PIN_WRITE: {
|
|
||||||
mp_obj_t dest[3];
|
|
||||||
mp_load_method(obj, MP_QSTR_value, dest);
|
|
||||||
dest[2] = (arg == 0 ? mp_const_false : mp_const_true);
|
|
||||||
mp_call_method_n_kw(1, 0, dest);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC const mp_pin_p_t pinbase_pin_p = {
|
|
||||||
MP_PROTO_IMPLEMENT(MP_QSTR_protocol_pin)
|
|
||||||
.ioctl = pinbase_ioctl,
|
|
||||||
};
|
|
||||||
|
|
||||||
const mp_obj_type_t machine_pinbase_type = {
|
|
||||||
{ &mp_type_type },
|
|
||||||
.name = MP_QSTR_PinBase,
|
|
||||||
.make_new = pinbase_make_new,
|
|
||||||
.protocol = &pinbase_pin_p,
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // MICROPY_PY_MACHINE
|
|
@ -1,13 +0,0 @@
|
|||||||
// Copyright (c) 2016 Paul Sokolovsky
|
|
||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#ifndef MICROPY_INCLUDED_EXTMOD_MACHINE_PINBASE_H
|
|
||||||
#define MICROPY_INCLUDED_EXTMOD_MACHINE_PINBASE_H
|
|
||||||
|
|
||||||
#include "py/obj.h"
|
|
||||||
|
|
||||||
extern const mp_obj_type_t machine_pinbase_type;
|
|
||||||
|
|
||||||
#endif // MICROPY_INCLUDED_EXTMOD_MACHINE_PINBASE_H
|
|
@ -1,44 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
// SPDX-FileCopyrightText: Copyright (c) 2016 Damien P. George
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#include "py/runtime.h"
|
|
||||||
#include "py/mperrno.h"
|
|
||||||
#include "extmod/machine_pulse.h"
|
|
||||||
|
|
||||||
#if MICROPY_PY_MACHINE_PULSE
|
|
||||||
|
|
||||||
mp_uint_t machine_time_pulse_us(mp_hal_pin_obj_t pin, int pulse_level, mp_uint_t timeout_us) {
|
|
||||||
mp_uint_t start = mp_hal_ticks_us();
|
|
||||||
while (mp_hal_pin_read(pin) != pulse_level) {
|
|
||||||
if ((mp_uint_t)(mp_hal_ticks_us() - start) >= timeout_us) {
|
|
||||||
return (mp_uint_t)-2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
start = mp_hal_ticks_us();
|
|
||||||
while (mp_hal_pin_read(pin) == pulse_level) {
|
|
||||||
if ((mp_uint_t)(mp_hal_ticks_us() - start) >= timeout_us) {
|
|
||||||
return (mp_uint_t)-1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return mp_hal_ticks_us() - start;
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC mp_obj_t machine_time_pulse_us_(size_t n_args, const mp_obj_t *args) {
|
|
||||||
mp_hal_pin_obj_t pin = mp_hal_get_pin_obj(args[0]);
|
|
||||||
int level = 0;
|
|
||||||
if (mp_obj_is_true(args[1])) {
|
|
||||||
level = 1;
|
|
||||||
}
|
|
||||||
mp_uint_t timeout_us = 1000000;
|
|
||||||
if (n_args > 2) {
|
|
||||||
timeout_us = mp_obj_get_int(args[2]);
|
|
||||||
}
|
|
||||||
mp_uint_t us = machine_time_pulse_us(pin, level, timeout_us);
|
|
||||||
// May return -1 or -2 in case of timeout
|
|
||||||
return mp_obj_new_int(us);
|
|
||||||
}
|
|
||||||
MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(machine_time_pulse_us_obj, 2, 3, machine_time_pulse_us_);
|
|
||||||
|
|
||||||
#endif
|
|
@ -1,16 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
// SPDX-FileCopyrightText: Copyright (c) 2016 Damien P. George
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#ifndef MICROPY_INCLUDED_EXTMOD_MACHINE_PULSE_H
|
|
||||||
#define MICROPY_INCLUDED_EXTMOD_MACHINE_PULSE_H
|
|
||||||
|
|
||||||
#include "py/obj.h"
|
|
||||||
#include "py/mphal.h"
|
|
||||||
|
|
||||||
mp_uint_t machine_time_pulse_us(mp_hal_pin_obj_t pin, int pulse_level, mp_uint_t timeout_us);
|
|
||||||
|
|
||||||
MP_DECLARE_CONST_FUN_OBJ_VAR_BETWEEN(machine_time_pulse_us_obj);
|
|
||||||
|
|
||||||
#endif // MICROPY_INCLUDED_EXTMOD_MACHINE_PULSE_H
|
|
@ -1,159 +0,0 @@
|
|||||||
// Copyright (c) 2017 Paul Sokolovsky
|
|
||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#include "py/mpconfig.h"
|
|
||||||
#if MICROPY_PY_MACHINE
|
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#include "py/obj.h"
|
|
||||||
#include "py/runtime.h"
|
|
||||||
#include "extmod/virtpin.h"
|
|
||||||
#include "extmod/machine_signal.h"
|
|
||||||
|
|
||||||
// Signal class
|
|
||||||
|
|
||||||
typedef struct _machine_signal_t {
|
|
||||||
mp_obj_base_t base;
|
|
||||||
mp_obj_t pin;
|
|
||||||
bool invert;
|
|
||||||
} machine_signal_t;
|
|
||||||
|
|
||||||
STATIC mp_obj_t signal_make_new(const mp_obj_type_t *type, size_t n_args, const mp_obj_t *args, mp_map_t *kw_args) {
|
|
||||||
mp_obj_t pin = args[0];
|
|
||||||
bool invert = false;
|
|
||||||
|
|
||||||
#if defined(MICROPY_PY_MACHINE_PIN_MAKE_NEW)
|
|
||||||
mp_pin_p_t *pin_p = (mp_pin_t *)mp_proto_get(QSTR_pin_protocol, pin);
|
|
||||||
|
|
||||||
if (pin_p == NULL) {
|
|
||||||
// If first argument isn't a Pin-like object, we filter out "invert"
|
|
||||||
// from keyword arguments and pass them all to the exported Pin
|
|
||||||
// constructor to create one.
|
|
||||||
mp_obj_t *pin_args = mp_local_alloc((n_args + n_kw * 2) * sizeof(mp_obj_t));
|
|
||||||
memcpy(pin_args, args, n_args * sizeof(mp_obj_t));
|
|
||||||
const mp_obj_t *src = args + n_args;
|
|
||||||
mp_obj_t *dst = pin_args + n_args;
|
|
||||||
mp_obj_t *sig_value = NULL;
|
|
||||||
for (size_t cnt = n_kw; cnt; cnt--) {
|
|
||||||
if (*src == MP_OBJ_NEW_QSTR(MP_QSTR_invert)) {
|
|
||||||
invert = mp_obj_is_true(src[1]);
|
|
||||||
n_kw--;
|
|
||||||
} else {
|
|
||||||
*dst++ = *src;
|
|
||||||
*dst++ = src[1];
|
|
||||||
}
|
|
||||||
if (*src == MP_OBJ_NEW_QSTR(MP_QSTR_value)) {
|
|
||||||
// Value is pertained to Signal, so we should invert
|
|
||||||
// it for Pin if needed, and we should do it only when
|
|
||||||
// inversion status is guaranteedly known.
|
|
||||||
sig_value = dst - 1;
|
|
||||||
}
|
|
||||||
src += 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (invert && sig_value != NULL) {
|
|
||||||
*sig_value = mp_obj_is_true(*sig_value) ? MP_OBJ_NEW_SMALL_INT(0) : MP_OBJ_NEW_SMALL_INT(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Here we pass NULL as a type, hoping that mp_pin_make_new()
|
|
||||||
// will just ignore it as set a concrete type. If not, we'd need
|
|
||||||
// to expose port's "default" pin type too.
|
|
||||||
pin = MICROPY_PY_MACHINE_PIN_MAKE_NEW(NULL, n_args, n_kw, pin_args);
|
|
||||||
|
|
||||||
mp_local_free(pin_args);
|
|
||||||
} else
|
|
||||||
#endif
|
|
||||||
// Otherwise there should be 1 or 2 args
|
|
||||||
{
|
|
||||||
if (n_args == 1) {
|
|
||||||
if (kw_args == NULL || kw_args->used == 0) {
|
|
||||||
} else if (kw_args->used == 1 && kw_args->table[0].key == MP_OBJ_NEW_QSTR(MP_QSTR_invert)) {
|
|
||||||
invert = mp_obj_is_true(kw_args->table[0].value);
|
|
||||||
} else {
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
error:
|
|
||||||
mp_raise_TypeError(NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
machine_signal_t *o = m_new_obj(machine_signal_t);
|
|
||||||
o->base.type = type;
|
|
||||||
o->pin = pin;
|
|
||||||
o->invert = invert;
|
|
||||||
return MP_OBJ_FROM_PTR(o);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC mp_uint_t signal_ioctl(mp_obj_t self_in, mp_uint_t request, uintptr_t arg, int *errcode) {
|
|
||||||
(void)errcode;
|
|
||||||
machine_signal_t *self = MP_OBJ_TO_PTR(self_in);
|
|
||||||
|
|
||||||
switch (request) {
|
|
||||||
case MP_PIN_READ: {
|
|
||||||
return mp_virtual_pin_read(self->pin) ^ self->invert;
|
|
||||||
}
|
|
||||||
case MP_PIN_WRITE: {
|
|
||||||
mp_virtual_pin_write(self->pin, arg ^ self->invert);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// fast method for getting/setting signal value
|
|
||||||
STATIC mp_obj_t signal_call(mp_obj_t self_in, size_t n_args, size_t n_kw, const mp_obj_t *args) {
|
|
||||||
mp_arg_check_num_kw_array(n_args, n_kw, 0, 1, false);
|
|
||||||
if (n_args == 0) {
|
|
||||||
// get pin
|
|
||||||
return MP_OBJ_NEW_SMALL_INT(mp_virtual_pin_read(self_in));
|
|
||||||
} else {
|
|
||||||
// set pin
|
|
||||||
mp_virtual_pin_write(self_in, mp_obj_is_true(args[0]));
|
|
||||||
return mp_const_none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC mp_obj_t signal_value(size_t n_args, const mp_obj_t *args) {
|
|
||||||
return signal_call(args[0], n_args - 1, 0, args + 1);
|
|
||||||
}
|
|
||||||
STATIC MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(signal_value_obj, 1, 2, signal_value);
|
|
||||||
|
|
||||||
STATIC mp_obj_t signal_on(mp_obj_t self_in) {
|
|
||||||
mp_virtual_pin_write(self_in, 1);
|
|
||||||
return mp_const_none;
|
|
||||||
}
|
|
||||||
STATIC MP_DEFINE_CONST_FUN_OBJ_1(signal_on_obj, signal_on);
|
|
||||||
|
|
||||||
STATIC mp_obj_t signal_off(mp_obj_t self_in) {
|
|
||||||
mp_virtual_pin_write(self_in, 0);
|
|
||||||
return mp_const_none;
|
|
||||||
}
|
|
||||||
STATIC MP_DEFINE_CONST_FUN_OBJ_1(signal_off_obj, signal_off);
|
|
||||||
|
|
||||||
STATIC const mp_rom_map_elem_t signal_locals_dict_table[] = {
|
|
||||||
{ MP_ROM_QSTR(MP_QSTR_value), MP_ROM_PTR(&signal_value_obj) },
|
|
||||||
{ MP_ROM_QSTR(MP_QSTR_on), MP_ROM_PTR(&signal_on_obj) },
|
|
||||||
{ MP_ROM_QSTR(MP_QSTR_off), MP_ROM_PTR(&signal_off_obj) },
|
|
||||||
};
|
|
||||||
|
|
||||||
STATIC MP_DEFINE_CONST_DICT(signal_locals_dict, signal_locals_dict_table);
|
|
||||||
|
|
||||||
STATIC const mp_pin_p_t signal_pin_p = {
|
|
||||||
MP_PROTO_IMPLEMENT(MP_QSTR_protocol_pin)
|
|
||||||
.ioctl = signal_ioctl,
|
|
||||||
};
|
|
||||||
|
|
||||||
const mp_obj_type_t machine_signal_type = {
|
|
||||||
{ &mp_type_type },
|
|
||||||
.name = MP_QSTR_Signal,
|
|
||||||
.make_new = signal_make_new,
|
|
||||||
.call = signal_call,
|
|
||||||
.protocol = &signal_pin_p,
|
|
||||||
.locals_dict = (void *)&signal_locals_dict,
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // MICROPY_PY_MACHINE
|
|
@ -1,13 +0,0 @@
|
|||||||
// Copyright (c) 2017 Paul Sokolovsky
|
|
||||||
// SPDX-FileCopyrightText: 2014 MicroPython & CircuitPython contributors (https://github.com/adafruit/circuitpython/graphs/contributors)
|
|
||||||
//
|
|
||||||
// SPDX-License-Identifier: MIT
|
|
||||||
|
|
||||||
#ifndef MICROPY_INCLUDED_EXTMOD_MACHINE_SIGNAL_H
|
|
||||||
#define MICROPY_INCLUDED_EXTMOD_MACHINE_SIGNAL_H
|
|
||||||
|
|
||||||
#include "py/obj.h"
|
|
||||||
|
|
||||||
extern const mp_obj_type_t machine_signal_type;
|
|
||||||
|
|
||||||
#endif // MICROPY_INCLUDED_EXTMOD_MACHINE_SIGNAL_H
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user