Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Functions
System functions (time, random numbers)
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Mono Wireless C++ Library for TWELITE.
This page contains information that is still under development. The contents of this page may not yet be reflected in the public source code.
The MWX library is intended to simplify the code notation of the TWELITE radio module. a program created with MWX is called an act. There are two types of acts: descriptions by loops and descriptions by events (called BEHAVIOR).
This page introduces the main features of ACT.
Description by loop (setup()
, loop()
). Suitable for writing small-scale functions.
Event-driven application description. It is possible to define various event and interrupt handlers, as well as a state machine within a class that is suitable for describing complex behavior of the application, and write code with good visibility. This description is called BEHAVIOR.
Simplify peripheral procedures. Defines class objects to handle commonly used UART, I2C, SPI, ADC, DIO, timer, and pulse counter.
A simple relay network is defined. This relay network is implemented in the same way as the TWELITE standard application, and is characterized by the fact that individual addresses are managed by 8-bit logical IDs, and that wireless packets can be sent to the network immediately after power-on because there is no communication to build the network.
Board definition for PAL and MONOSTICK. Easy handling of sensors etc. on the board.
Please refer to the section. If you have any questions or concerns, please contact our support desk.
Installing the TWELITE SDK
Download the TWELITE STAGE SDK distribution archive (e.g. ZIP) and extract it to an appropriate folder.
The folder names of each level of the destination folder must not contain anything other than one-byte numbers 0..9
, one-byte alphabets a..zA..Z
, and some symbols -_.
. In short, it must not contain spaces, Kanji characters, Hiragana characters, etc.
This is the end of the normal installation process. Refer to the following "Set Environmental variables" and below as necessary.
If you want to build with other than TWELITE STAGE application, please set the environment variables (e.g. MWSDK_ROOT
).
MWSDK_ROOT
, MWSDK_ROOT_WINNAME
(for Windows10) need to be set.
In this example, the name of the extracted folder is C:\MWSTAGE
. If you have installed the software in a different folder, please change the name.
Run C:\MWSTAGE\Tools\SET_ENV.CMD
to set the following environment variables:
MWSDK_ROOT
MWSDK_ROOT_WINNAME
For example, the following variables are set:
To uninstall the TWELITE STAGE SDK from the installed PC, please do the following:
Run UNSET_ENV.cmd
. This will unset the environment variables.
Delete the MWSTAGE folder.
Set the development environment and shell to reflect the MWX_ROOT environment variable.
There are several ways to do this, but add the following setting to the .profile
of your home folder (if the file does not exist, please create a new one). You can even build VSCode with this setting.
MWSDK_ROOT=/foo/bar/MWSTAGE/MWSDK/
export MWSDK_ROOT
To add it without using the editor, enter the command as follows. The $ is a prompt and will be displayed differently depending on your environment. The /foo/bar/MSWSDK
part should be rewritten according to the installed folder.
Set the development environment and shell to reflect the MWX_ROOT
environment variable.
There are several ways to do this, but add the following settings to .profile
in your home folder (create a new file if you don't have one). You can even build VSCode with this setting.
MWSDK_ROOT=/foo/bar/MWSTAGE/MWSDK/
export MWSDK_ROOT
To add it without using the editor, enter the command as follows. The $ is a prompt and will be displayed differently depending on your environment. The /foo/bar/MSWSDK
part should be rewritten according to the installed folder.
The following is an example folder after extracting the TWELITE STAGE SDK archive. (Windows, c:\work\MWSTAGE...
)
See for more information.
License
This document is also handled under MW-SLA as a part of this library package part.
This software is not officially supported by Mono Wireless Ltd. Please note that we may not be able to respond to your inquiries. Please understand beforehand.
In response to reports of defects, Mono-Wireless, Inc. does not promise to fix or improve the problem.
There are also cases where the software may not work depending on the customer's environment, such as the package installed.
Creating a new project
To create a new project, copy the folder of an already existing sample act with a different name and edit the file name.
The destination folder does not have to be a folder under the MWSDK. However, the folder name must not contain any whitespace characters or Japanese names.
The file structure of the project is as follows (we'll use PingPong
as an example)
Copy this PingPong
folder to another location (but without Japanese characters or spaces in the folder name).
The only thing you need to edit is the file name PingPong.cpp
. Change it to AlphaBravo.cpp
, the same as the folder name.
Run build\build-BLUE.cmd
and if a BIN file is generated, it' is done (Windows 10).
On Linux/macOS, run make TWELITE=BLUE
to see if the build succeeds.
To add files to be built, edit build/Makefile
. The .c
.cpp
files directly under the project will be added automatically, but other files will need to be edited.
If you use VSCode, edit the definition under .vscode as necessary.
Most of the examples included in the TWELITE STAGE SDK are as follows
The source code for the TWELITE STAGE SDK library cites ${env:MWSDK_TWENET_LIBSRC}/include/**
${env:MWSDK_TWENET_LIBSRC}/src/**
. This environment variable MWSDK_TWENET_LIBSRC
is automatically set when the project is opened in VSCode from the TWELITE STAGE app.
For the build task, no additional options such as -D
are set by default.
The (MW-SLA) applies to anything in this package that is not specifically described in the license.
See for how to edit the file.。
The MWX library is designed to make the programming of TWELITE modules easier and more extensible. Based on the TWENET C library that has been used in the MWSDK, the MWX library has been developed as a library for the application development layer.
The name of the MWX library is Mono Wireless C++ Library for TWELITE, where MW comes from MonoWireless, and C++ -> CXX -> double X -> WX, where MW and WX are combined to form MWX.
The code written using this library is called "ACT".
This section describes the notation used in this explanation.
This is called a universal reference, and is often used in standard libraries. In our library, auto&& is used in most cases
The namespace
, inline namespace
, and using
are used to redefine names and so on. Some of them are abbreviated in the explanation.
The MWX library has not been developed to support all the functions of the underlying libraries and functions (functions in the TWNET C library, microcontroller peripheral functions provided by semiconductor vendors, and IEEE802.15.4 functions).
The MWX library is written in the C++ language, and the act description is also written in C++. However, not all functions are available even in the C++ language. Please pay particular attention to the following points.
You can allocate memory with the new
and new[]
operators, but you cannot destroy the allocated memory, and most C++ libraries that allocate memory dynamically are virtually unusable.
The constructor of the global object is not called.
Note: If necessary, initialization can be done including a call to the constructor by initializing it as in the initialization function ((void*)&obj_global) class_foo();
).
exception
cannot be used.
Unable to use virtual
functions.
Due to the above limitations, only some of the C++ standard libraries such as STL can be used.
※ This is a description of what we know.
We have not conducted a comprehensive verification of the standard library as to whether it can be used or not, nor what might be available. If you find any inconvenience in operation, please try to implement it in another way.
The source code can be found at the followings:
{MWSDK install dir}/TWENET/current/src/mwx
Install and Build
In order to write an application using the MWX library (called ACT in this document) and run it, you need to set up a development environment.
After the release of the TWELITE STAGE distribution package, fixes and additions are stored in the GitHub repository. Please replace the location of the distribution package if necessary.
Click on the link for each release to clone the Git file or download the source code in zip format.
Replace the contents of the following folders.
Updated information prior to major releases may be posted on the above link.
changed Wire object that reserves memory in heap area.
changed function name from G_OCTET()
to G_BYTE()
](api-reference/funcs/utility/byte-array-utils.md to avoid name conflict in utils.h.
changed an order of vAHI_DioInterruptEnable()
in the attachIntDio()
for code efficiency.
added secondary network behavior the_twelite.network2 to support universal receiver (receive packets from NWK_LAYERED, NWK_SIMPLE or networkless packets in the same executable code.)
introduced MWX_Set_Usder_App_Ver() function to set application version during MWX intitialization, mainly for interactive mode.
added new[]
operators for EASTL
pre-compled most of source codes in MWX to quicker compile.
fixed: DIO events were being passed on to unrelated ports.
Added an internal procedure to allow output using Serial class objects in Interactive settings mode. (Serial._force_Serial_out_during_intaractive_mode()
)
Main revisions
Serial1
port and alternate port were not properly defined.
Enabled to change the baud rate of (Serial
UART0).
If you don't define a callback function, you can use the previous procedure.
Wrong definition ID for interactive mode setting<STG_STD>
and some default values.
Added support for changing the default values of channel and logical device IDs in addition to AppID in interactive mode settings<STG_STD>
.
added support for setting the_twelite and <NWK_SIMPLE>
objects in interactive mode <STG_STD>
object for some settings.
added support for setting the default value of the number of retransmissions in <NWK_SIMPLE>
.
Serial
(UART0) input and output from the application is not performed while the interactive mode screen<STG_STD>
is displayed.
added CUE::PIN_SET
, PAL???"":PIN_SET
(Since it is unnatural to use PIN_BTN
for CUEs without buttons)
Move namespace of random() to mwx::.
MONOSTICK watchdog setting is now done in 32ms units.
When sleep was performed using BRD_TWELITE
, the pins were not initialized correctly upon recovery.
Sorry. The following has not been translated.
Added method to receive other packets (without network usage) that are not in NWK_SIMPLE format when using NWK_SIMPLE. Add NWK_SIMPLE::receive_nwkless_pkt()
to initialize NWK_SIMPLE. When using this packet information, use only the TWENET C library layer structure by .get_psRxDataApp()
and the data array obtained by .get_payload()
. Information obtained from other methods of the incoming packet (auto&& rx = the_twelite.receiver.read()
) is undefined.
Refine get_stream_helper()
code and read/write position API.
Fixed bugs in smplbuf::get_stream_helper()
.
Modified so that div100()
, which calculates the quotient and remainder, can be output to Serial, etc.
Changed implementation of smplbuf<>
array class. The inheritance from mwx::stream
is removed to reduce memory consumption, and a helper class and an inheritance class are defined separately.
Added mwx_printf()
mwx_snprintf()` function.
added the_twelite.stop_watchdog()
and the_twelite.restart_watchdog()
functions.
mwx::stream
maintenance: operator bool()
is obsolete. Disable timeout when .set_timeout(0xff)
is set to 0xff in the read timeout setting. Add definition of <<
operator.
Add scale functions between 8bit and 0..1000 with no division.
Added division by 10,100,1000 (quotient and remainder at the same time) div10()
, div100()
, div1000()
. Restricted the value range and composed mainly of multiplication and bit shift.
Added corresponding methods for encrypted packets.
packet_rx::is_secure_pkt()
: determine if received packet is encrypted or not.
STG_STD::u8encmode()
: Obtain encryption setting in interactive mode.
STG_STD::pu8enckeystr()
: Obtain encryption key byte sequence in interactive mode.
Serial1: Default port is DIO11(TxD), DIO9(RxD) because they are usually assigned to I2C, though DIO14,15 overlap with I2C in the semiconductor specification.
The calculation of the baud rate is omitted for the main baud rates.
Serial: The proxy functions for Serial: available()
and read()
are now held only by void*
, and the specification memory is reduced by 8 bytes.
Add typedef boolean
.
Network: added support for encryption.
The first parameter is the encryption key, and the second parameter is true
, so that plaintext packets are also received. packets are also received.
Added sensor support for SHT3x and BME280
Sensor: added a mechanism to exchange configuration parameters and status in legacy code (C library wrapper class).
Sensors: I2C address can be specified for SHT3x and BME280.
Configuration: added hide_items()
. Unnecessary items can be deleted.
Added `H/W UTIL menu' to display DI status, I2C probe, and PAL EEPROM contents.
Configuration: Add encryption related menus.
I2C related fixes (to improve compatibility with code implemented using TwoWire class)
Processing of requestFrom(false)
did not work correctly because there was no code to send the NO_STOP message when processing requestFrom(false)
.
Added TwoWire
class name alias.
Modified not to initialize multiply in begin()
process.
Added setClock()
method (but it is a dummy function and does nothing).
Added WIRE_CONF::WIRE_? KHZ
added. Added the main configuration values for the bus clock.
Addition of delayMilliseconds()
function
Addition of digitalReadBitmap()
function
Improved accuracy of delay()
.
Fixed the problem that Serial1
instance was not defined
Fixed problem with Analogue
interrupt handler not being called
Support for MWSDK202020_05
Duplicate checker duplicate_checker was not removed as expected due to incomplete initialization, etc.
The implementation of format() was made less machine-dependent. If 64-bit arguments are included, the number of arguments is limited.
The fix assumes MWSDK2020_05.
Updates are recommended for this fix.
Support for MWSDK2020_04
Fixed initialization problem with Timer0..4
Changed internal processing of mwx::format()
Added experimental code to support interactive mode
This fix assumes MWSDK2020_04.
We recommend an update to this fix.
Fixed problems with handling of relay flags in packets
We recommend an update to this correction.
First release (included in SDL Dec 2019)
Other updates to the MWSDK may be required. Please refer to the release description at the time of the update; see for information on updating the MWSDK.
The source code of the library is available on GitHub (). To replace the source code of the library, please follow the steps below.
added (At this time, only Parent Node reception is supported.)
added to describe placement new simply.
added support of
Added board support for TWELITE ARIA and sensor definition
Added event callbacks to notify received packets () and completed transmission ().
Added board behavior () for TWELITE CUE.
Add EEPROM class object. ()
Samples ()
Added pktparser class ()
Added sample ()
sample serparser/pktparser
so that it can be built on other platforms ()
Added NOTICE PAL / PCA9632 support (Description , sample )
()
Added
Channel Manager. Implement
()
DESC
In order to create a development environment, you need to install the software and agree to the license agreement. In addition, you may need to configure security settings on your PC or workstation.
Although we take great care when distributing the software, we ask that you also check for viruses.
Please check with the administrator of your environment regarding your security approach and operations (e.g., whether or not to install external applications).
In addition, the installation and operation of the development environment may require the intervention and configuration of the operating system (e.g., running an application whose developer is unknown; many of the development environments and tools introduced here do not have a built-in mechanism to prove the origin of the application) Please refer to the general information on how to configure.
To write an application using the MWX library, you will need the following:
MWSDK(Software Development Environment)
An editor for development (Microsoft's VisualStudio Code will be introduced)
Compiler toolchains, etc., are relatively less dependent on the environment, so they can be expected to work in many environments, but we recommend the Windows 10 version that is currently supported. If your system does not work due to differences in the operating environment, please prepare a separate environment by referring to the environment we have confirmed.
The following is the version we are using for development.
Windows11 21H2 (Visual Studio 2019)
FTDI driver must be running (to run MONOSTICK, TWELITE R)
Compiler toolchains, for example, are relatively less dependent on the environment, so they can be expected to work in many environments, but we recommend distributions that are currently supported. If your system does not work due to differences in the operating environment, please prepare a separate environment by referring to the environment that we have confirmed.
The following is the version we are using for development.
Ubuntu 18.04 LTS 64bit
Ubuntu 20.04 LTS 64bit
32bit systems are not supported.
Compiler toolchains, for example, are relatively less dependent on the environment, so they can be expected to work in many environments, but we recommend distributions that are currently supported. If your system does not work due to differences in the operating environment, please prepare a separate environment by referring to the environment that we have confirmed.
The following is the version we are using for development.
macOS 10.14 (Mojave, Intel)
macOS 12.4 (Monterey, Apple Silicon)
About Visual Studio Code and other development environments
For information on how to run and use the development environment, please refer to the information of its developer or community.
The build results for Linux/macOS are different from the results for Windows 10. As far as we know, there is no difference in the normal operation, but the binary size tends to be a few percent larger, especially since the LTO of gcc is disabled.
If you have any doubts about the operation, be sure to run the build on Windows 10 and confirm that it reproduces before contacting us.
mwx
twesettings
TWENET C
1.3.5
mwx
twesettings
TWENET C
1.3.5
mwx
twesettings
TWENET C
1.3.5
mwx
twesettings
TWENET C
1.3.4
mwx
twesettings
TWENET C
1.3.4
mwx
twesettings
TWENET C
1.3.4
mwx
twesettings
TWENET C
1.3.3
This section describes the specifications, limitations, notes in this document, and design memos for the C++ language used in the MWX library.
The application loop description is intended to be similar to the commonly used API system, but the implementation should be tailored to the characteristics of TWELITE.
TWENET is an event-driven code description, and it should be classed so that it can be handled. The above classifications will encapsulate the behavior of the application.
Event-driven and loop descriptions should be able to coexist.
Simplify procedures by classifying typical peripherals. Make them accessible by loop descriptions whenever possible.
Simplify the procedures for using the boards we sell, such as MONOSTICK/PAL, by creating classes. (For example, to automate the use of an external watchdog timer.
Application classes and board classes should be made available through a unified procedure, introducing the idea of polymorphism. (For example, to load application classes with several behaviors at startup, and to avoid defining the connection code of the TWENET C library each time).
There are no restrictions on the use of C++ functionality. For example, it provides a means to simplify typical procedures such as packet construction and decomposition, which are complicated in handling wireless packets.
The operator ->
should be avoided as much as possible, and the API should be based on reference types in principle.
gcc version 4.7.4
C++11 (For compiler support status, please refer to the general information.)
※ This is a description of what we know.
You can allocate memory with the new and new[] operators, but you cannot destroy the allocated memory; most C++ libraries that allocate memory dynamically are virtually unusable. It is used for objects that are created only once and not destroyed after that.
The constructor of the global object is not called.
Note: If necessary, you can initialize the constructor call by using the initialization function (setup()
) as shown in (new ((void*)&obj_global) class_foo();
).
Exception cannot be used.
Unable to use virtual function.
This section contains information that will help you understand the code when referring to the MWX library code.
Due to the limited time available for implementation, some of the details may not be sufficiently developed. For example, const is not fully taken into account in many classes.
We have the following policy for namespaces.
In principle, definitions are placed in a common namespace mwx.
We want to be able to use namespaces without identifiers, but we want to require identifiers for some definitions.
Class names should be relatively long, and those used by users should be defined as aliases.
Classes, Functions, and constants are defined within the namespace of mwx names (more precisely, mwx::L1 enclosed in inline namespace L1), with a few exceptions. inline namespace is specified so that definitions that require the specification of mwx:: can coexist with those that do not. The reason why inline namespace is specified is to allow definitions that require the specification of mwx:: to coexist with those that do not.
Most of the definitions do not require namespace names to be specified by using namespace. These specifications are made in using_mwx_def.hpp
in the library.
Exceptionally, relatively short names can be specified as mwx::crlf
, mwx::flush
. These are placed in the inline namespace mwx::L2; using namespace mwx::L2; will allow them to be used without specifying the namespace name.
Also, some class names have a using specification.
The std::make_pair
used in the MWX library is specified using.
The following example shows how to implement an interface called interface()
in a Derived
class that inherits from Base
, and calls the Derived::print()
method from Base
.
The following are the main classes used in the MWX library.
Basic parts of event processingmwx::appdefs_crtp
state machine public mwx::processev_crtp
stream mwx::stream
In the CRTP class, the class from which it inherits is different for each instance. For this reason, it is not possible to cast it to a parent class and treat it as a member of the same family, nor is it possible to use advanced polymorphism such as virtual functions or RTTI (runtime type information).
The following is an example of implementing the above CRTP example with a virtual function: CRTP cannot manage instances together in the same array as in Base* b[2].
The MWX library solves this problem by defining a dedicated class for storing class instances of CRTP and defining a similar interface to this class. An example code is given below.
The VBase class member variable p_inst stores a pointer to an object of type Base , and pf_intrface is a member function pointer to Base::s_intrface. Base::s_intrface invokes the T::intrface method by being passed an object instance of itself as an argument and static_casting it to the T type.
Storage in VBase is implemented here by overloading the = operator (see below for source examples).
In the above example, when making a call to b[0].intrface(), Base::s_intrface() will be called with reference to the VBase::pf_intrface function pointer. In addition, a call to Derived1::intrface() will be made. This part is expected to be expanded inline by the compiler.
It is also possible to perform a conversion from the VBase type to the original Derived1 or Derived2 through a forced cast, but there is no way to directly know the type of the pointer stored in void*. Although there is no completely safe way to do this, a unique ID (TYPE_ID) is provided for each class as shown below, and the ID is checked when the cast is executed (get() method). If the get() method is called with a different type, an error message will be displayed.
B
If a pointer is stored as a Base type, it may not be correctly converted to a T type (e.g., when T has multiple inheritance), so a static_assert is used to determine at compile time that the pointer is derived from a Base type by using is_base_of in <type_trails>.
The microcontroller in the TWELITE module does not have enough memory nor does it have advanced memory management. However, the area from the end of the microcontroller's memory map to the stack area is available as a heap area, which can be allocated as needed. An overview of the memory map is shown in the figure below, where APP is the RAM area allocated by the application code, HEAP is the heap area, and STACK is the stack area.
Even if it is not possible to delete, the new operator may be useful in some situations. For this reason, the new and new[] operators are defined as follows: pvHear_Alloc() is a function for allocating memory provided by the semiconductor library, and the same is true for u32HeapStart and u32HeapEnd. 0xdeadbeef is a dummy address. Please do not point out that it is strange that beef is dead.
Since exceptions cannot be used, there is no way to deal with failures. Also, if you continue to allocate without being aware of the memory capacity, there is a possibility of interference with the stack area.
The MWX library does not use the container classes provided by the standard library, considering the small resources of the microcontroller and the lack of dynamic memory allocation, but defines two simple container classes. The container classes have defined iterators and begin() and end() methods, so you can use some of the range for statements and STL algorithms.
smplbuf
It is an array class that manages the maximum area (capacity) and the usable area (size) whose size can be specified within the maximum area. This class also implements the stream interface, so data can be written using the << operator.
smplque
The FIFO queue is implemented. The size of the queue is determined by template parameters. There is also a template argument to manipulate the queue using interrupt inhibition.
In the container class, the memory allocation method is specified as a parameter of the template argument.
alloc_attach
Specify the buffer memory that has already been allocated. This is used when you want to manage the memory area allocated for the C library, or when you want to process the same buffer area as a fragmented area.
alloc_static
Allocate as a static array in the class. The size is determined in advance or used as an area for temporary use.
alloc_heap
Allocate to the heap area. Once allocated to the system heap, it cannot be discarded, but it is suitable for use in initialization to allocate an area according to application settings.
In the MWX library, variable number arguments are used for operations on byte sequences, bit sequences, and printf equivalent operations. The example below shows the process of setting 1 to the specified bit position.
In this process, the parameter pack of template (typename... part of template) to perform recursive processing to expand the arguments. In the above example, since constexpr is specified, the calculation is done at compile time and the result is equivalent to macro definition or const value specification such as b2. It can also behave as a function that dynamically calculates variables as arguments.
In the following example, the expand_bytes function is used to store a value in a local variable from the received packet data string. In the case of using a parameter pack, since the type of each argument can be known, it is possible to store values of different sizes and types from the byte string of the received packet, as shown below.
An iterator is an abstraction of a pointer, which has the effect of making it possible to access data structures as if they were pointers, even if the data structures are not memory-contiguous, for example.
The following example shows the use of an iterator for a FIFO queue that cannot be accessed continuously with a normal pointer, and also an iterator that extracts only a specific member of the FIFO queue structure (the X axis in the example).
The following is an excerpt of the implementation of an iterator for the smplque class. In this iterator, the queue object is managed by its entity and its index. The part of the queue that is discontiguous in memory (ring buffer structure where the next to the tail must point to the beginning) is solved by smplque::operator []. If the addresses of the objects match and the indices match, the iterators point to the same thing.
This implementation part also includes the typedefs required by , allowing more STL algorithms to be applied.
構造体を格納したコンテナ中の、特定構造体メンバーだけアクセスするイテレータは少々煩雑です。構造体のメンバーにアクセスするメンバー関数を予め定義しておきます。このメンバー関数をパラメータ(R& (T::*get)()
)としたテンプレートを定義します。Iter
はコンテナクラスのイテレータ型です。
Iterators that access only specific structure members in the container that contains the structure are a bit complicated. Define a member function to access the structure members in advance. Next, define a template with this member function as a parameter (R& (T::*get)()
). "Iter
" is the iterator type of the container class.
The operator *
that accesses the value calls the member function described above. The *_p
is the axis_xyzt
structure, and (*_p.*get)()
calls _p->get_x()
if &axis_xyzt::get_x
is specified in T::*get
.
The _axis_xyzt_iter_gen
class implements only begin()
, end()
and generates the above iterators. Now you can use range for statements and algorithms.
This class name is very long and difficult to write in the source code. We will prepare a generator function to generate this class. In the example below, it is get_axis_x()
in the last line. By using this generator function, the description becomes as simple as auto&& vx = get_axis_x(que);
as shown in the beginning.
This iterator for extracting only the axes can also be used with the smplbuf
class of array type as well.
In order to describe the application behavior by user-defined classes, typical handlers need to be defined as mandatory methods, but it is complicated to define all the other numerous interrupt handlers, event handlers, and state machine state handlers. Ideally, only those defined by the user should be defined, and only that code should be executed.
In the MWX library, a large number of DIO interrupt handlers (on TWELITE hardware, a single interrupt, but for ease of use, a handler is assigned to each DIO) are defined as empty handlers using templates, and user-defined member functions are defined by specializing the templates.
The actual user-described code has been simplified by macroizing and including header files, but the above includes the code necessary for the explanation.
The my_app_def::cbTweNet_u8HwInt()
is called from the interrupt handler from TWENET. in the cpp file, only int_dio_handler<12>
is instantiated with the specialization described in it. file is instantiated from a template in the hpp file. The rest are instantiated from templates in the hpp file.
Eventually, we can expect that the compiler optimization will determine that codes other than number 12 are meaningless and disappear from the code (however, we do not guarantee that they will be optimized as described above).
In other words, in user code, if you want to define the behavior at interrupt 12, just write int_dio_handler<12>
(Note: to enable DIO interrupt, you need to call attachInterrupt()
). Handlers that are not registered are expected to be low-cost calls due to compile-time optimization.
The stream class is mainly used for input/output of UART (serial port), and MWX library mainly defines procedures for output. But some of them are also defined for input.
This section describes the implementation required by the derived class.
The above is an implementation of the write()
method that writes a single character. The stream<serial_jen>
of the parent class accesses the serial_jen::write()
method using the get_Drived()
method to perform casting.
Define methods such as write()
, read()
, flush()
, and available()
as needed.
By get_pfcOutput()
, the vOutput()
function defined in the derived class is specified, and pvOutputContext
is passed as its parameter. In the above example, when the <<
operator is called with int type, serial_jen::vOutput()
and TWE_tsFILE*
which is already set for UART are passed to the fctprintf()
function.
In the Wire class, it is necessary to manage the communication from start to end when sending and receiving with a 2-wire device. This section describes the contents of the description of using the worker object.
This is an excerpt of the periph_twowire::writer
class, which inherits from mwx::stream<writer>
to implement the stream interface, and implements the write()
and vOutput()
methods to use the steam interface. To use the steam interface, the write()
and `vOutput() methods are implemented.
The constructor calls the method to start communication for 2-wire serial and the destructor calls the method to end communication. Also, the operator bool()
operator returns true if the communication of the 2-wire serial device is successfully started.
The get_writer()
method creates an object wrt
. Due to the Return Value Optimization (RVO) of the C++ compiler, the writer
is created directly in the wrt
, so no copy is made and the bus running in the constructor is not initialized multiple times. However, RVO is not guaranteed by the C++ specification, and just in case, the MWX library defines copy, delete assignment operators, and move constructors (although it is unlikely that move constructors will be evaluated).
The wrt in the if clause is first initialized by the constructor and starts communication at the same time. If there is no error at the start of communication, the bool operator at the time of conditional judgment returns true, and the processing in the scope of the if clause takes place. If there is no error at the start of communication, the bool operator at the conditional judgment returns true, and the processing in the if clause scope is performed. When the scope is exited, the destructor terminates the 2-wire serial bus. If there is no communication partner, false will be returned and the wrt object will be destroyed.
It overrides the definition of operator << (int)
, which is specific to Wire and SPI. The default behavior of the stream is to convert numeric values to strings and output them, but Wire and SPI rarely write numeric strings to the bus, and on the contrary, we often want to input literals of numeric type such as configuration values. We will change this behavior.
In this example, the int type values are truncated to 8 bits and the values are output.
The following is a supplement to the terminology used in this document.
The explanation of terms may not be in accordance with the definitions provided in standards and other documents.
Software Development Environment
This is the radio standard used by the TWELITE radio module; as long as you use the MWX library, you do not need to be aware of the details of the radio standard.
The smallest unit of communication in wireless communications.
The maximum amount varies depending on the communication method and settings during communication, but in the MWX library standard communication <NWK_SIMPLE>, the amount of data a user can send in one packet is 90 bytes.
It refers to the body of data contained in a wireless packet.
It refers to a radio station in a wireless network.
A program created using this library. This refers to its source code or the program that runs.
A program in the form of an event, among other ACTs. The source code or the program that runs it.
BEHAVIORs are described by a single class definition, which describes callback functions from TWENET, events, and interrupt processing, all in one place. There are three types of behaviors available in the MWX library:
Application BEHAVIOR: A class defined by the user that describes the application in an event-driven manner.
Board BEHAVIOR: A class to simplify the use of the functionality of the board that implements the TWELITE radio module.
Network BEHAVIOR: A class for simplifying procedures in wireless networks.
Behavior names are enclosed in < >. For example, the behavior name for a simple relay network is <NWK_SIMPLE>.
In the description of this library, objects that are declared globally from the beginning in the library are referred to as class objects: Serial
, Wire
, etc. These class objects can be used without any procedure or by performing the start procedure.
Class objects that consume relatively large amounts of memory allocate memory along with the initialization parameters during the initialization procedure (.setup() or .begin() method).
The C++ language.
One of the versions of the C++ standard, meaning C++ as of 2011, which was standardized by ISO in 2011. It has been greatly enhanced since its predecessor, C++03. Newer versions such as C++14 and C++17 are available.
A collection of procedures that focus on some data in a single place. A structure contains the procedures for handling that structure. It actually develops into a much deeper topic, but please refer to technical books.
In C++, the keywords struct and class are essentially the same thing, and a class is a class regardless of which keyword it is declared with.
If the above class definition was also done in C, for example, it would look like the following:
It is a class that includes existing C libraries and their internal structures, and adds C++ specific functionality to make it easier to use. In some cases, the description "wrapped ~structure" is used in the explanation.
A function that is defined in a class and is associated with the class.
A class is materialized (allocated memory).
In this explanation, object and instance are treated as having the same meaning.
Initialization procedure at the time of object creation.
This is the procedure when the object is destroyed, paired with the constructor.struct myhello {
In C++, polymorphism is achieved by virtual classes. Specifically, a class that defines a pure virtual function specified by the virtual keyword.
In the C/C++ language, think of it as a scope enclosed in { }. The objects created in this scope are destroyed when they leave the scope. The destructor is called at this time.
The following is an explicitly scoped version of helo2, which is discarded after line 8 and the destructor is called.
MThe MWX library uses the following notation. Here, the validity period of an object declared within the conditional expression of an if statement (older C languages such as C89 do not allow declarations in such a place) is within {} of the if statement.
For example, a two-wire serial bus is a procedure where there is a start and end procedure and the bus is manipulated by an object only during that time. After the object is created, if the bus is properly connected, the true clause of the if statement is executed and the bus is written or read by the created object. When the bus read/write operation is completed, the if statement is exited and the destructor is called to terminate the bus usage.
The namespace are actively used in C++ to avoid duplication of definition names. To access a definition in a namespace, use::.
Think of a template as an extension of a C macro.
This example defines a simple array, where T and N are template parameters, where T is the type name and N is a number, and defines an array class of type T with N elements.
In C++11, NULL pointers are now written as nullptr.
In C++, reference types are available. This is similar to access by pointer, but with the restriction that it must refer to an object.
For functions with pass-by-reference parameters like the one below, the value of i
can be rewritten in incr()
.
In the example of the template explanation, the return type of operator[]
is changed to T&
. By doing so, it becomes possible to perform assignment operations directly on the data inside the array, such as a[0]=1
.
C++11 introduces the auto keyword for type inference. This allows the compiler to infer the type of an object from its initialization description, thus omitting the need to specify the specific type name. This is effective in cases where class names using template are very long.
In most of the explanations, auto&&, which is called universal reference, is used. Universal references can be written here without being aware of passing references.
A class for storing multiple objects of a specific data type such as arrays is called a container. An array class such as myary mentioned in the template example is also called a container.
A pointer in C can be thought of as a way to access a contiguous set of memory elements in a continuous manner from the beginning to the end. The simplest implementation of a FIFO queue is a ring buffer, but there is no memory continuity. Even such a data structure can be described in the same way as a pointer using an iterator.
The methods .begin()
and .end()
are used to get the iterator. The iterator that points to the beginning of the container is obtained with .begin()
. The iterator that points to the next to the end is obtained with .end()
. The reason for using the next to the end instead of the end is for the clarity of the loop description in the for and while statements, and to handle the case where the number of elements stored in the container is zero.
In the above, some_process()
is applied to each element of the que
using the iterator p
. p
is incremented by the ++
operator as an iterator that points to the next element. Even if the container has a data structure that cannot be described by a pointer, it can be processed in the same way as using a pointer.
Since .end()
indicates the next to the end, the end decision of the while statement is as simple as (p ! = e)
, which is concise. If there is no element in the queue, .begin()
returns the same iterator as .end(). If there are no elements in the queue, .begin()
will return the same iterator as .end(), which is the next iterator after the iterator for the unstored elements.
For a contiguous container in memory, its iterator will usually be a normal pointer. It is not expected to be a large overhead during its operation.
The C++ standard library includes the STL (Standard Template Library), which is part of the MWX library.
In C, for example, the process of finding the maximum or minimum value is written separately depending on the type. In C++, you can use templates, iterators, etc. to describe such operations independently of their types. This is called an algorithm.
For example, the algorithm to find the maximum value as shown above. This algorithm is type-independent. (It is called generic programming.)
Here we specify an iterator for que and apply the algorithm std::minmax_elenet
to obtain its maximum and minimum. std::minmax_elemet
is defined in the C++ standard library. Its return value is std::pair
, which combines any two values. The algorithm calculates the maximum and minimum values if the elements indicated by the iterator can be compared with each other using operators such as <
,>
,==
. The return type is also derived from the iterator type.
Installing VSCode
VisualStudio Code (VSCode) is recommended to make Act (source code) writing easier. The attached Act contains a file that has been configured to ensure that the code is interpreted properly in VSCode.
VSCode reads source files and header files and interprets the source code, thus providing function definition information and function and method name completion to help you write source code. The MWX library requires more header files to be loaded than the C library. In comparison to C development, the MWX library requires more header files to be loaded and the editor may be slower in some environments.
The project settings of VSCode requires some information, such as library source code location, to analyse source codes. These inforamtion are passed by TWELITE STAGE app as environmental variable when launching VSCode. Therefore, application instance of VSCode should not be present when launching from TWELITE STAGE app, otherwise VSCode will not refer to these environmental values.
Please note that this support does not cover questions about how to install or use VSCode. Please refer to the information available in the public domain.
Depending on your environment, you may need to configure security settings or other settings in order to install the software. Please check with your system administrator whether installation is possible, and refer to the distributor or general information for instructions.
VSCode allows you to do the following:
Editing the source code
The IntelliSense based on source code interpretation (*not all definitions can be guaranteed to be interpreted correctly)
VSCode can be downloaded from the link below.
The code
command must be enabled to invoke VSCode from TWELITE STAGE.
The following information is from code.visualstudio.com
To enable Visual Studio Code to interpret C/C++ language descriptions, install a plugin.
C/C++ for Visual Studio Code
The MWX library examples include a .vscode
definition. This definition uses the MWSDK_ROOT environment variable to identify the source code of the library (under {MWSDK_ROOT}/TWENET/current
).
When launching VSCode from TWELITE STAGE, those settings such as MWSDK_ROOT
will be setup automatically. In some cases, such as when you have already started VSCode, the settings may not be reflected.
VSCode's interpretation of source code is not always the same as the compiler's interpretation. Also, depending on how the source code is edited, the interpretation may be more incomplete.
Build definition Makefile
The Makefile is stored in build/Makefile and is pre-defined to build the act by running the make command.
After copying the project folder from another environment, make sure to delete the build/objs_???
folder. If any intermediate files from the other environment remain, make will fail.
(MWSDK 2020-04)
You can avoid errors by adding USE_APPDEPS=0 to clean and then running make again.
$ make USE_APPDEPS=0 TWELITE=BLUE clean
...
$ make TWELITE=BLUE
``
Specify the build target as BLUE or RED; for TWELITE BLUE, use make TWELITE=BLUE
.
Run the build. Usually, you can omit this and use make TWELITE=BLUE
.
Remove intermediate files from the build. Do this as make TWELITE=BLUE clean
.
Remove all intermediate files. Do this as make cleanall
, the same as removing all of the objs_???
folder in the build folder.
When set to 1 (the default), the build file is determined based on file dependencies. For example, if there is a change in a header file, the associated source file will be recompiled.
If set to 0, the makefile will not error if there are inconsistent intermediate files left.
Depending on the size of the act, and when defining behaviours, the source files are usually split and built separately.
One of the build files is {project folder name.cpp}.
If you want to define other files, edit the build/Makefile
in your project folder.
The above is an example Makefile with sample PAL_AMB-bhv.
Specify the version number. This will be reflected in the build result file name.
During compilation, it is passed as a definition like -DVERSION_MAIN=0
-DVERSION_SUB=1
-DVERSION_VAR=0
.
(MWSDK 2020-04)
If you do not place files in subfolders, you no longer need to specify additions. All .c .cpp files in the project file will be added.
When you add a source file, you need APPSRC_CXX
and APP_COMMON_SRC_DIR_ADD?
.
Append the name of the source file to `APPSRC_CXX'. This file name must not contain a folder name. Anything in a subfolder should also be specified without a folder (i.e. if the same filename is in a subfolder, the build will fail)
Next, specify the search path if the source files are stored in a location other than the project folder. You can set up to four.
The folder specification is relative to the Makefile.
A number of other options can be passed to the compiler linker.
Building ACT
The application program written in the MWX library is called ACT. The first step is to build and write it.
About the build folder structure
About the build script
About building with Visual Studio Code (Described as VSCode)
Depending on the OS environment, security warnings may appear when running each executable program (e.g. make
, build toolchain like gcc
). It is necessary to configure the settings to suppress the warning. (Please consult with yourself or your system administrator to determine if you wish to run the program with the warning suppressed.)
Open the folder MWSDK_ROOT
(e.g. C:\MWSDK
) where you have installed the MWSDK. It has the following structure
Tool act files such as compilers are stored under Act_samples
. (Some of them are omitted below)
These acts are simple examples to help you write your own MWX library, but many of them have the following features
Obtaining sensor values
After obtaining the sensor value, send a wireless packet to the master
Sleeps for a period of time (or waits for an interrupt) after transmission is complete
The Parent-MONOSTICK act is used to receive and display packets. This act for the parent is output in ASCII format. (:00112233AABBCC.... .FF[CR][LF]
, and the middle part is a hexadecimal byte expressed by two ASCII characters. The trailing ? is also a two-character byte, but it becomes a checksum byte called LRC. Reference: ASCII format)
When trying to get it to work in practice, try the following combinations:
Now let's take a look inside the PingPong folder from within ACT.
You must have a .cpp
file with the same name as the folder directly under the folder.
The ACT file PingPong.cpp is located directly under the PingPong folder. If you change the name of the folder, make sure to rename the .cpp file to the same name as the folder.
Next, open the build folder.
It contains the scripts and Makefiles needed for the build.
Build by running make TWELITE={BLUE or RED}
in the folder containing this Makefile
. Building with VSCode is the same, calling make internally.
The TWELITE STAGE app can be used to build, write, and run. This section describes the TWELITE STAGE application from startup to build.
Connect MONOSTICK or TWELITE R to your USB port.
TWELITE is a sensitive electronic component and should be handled with care. Typical precautions are listed below.
Especially when TWELITE R is used, the electronic board is often in direct contact with the outside without a case, which may cause unintended shorts, noise, or other problems that prevent the USB device from operating properly.
In this case, closing the application and unplugging and plugging in the USB device will usually restore it. In the worst case, the USB device may be damaged or the PC may be corrupted.
Also, handle electronic circuit boards with care.
Circuit error.
Check the circuit again before turning on the power.
Be careful not to reverse-insert batteries or over-voltage.
Static electricity
Even a voltage that is not human-sensitive can cause a semiconductor failure. Even simple measures such as touching metal parts before working, wristbands, and special mats can have a reasonable effect.
Short-circuits caused by touching metal objects, etc.
Make sure that there are no metal objects near the electronic board. If clips or other objects are scattered around, this may cause a short circuit, or even a dangerous situation in which the battery heats up due to a large discharge.
Below is an example of the screen while the TWELITE STAGE application is running. The main screen on the left and the command prompt screen are shown, but the main screen is usually used. The command prompt screen is not normally used, but it displays various information and input data from the TWELITE microcontroller serial port.
The main operations on the main screen are as follows
Left mouse click (selection)
Double right mouse click (return to previous screen)
Quickly press ESC
twice, or ESC
once on some screens (return to previous screen)
Hold down the Alt(Cmd) key (help screen)
Normal keyboard input (follow the screen)
This is the first screen that appears when you start the TWELITE STAGE application. If TWELITE R or MONOSTICK is connected in advance, it will be listed on this screen. Select the TWELITE you wish to operate on this screen. It is also possible to select the TWELITE without selecting it on this screen.
After exiting the serial port selection screen, the main menu appears. Select the "Application Rewrite" menu for build and write.
Before selecting the application programming menu, please check the TWELITE connection and serial port selection. The serial port selection status can be checked on the help screen that appears by holding down the Alt(Cmd) key.
There are several categories of projects that can be referenced from the TWELITE STAGE application. HELP
on the right side displays related information in a browser. Foldr
displays the folder where the project is located.
If TWELITE is already connected, the TWELITE model is determined when the menu is selected. (Inside the TWELITE STAGE application, the build is performed according to the TWELITE model that has been determined.)
If an error occurs here, return to the main menu from this screen and reselect the menu. If necessary, deselect the serial port by pressing Alt(Cmd)
+ 0
on the TWELITE STAGE application and check the various connections, including the USB connection. Some USB connection errors may not be resolved until you reboot your computer.
Here, select "Act Build & Wrt" from the " Wrt Firmware" menu.
Project names, such as sample acts, are listed. The HELP
on the right side displays the related information in a browser. The foldr
displays the folder where the project is located.
Here, select BRD_APPTWELITE
in the project selection screen shown earlier.
Once selected, writing will be performed as shown in the following screen example. If an error is displayed, follow the on-screen instructions or return to the previous screen and try again.
In Interactive settings mode, you can configure various settings, including TWELITE's wireless CHANNEL.
Return to the root menu and select Viewer
> Terminal
.
This is a very simple terminal where you can check messages from TWELITE and input data into TWELITE.
The screen displays a message when a wireless transmission is made approximately every second. You can also transition to the Interactive settings mode screen by typing + + +
.
VSCode is a powerful editor for source editing, but it is also possible to build firmware for TWELITE microcontrollers on VSCode.
VSCode is started from the TWELITE STAGE application from the project listing in Build&Write menu. (Note: The settings is required at TWELITE STAGE App [Setting Menu] > Wrt Firmware > Open a folder with VSCode
. For simplicity of configuration, the executable TWELITE_Stage_VSCode.{extension}
is available for Windows, Linux, and macOS.)
Set "Open folder with code (VSCode)" to 1
in STAGE settings.
Press [VSCode]
on the right end of the list of builds.
If VSCode is already launched, the necessary settings by setting the system environment variables may not be reflected. In this case, exit VSCode and start VSCode again from the TWELITE STAGE application.
Because of the use of system environment variables to reflect information, in principle, simultaneous operation of multiple TWELITE STAGES that reference different library folders may cause problems. If you open Terminal on VSCode and the environment variable MWSDK_ROOT
is properly set, you can expect the build to be successful.
Firstly, open a workspace from TWELITE STAGE app that you want build. The attached workspace in TWELITE STAGE SDK has build task definitions for TWELITE microcontroller.
In the example below, a workspace is opened with an example screen of the English interface.
Open a workspace and select [Terminal>Run Task...]
.
Select the type of TWELITE radio module (BLUE/RED) and the act name to build. In the example below we have selected Build for TWELITE BLUE PingPong (for TWELITE BLUE/PingPong act). The build will start immediately after selection.
The progress of the build is shown in the TERMINAL section at the bottom of the screen.
If the build is successful, you will see a message about the creation of the .elf
file, along with its size information (text data bss dec hex filename
), as shown in the highlighted part of the screenshot above.
Also, a BIN file (in the above example, PingPong_BLUE_???.bin
) file should be created under the build
folder. Please check it.
If the build does not work, check the error messages first; it is often easy to identify the cause of an error from a message on a line containing the string error.。
To be sure, clean (remove intermediate files in the objs_???
folder) and rerun the build to be sure. (All operations, including make clean
, will fail if there are intermediate files left over from builds in other environments).
Additional information about building in the command line environment.
A working knowledge of the command line (e.g. bash) is required.
Depending on the OS environment, security warnings may appear when running each executable program. It is necessary to configure the settings to suppress the warning. (Please consult with yourself or your system administrator to determine if you wish to run the program with the warning suppressed.)
To build by command line, run make
in a window where bash
(or some other shell) is running. Make sure that the environment variable MWSDK_ROOT
is set correctly beforehand. For example, if you install to /work/MWSTAGE/MWSDK
, set ~/.profile
as follows.
Run make
from the command line (bash). If make
is not available, you need to install a package. (The following is an example for Ubuntu Linux)
On Windows, run {MWSTAGE SDK install}/MWSDK/WIN_BASH.cmd
. Environment variables and make utility are already set.
A build should look like this:
When the build is done, the objs_???
folder is created and an intermediate file is created in it. This file is dependent on the environment in which it was compiled, so if any files from other environments are left, make will fail.
Since virtual functions (virtual) and run-time type information (RTTI) are not available, and even if they were available, they would be difficult to perform, is used as an alternative design method. CRTP is a template pattern for calling methods of a child class from the parent class from which it is inherited.
For formatting output, we use Marco Paland's , which needs to be implemented for use with the MWX library. In the following example, the derived class serial_jen
needs to define the vOutput()
method for 1-byte output, and save the auxiliary information for output in the parent class pvOutputContext
since vOutput()
is a static method. The other is to save the auxiliary information in the pvOutputContext
of the parent class since vOutput()
is a static method.
The MWX library does not use virtual functions due to compiler limitations and performance reasons. We use a to achieve polymorphism.
Due to the of the C/C++ compiler for TWELITE, only a few features are available.
- PATH must be set so that the code
command can be executed.
This page describes several methods of building, all of which ultimately involve running the make command. Please refer to the for details.
If it is a small act, you can write it in this .cpp
file. If you have a larger act, you can build it in multiple files by referring to the .
Launch the executable TWELITE_Stage.{extension}
located in the {TWELITE SDK installation} folder. (reference: )。
(Reference: )
(Reference: )
(Reference: )
(Reference: )
Reference: )
(Reference: )
When the programming is successfully completed, it will continue to (settings screen). However, the screen will not be displayed unless the firmware supports Interactive settings mode.
(Reference: )
(Reference: )
See the for more details.
CXXFLAGS
Specify compilation options for C++ source files.
CFLAGS
Specify compile options for C/C++ source files.
INCFLAGS
Specify the include file for the header file.
OPTFLAGS
Define this if you have a special reason for wanting to apply a compile option other than -Os
.
LDFLAGS
Specify linker options. (This is not mentioned in the comments of the Makefile above, but can be specified)
make TWELITE=BLUE
build for TWELITE BLUE
make TWELITE=RED
build for TWELITE RED
make cleanall
Delete intermediate files
The parent device is started with the M1 pin low (GND level). In normal mode (always running), you can see it works like App_TweLite.
The system works with two children. When one of them sends a ping packet, the other sends a pong packet back.
Other
You can check the transmission of packets of the act for the child machine.
Sample Acts
To help you understand how the ACT works, we have prepared some samples.
Several samples are provided to help you understand how the act works.
This is an example of a wireless sensor implementation that connects an I2C sensor and sends wireless packets while performing a brief operation by sleeping. Since this is a relatively simple and typical structure, it is recommended that you refer to it after reviewing act0
through act4
.
These are samples of sending or receiving wireless packets, each implemented from a slightly different perspective.
Refer to this when implementing your own Receiving Parent Node application.
The explanation of the ACT using Interactive settings mode describes the general flow of the program. There are no major differences between the explanations for any of the samples. (Here we quote BRD_I2C_TEMPHUMID)
This sample obtains sensor information from built-in peripherals and external sensor devices.
Although a standard PAL application is written into TWELITE PAL, it is possible to write a description using ACT without using the PAL application. The MWX library provides standard procedures for MOTION SENSE PAL use.
This is a sample for various PAL boards. This sample acquires sensor values, transmits these valuse, and sleeps on the PAL board.
The following is an example of an advanced application, which is a little more complicated to describe than the above ACT.
The following items are common settings in the Act sample and are explained below.
Other platforms
Must be able to compile in C++11.
Ability to use C++11 standard library headers (utility, algorithm, functional, iterator, etc.)
new/delete/virtual are not used.
Memory allocation by new may be used in exceptional cases.
In serparser/pktparser, alloc_heap which uses new operator is processed by delete.
(Reference) However, the mwx library has been designed on the assumption that delete is not taken into account.
act0
Templates with no processing description
act1
L-tica (flashing LED)
act2
L-tica with timer
act3
L-tica with two timers
act4
LED lighting using a button (switch)
Parent application (for MONOSTICK)
This act uses MONOSTICK as a parent device. It outputs the data payload of packets from the child machine to the serial port. It can display packets in many samples of sample acts.
This act includes
Receiving wireless packets
Data interpretation of received packets
Receives packets from the child of the sample act and outputs them to the serial port.
Parent Device
Child Device
Sample Act Child Setup (e.g. Slp_Wk_and_Tx
, PAL_AMB
, PAL_MAG
, PAL_MOT???, etc...
)
Please check the following default settings at first.
Application ID: 0x1234abcd
Channel: 13
<NWK_SIMPLE> Loads the definition of a simple relay net
<STG_STD> Loads the interactive mode definition
Declaration of default values, function prototypes, etc.
In setup()
, first load the <MONOSTICK>
board behavior, the <STG_STD>
interactive mode behavior, and the <NWK_SIMPLE>
behavior using use<>
. This procedure is always done in setup()
.
appname
→ Act name that appears in the title line of the configuration screen
appid_default
→ Default Application ID
ch_default
→ Default channel
lid_default
→ Default value of device ID (LID)
.hide_items()
→ Item Hide Settings
Always call .reload()
before reading the configuration values. Each set value has its own method for reading, such as .u32opt1()
.
Procedures for controlling LED lighting are available in the <MONOSTICK>
board behaviour.
The first line sets the red LED to switch on for 200 ms after receiving a radio packet. The first parameter LED_TIMER::ON_RX
means when a radio packet is received; the second specifies the lighting time in ms.
The second line specifies the blinking of the LEDs: the first parameter LED_TIMER::BLINK
specifies the blinking, the second parameter is the blinking on/off switching time: every 500ms the LEDs are switched on and off (i.e. Repeat blinking with a 1 s cycle).
Procedure for starting the_twelite
, which did not appear in act0..4, but must be called if you have configured the_twelite
or registered various behaviours.
There is no processing during loop()
in this sample.
Callback function called when a packet is received. In this example, some output is produced for the received packet data.
The analyze_payload()
called at the end of the function contains code to interpret some sample act packets. Refer to the code in correspondence with the packet generation part in the sample act.
This function first reads the four-character identification data into the fourchars[4]
array.
Processing is then carried out for the 4-byte header. Here, the packets of the sample act Slp_Wk_and_Tx are interpreted and the contents are displayed.
Set b_handled
to true so that the other interpreters' decisions are skipped.
"TXSP" packets contain the values of a system timer count of type uint32_t
and a dummy counter of type uint16_t
. Each variable is declared and read using the expand_bytes()
function. The difference from the above is that the first parameter is np
as the first pointer to read. The tick_ms
and u16work_ct
are given as parameters and the value stored in the payload is read as a big-endian format byte sequence.
If the readout is successful, the contents are output and the process is complete.
It is structured by ASCII format in a user-defined order.
The first line declares a buffer as a local object to store the data sequence before conversion to ASCII format.
The second line uses pack_bytes()
to store the data sequence into the buf mentioned earlier. See comments in the source code for the data structure. The pack_bytes()
parameter can also be a container of the form smplbuf_u8 (smplbuf<uint8_t, ALLOC>)
.
Lines 13, 14 and 17 are the declaration, configuration and output of the serial parser.
The first output (which is prevented from being executed by if(0)
) displays all data including the control data of <NWK_SIMPLE>
. There are 11 bytes of control data. Normally, the control information is not directly referenced.
The second line sets the buffer for the serial parser. It specifies an already existing data array, i.e. the payload part of the incoming packet. serparser_attach pout
declares the serial parser using an already existing buffer. The first parameter of pout.begin()
specifies the corresponding format of the parser as PARSER::ASCII
, i.e. ASCII format; the second specifies the first address of the buffer; the third specifies the The length of valid data in the buffer and the fourth specifies the maximum length of the buffer. The fourth parameter has the same value as the third, as it is for output and is not used for format interpretation.
Output to the serial port in line 6 using the >>
operator.
The Serial << mwx::flush
in line 7 is a specification to wait until the output of data that has not been output here is finished. (Serial.flush()
is the same process.)
Template code.
This act includes
Sending radio packets ('t
' key)
Sleep ('s
' key)
Set the_twelite
to set the application ID APP_ID
, the radio channel CHANNEL
and the receive presence.
It also generates nwk
and specifies the child machine address 0xFE
. This address means that this is an unnamed child machine that has not been addressed by a child machine.
This is the procedure to start the_twelite
, it didn't appear in act0..4, but you should call it if you set up the_twelite
or register various behaviors.
Called only once after setup()
on startup. Only displays a message.
If Serial.available()
is true
, the input from the serial port is stored. It reads one character from the serial and processes it according to the input character.
t
for wireless transmissionWhen 't
' is input, sending is done. In this sample, the tx_busy flag is used to prevent continuous input.
s
to sleep.The system will sleep for 5000ms=5 seconds. After recovery, wakeup()
is executed.
First to be called on sleep wake up. Display of message only.
This is the minimum procedure for making a transmission request.
When you leave this function, the request has not yet been executed. You need to wait a while. In this example we have set a delay of 100-200ms for the start of the transmission, so the transmission will not start until 100ms at the earliest.
Called on completion of a transmission; ev contains the transmission ID and completion status.
When a packet is received, the sender's address information is displayed.
Slp_Wk_and_Tx is a template source code for an application which, after a regular wake-up, does something (e.g. acquires sensor data) and sends the result as a wireless packet.
This act includes
The control structure of a typical intermittent operation (sleep -> wake -> measure -> radio transmission -> sleep)
Generation and transmission procedures for outgoing packets and waiting for completion
After starting up, the system goes through an initialization process and then goes to sleep.
setup()
Initialise
begin()
Run sleep
After waking up from sleep, the state variables are initialized and the actions are performed in the following order
wakeup()
wakes up from sleep, performs each initialization
loop()/INIT->WORK_JOB state
: does some processing (in this Act, the counter is updated every TickCount for 1ms and moves to TX state after a count determined by a random number)
loop()/TX state
: Make a request to send
loop()/WAIT_TX state
: Waiting for transmission completion
loop()/EXIT_NORMAL state
: Run sleep (back to 1.)
loop()/EXIT_FATAL state
:Resetting the module if an error occurs
To send packets, <NWK_SIMPLE>
is included. Also, basic definitions such as application ID are in "Common.h"
.
In order to describe the sequential processing in loop()
, this sample uses the concept of a state machine (state transition). It uses <SM_SIMPLE>
, which summarizes the processing of very simple state transitions.
An enum STATE
is defined in Common.h
for the following states.
The step
declared here contains functions for managing state, timeouts and waiting for processing.
In this sample we do not process the sensor data, but we prepare dummy data for explanation.
Initializes variables and class objects.
Initialization of the step
state machine
Initialization of the_twelite
class object
Register and initialize the network <NWK_SIMPLE>
(register DEVICE_ID
).
This is followed by the initiation of class objects and hardware.
This is the procedure to start the_twelite
, it didn't appear in act0..4, but you should call it if you set up the_twelite
or register various behaviors.
Called only once, immediately after setup()
. The SleepNow()
function is called to perform the first sleep procedure.
Called immediately after waking up. Here, it initializes the sensor data area and outputs a message on waking.
The above code is a simplified version of the actual code.
step.b_more_loop()
is set to true if there is a state transition by .next()
. The purpose of this is to execute the code of the next state (case clause) without escaping loop()
when a state transition occurs.
Below is a description of each state.
Initialises the sensor values of the dummies. One is determined randomly by an add counter and one by a counter stop value.
In the WORK_JOB state, we work on a timer every 1ms; every Tick timer becomes TickTimer.available()
; every Tick timer adds a counter and when it reaches dummy_work_ct_max
, we move to the next state STATE::TX
.
Call the Transmit()
function to request packet transmission. If the request succeeds, the function transits to STATE::WAIT_TXEVENT
and waits for the completion of transmission. Here, we use the timeout and flag functions of the SM_SIMPLE state machine to wait for completion (it is simple to determine the value of a variable by changing its value in the waiting loop).
We don't usually expect a single request to fail, but if it does, it will go to the STATE::EXIT_FATAL
exception handling state.
You should not sleep at this point, as the packet has not yet been sent. In most cases, you should wait for transmission to complete before continuing.
Waiting for completion of transmission is judged by setting the flag of the state machine function with on_tx_comp()
described later. The timeout is judged by the elapsed time since .set_timeout()
was done by calling .is_timeout()
.
Normally you will get a completion notice whether the transmission succeeds or fails, but it will time out and go to STATE::EXIT_FATAL
for exception handling.
Call SleepNow()
to start the sleep process.
As a critical error, a system reset is performed.
Periodic sleep is performed. The sleep time is set to a constant time blur using the random()
function. This is because when the transmission cycles of several devices are synchronized, the failure rate may increase significantly.
Before sleeping, the state of the SM_SIMPLE state machine is set by calling .on_sleep()
.
This function requests to send a wireless packet to the parent device with ID=0x00
. The data to be stored is a four-character identifier (FOURCC
) commonly used in the Act sample, plus the system time [ms] and a dummy sensor value (sensor.dummy_work_ct_now
).
The first step is to get an object that contains the transmitted packets. This object can then be manipulated to set the data and conditions for transmission.
In the mwx library, the if
statement is used to get an object and the bool
decision of the object is true
. Here, a board object is retrieved by the_twelite.network.use<NWK_SIMPLE>()
, and a packet object is retrieved by .prepare_tx_packet()
of the board object. Failure to retrieve the packet object is not normally expected, but when it does occur, it is when the transmit queue is full and the transmit request cannot be accepted. Since this example is for a single transmission only, the error is limited to an unexpected serious problem.
This function can be specified by a variable number of arguments. The first parameter is an array object obtained from .get_payload()
.
make_pair(FOURCC,4)
: make_pair is from the C++ standard library and creates a std::pair object. It means to write out 4 bytes from the beginning for string type.(This is done to explicitly specify the number of bytes to be written, as the topic of including or excluding the end of an array of string types is confusing)
If uint32_t
type data is specified, 4 bytes of data are written in big-endian order.
The same applies to data of type uint16_t
.
Finally, .transmit()
is called to request sending. The return value is of the type MWX_APIRET
. After the request, the actual transmission takes place, which may take several to several tens of milliseconds to complete, depending on the transmission parameters and the size of the transmission. On completion, on_tx_comp()
is called.
This is a system event that is called when the transmission is complete. Here, it is set to complete by .set_flag()
.
Send a PING wireless packet from one of the two serially connected TWELITEs and receive a PONG wireless packet back from the other.
This ACT includes.
Sending a prompt response from the receipt of a wireless packet
Transmission with direct address of the peer
Two of any of the following.
Sample act common declarations
Prototype declarations for longer processes (sending and receiving), since they are made into functions
Variables for holding data in the application
The general flow of the program is the initial setup of each section and the start of each section.
This object is the core class object for manipulating TWENET.
the_twelite
に設定を反映するには <<
を用います。
Use <<
to reflect the setting in the_twelite
.
TWENET::appid(APP_ID)
to specify the Application ID.
TWENET::channel(CHANNEL)
to specify the channel.
TWENET::rx_when_idle()
Specifies that the receive circuit is open.
Next, register the network.
The first line is written in the same way as the board registration, specifying <>
as <NWK_SIMPLE>
.
The second line specifies <NWK_SIMPLE>
, specifying 0xFE
(WK_SIMPLE
is a Child Node with an unset ID).
The third line specifies the maximum number of relays. This explanation does not touch on relaying, but packets are relayed when operating with multiple units.
Execute the_twelite.begin()
at the end of the setup()
function.
Class object that handles ADCs (analog-to-digital converters).
Initialization Analogue.setup()
. The parameter true
specifies to wait in place until the ADC circuit is stable.
To start the ADC, call Analogue.begin()
. The parameter is a bitmap corresponding to the pin to be ADC'd.
The pack_bits()
function is used to specify the bitmap. It is a function with variable number of arguments, each argument specifies a bit position to be set to 1. For example, pack_bits(1,3,5)
returns the binary value 101010
. This function has the constexpr
specification, so if the parameters are constants only, they are expanded to constants.
The parameters are specified as PIN_ANALOGUE::A1
(ADC0) and PIN_ANALOGUE::VCC
(module supply voltage).
The second parameter is specified as 50
, and the ADC operation is started by default with TickTimer, which is set to
Detects changes in DIO (digital input) values; Buttons only detect a change in value after the same value has been detected a certain number of times in order to reduce the effects of mechanical button chattering.
Initialization is done with Buttons.setup()
. The parameter 5 is the number of detections required to determine the value, but it is the maximum value that can be set. Internally, the internal memory is allocated based on this number.
The start is done with Buttons.begin()
The first parameter is the DIO to be detected. The second parameter is the number of detections required to determine the state. The third parameter is the detection interval. Since 10
is specified, the HIGH and LOW states are determined when the same value is detected five times in a row every 10 ms.
Serial objects can be used without initialization or initiation procedures.
Outputs a string to the serial port. mwx::crlf
is a newline character.
Loop function are called as callback functions from the main loop of the TWENET library. The basic description here is to wait for the object to be used to become available and then process it. This section describes the use of some objects used in ACT.
The main loop of the TWENET library processes incoming packets and interrupt information stored in the FIFO queue in advance as events, after which loop()
is called. After exiting loop()
, the CPU enters DOZE mode and waits until a new interrupt occurs with low current consumption.
Therefore, code that assumes the CPU is always running will not work well.
While Serial.available()
is true
, there is input from the serial port. The data is stored in the internal FIFO queue, so there is some leeway, but it should be read out promptly. To read data, call Serial.read()
.
Here, the vTransmit()
function is called to send a PING packet in response to a 't'
key input.
It becomes available at the timing when a change in DIO (digital IO) input is detected, and is read by Buttons.read()
.
The first parameter is a bitmap of the HIGH/LOW of the current DIO, ordered from bit0 to DIO0,1,2,... . and so on, starting from bit 0. For example, for DIO12, HIGH / LOW can be determined by evaluating btn_state & (1UL << 12)
. If the bit is set to 1, it is HIGH.
The vTransmit()
is called at the timing when the button is released except for the initial confirmation. To make the timing of the press (! (btn_state && (1UL << PIN_BTN)))
to invert the condition logically.
This function requests TWENET to send a wireless packet. At the end of this function, the wireless packet is not yet processed. The actual transmission will be completed in a few ms or later, depending on the transmission parameters. This section describes typical transmission request methods.
Get a network object with the_twelite.network.use<NWK_SIMPLE>()
. Use that object to get a pkt
object by .prepare_tx_packet()
.
Here it is declared in the conditional expression of the if statement. The declared pkt
object is valid until the end of the if clause. pkt object gives a response of type bool, which here is true
if there is a free space in TWENET's send request queue and the send request is accepted, or false
if there is no space.
Packets are configured using the <<
operator as in the_twelite
initialization setup.
Specify the destination address in the tx_addr()
parameter. If it is 0x00
, it means that you are the Child Node and broadcast to the Parent Node, and if it is 0xFE
, it means that you are the Parent Node and broadcast to any Child Node.
The tx_retry()
parameter specifies the number of retransmissions. In the example 3
means that the number of retransmissions is 3, i.e., the packet is sent 4 times in total. Sending only one wireless packet may fail a few percent of the time even under good conditions.
tx_packet_delay()
Sets the transmission delay. The first parameter is the minimum wait time to start sending and the second is the maximum wait time. The third is the retransmission interval. The third is the retransmission interval, meaning that a retransmission is performed every 20 ms after the first packet is sent.
Payload means a loaded item, but in wireless packets it is often used to mean "the main body of data to be sent". In addition to the main body of data, the data in a wireless packet also contains some auxiliary information, such as address information.
For correct transmission and reception, please be aware of the data order of the data payload. In this example, the data order is as follows. Construct the data payload according to this data order.
The data payload can contain 90 bytes (actually a few more bytes).
Every byte in an IEEE802.15.4 wireless packet is precious. There is a limit to the amount of data that can be sent in a single packet. If a packet is split, the cost of the split packet is high because it must take into account transmission failures. Also, sending one extra byte consumes energy equivalent to approximately 16 µs x current during transmission, which can be significant, especially for battery-powered applications. {endhint %}
Let's actually build the data structure of the above data payload. The data payload can be referenced as a container of type simplbuf<uint8_t>
via pkt.get_payload()
. In this container, we build the data based on the above specification.
It can be written as above, but the MWX library provides an auxiliary function pack_bytes()
for data payload construction.
The first parameter of pack_bytes
specifies the container. In this case, it is pkt.get_payload()
.
The parameters after that are variable arguments, specifying as many values of the corresponding type in pack_bytes
as needed. The pack_bytes
internally calls the .push_back()
method to append the specified value at the end.
The third line, make_pair()
, is a standard library function to generate std::pair
. This is specified to avoid confusion of string types (specifically, whether or not to include null characters when storing payloads). The first parameter of make_pair()
is the string type (char*
, uint8_t*
, uint8_t[]
, etc.) The second parameter is the number of bytes to store in the payload.
Lines 4, 5, and 6 store values of numeric types (uint8_t
, uint16_t
, uint32_t
). Numeric types such as signed, or even the same numeric type such as char
are cast to the three types listed on the left and submitted.
analogRead()
and analogRead_mv()
get the result of ADC. The former is the ADC value (0..1023) and the latter is the voltage[mv](0..2470). The supply voltage of the module is read internally from the value of the voltage divider resistor, so we use adalogRead_mv()
to perform that conversion.
This completes the packet preparation. Now all that remains is to make a request for transmission.
Packets are sent using the pkt.transmit()
method of the pkt
object.
This is the process when there is an incoming packet.
First, the data of the incoming packet is passed as parameter rx
. From rx
, the address information and data payload of the wireless packet is accessed.
In the next line, the received packet data refers to the source address (32-bit long address and 8-bit logical address) and other information.
The MWX library provides a function expand_bytes()
as a counterpart to pack_bytes()
used in transmit()
.
Lines 1 through 3 specify variables to store data.
The first parameter specifies the first iterator of the container (a uint8_t*
pointer), which can be retrieved by the .begin()
method. The second parameter is the next iterator after the end of the container and can be retrieved with the .end()
method.
The third and subsequent parameters enumerate variables. The payloads are read and stored in the order in which they are listed.
The process sends a PONG message if the identifier of the 4-byte string read in msg
is "PING"
.
It then displays information on packets that have arrived.
The format()
is used because numeric formatting output is required. helper class that allows the same syntax as printf() for >>
operators, but limits the number of arguments to a maximum of 8 (for 32-bit parameters). (A compile error will occur if the limit is exceeded. Note that Serial.printfmt()
has no limit on the number of arguments.)
The mwx::crlf
specifies a newline character (CR LF), and mwx::flush
waits for completion of output. (mxw::flush
may be written as Serial.flush()
)
This ACT includes
Sending and receiving wireless packets
Uses the environmental sensor PAL AMPIENT SENSE PAL to acquire sensor values.
Use the sleep function to operate with coin cell batteries.
The first step is to initialize variables, etc. Here we are initializing the state machine step.
The next step is to initialize and read out the Interactive settings mode.
Here you can get the set object, reflect the application name, reflect the default Application ID, and delete unnecessary items in the settings menu.
Next, the state of the SET pin is read. Since this sample operates intermittently in sleep mode, Interactive settings mode transition by +++ input is not possible. Instead, the sample transitions to Interactive settings mode with the SET pin = LO state at startup. In this case, SETTINGS::open_at_start()
is specified, which means that the interactive mode screen will be displayed as soon as setup()
is finished.
Finally, .reload()
is executed to read the set values from the EEPROM. The configuration values are copied to each variable.
Next, configure the LED settings. (In an application that sleeps and wakes for a short period of time, this is almost the same as setting the LED to turn on during wake-up.)
Since this ACT exclusively transmits wireless packets, the TWENET configuration does not include a specification (TWENET::rx_when_idle()
) to open the receive circuit during operation.
The sensors on the board use the I2C bus, so start using the bus.
It is not convenient to have the main loop running during Interactive settings mode, so it is fixed in this state.
Start sensor data acquisition.
The sensors on the board can be accessed with the name .sns_LTR308ALS
or .sns_SHTC3
, and operations are performed on this object. Wait for the sensor to complete. If the sensor has not yet been acquired (.available()
is false
), a time elapsed event (.process_ev(E_EVENT_TICK_TIMER)
) is sent to the sensor.
When the above two sensors become available, the sensor value is retrieved and a transition is made to STATE TOK_TX.
The illuminance sensor can be obtained with .get_luminance() : uint32_t
.
The temperature and humidity sensor can be obtained as follows.
.get_temp_cent()
: int16_t
: temperature with 1℃ as 100 (2560 for 25.6℃)
.get_temp()
: float
: float value (25.6 for 25.6 ℃)
.get_humid_dmil()
: int16_t
: humidity at 1% as 100 (5680 for 56.8%)
.get_temp()
: float
: float value (56.8 for 56.8%)
The transmission procedure is the same as in the other ACT samples. Here, the settings are set to minimize one retransmission and retransmission delay.
The identifier FOURCHARS
and the sensor data are stored in the payload part of the packet. Of the values obtained, the temperature value is int16_t
, but is cast to uint16_t
because the data structure of the outgoing packet is to be stored unsigned.
Requests transmission. If the send request succeeds, prepare the send completion city. Specify .clear_flag()
to wait for the completion event and set_timeout(100)
for timeout in case of emergency. The unit of 100 in the parameter is milliseconds [ms].
This section determines timeouts and transmission completion events.
Processes sleepNow()
.
This is a system event called when transmission is complete. Here, .set_flag()
is used to indicate completion.
This section includes a collection of procedures for going to sleep.
Initialize the state of the state machine by .on_sleep(false)
before sleep. The parameter false' starts from
STATE::INIT(=0)` after returning from sleep.
Here, the time to wake up is set between 1750ms and 2250ms by a random number. This avoids continuous collisions with packets from other devices transmitting at a similar period.
Lines 8 and 9, this example goes to sleep waiting for output from the serial port. Usually, we want to minimize energy consumption, so we minimize (or eliminate) output from the serial port before sleep.
Line 12, to enter sleep, call the_twelite.sleep()
. In this call, the pre-sleep procedures of the hardware on the board are performed. For example, LEDs are turned off.
The sleep time is specified in ms as a parameter.
TWELITE PAL must always wake up once within 60 seconds to reset the watchdog timer. The sleep time must be specified not to exceed 60000
.
When the program wakes up from sleep, wakeup()
is called. After that, loop()
is called each time. Before wakeup()
, each peripheral such as UART and devices on the board are woken up. For example, it restarts the LED lighting control.
is a very simple example, without any radio functions, to give you an idea of the basic structure of Act.
Typical elements for implementing wireless sensors in TWELITE (, , , intermittent operation with sleep, etc.).
is a simple code that receives 1 byte commands from the UART, sends them and so on.
uses a state machine and intermittent operation with sleep, repeating the process of sleep recovery, radio transmission and sleep.
is a sample of sending packets from one side to the other and receiving packets back by the other side. It contains basic procedures for sending and receiving.
interprets the UART input into ASCII format using and sends it.
Note: can also be used to receive wireless packets for the ACTs included in this sample.
exclusively receives and outputs the reception result to the serial port. The wireless packets in this sample, which are addressed to the parent device (0x00) or to the child device broadcast (0xFE), can be received. It also includes a procedure to add the interactive mode to Act.
is an example code for universal packets receiver (e.g. TWENET layered tree network, App_Twelite, Act, ...). It alse uses EASTL library for container or some algorithm.
executes I2C sensor device read/write commands and wirelessly transmits measurements obtained from I2C sensors. It also uses the Interactive settings mode to Act.
Setting provides a higher degree of customisation of the interactive mode .
executes I2C sensor device read/write commands and wirelessly transmits measurements obtained from I2C sensors. It also uses the Interactive settings mode to Act.
provides bidirectional communication using digital input, analogue input, digital output and analogue output. It also contains the procedure for adding the interactive mode to Act.
uses a pulse counter function to count the number of pulses detected on the input port, even during sleep, and to transmit this data wirelessly.
is a behavioural example: in PAL_AMB the temperature and humidity sensors are called by the code inside the library, but this sample also includes its own procedure for accessing the temperature and humidity sensors via I2C bus.
is a sample that aims to save more power by allowing the TWELITE microcontroller to sleep briefly during the sensor's operating time, which can take several tens of milliseconds.
is a behavioural example: in PAL_AMB the temperature and humidity sensors are called by the code inside the library, but this sample also includes its own procedure for accessing the temperature and humidity sensors via I2C bus.
for continuous acquisition and wireless transmission of samples without interruption, using the accelerometer's FIFO and FIFO interrupts.
Acts with names starting with are intended to introduce features and APIs.
Build definitions are provided so that some features (, , Serial object for console) can be built on other platforms. Only the necessary files are cut out.
The build definitions are stored in the {mwx library storage}/stdio folder. See (link is on GitHub) for build instructions.
The acts starting with act0 are the ones introduced in , and although they are simple, we recommend you try them out first.
Interactive mode settings -
Conversion of byte strings to ASCII format -
Include board behavior for . This board support includes LED control and watchdog support.
The interactive mode is then set up and the settings are read out. The interactive mode provides standard items, but allows for some customization for each act you create.
Some settings can be directly reflected using objects. In addition, if you want to rewrite a specific value due to a DIP switch setting, for example, you can rewrite the value separately after it is reflected. In the above example, the application ID, channel, radio output, etc. are set in object, the LID and the retransmission count are set in the object, and then the LID is set to 0 again.
Reading is done using the function.The first and second parameters of this function follow the C++ standard library's practice of giving the first pointer .begin()
and the next .end()
of the payload section of the incoming packet. The following parameters are variable arguments, giving the data variables to be read. The return value is nullptr in case of an error, otherwise the next interpretation pointer. If interpreted to the end, .end()
is returned. The parameter here is uint8_t fourchars[4]
.
The first line declares the serial parser for output as a local object. It does not have an internal buffer, but diverts an external buffer and uses the output function of the parser to output the byte sequence in the buffer in
Input from serial port -
Digital (button) input -
It also initializes the Buttons object. This is a chatter-suppressing algorithm by successive references, which determines HI or LOW of the target port (PIN_BTN only) if the value is the same 5 times in 10ms. (N1, N2, ...)
pack_bits(N1, N2, ...)' does 1UL<<N1 | 1UL << N2 | ...
to make a bitmap.
The state is determined by continuous reference to . When the button state changes, it is output serially.
In the form of setup()
and loop()
, conditional branches which are difficult to read in loop()
tend to occur. In this Act, we use in loop()
and simple state transition by switch syntax to improve the visibility of the code.
Declares an (state transition) using the STATE
state enumeration.
This control structure uses the . It is a loop with do..while()
syntax. Inside the loop is a _switch case
_ clause, which splits the process according to the state obtained by .state()
. State transitions call .next()
to rewrite internal variables in the state machine to the new state values.
For the resulting pkt
object, set the conditions for transmission (destination, retransmission, etc.) using the <<
operator. specifies the destination of the packet. The tx_addr
specifies the destination of the packet, the specifies the number of retransmissions, and the specifies the transmission delay.
The payload of a packet is an array of derivatives obtained by pkt.get_payload()
. You can set the value of this array directly, but here we use to set the value.
The maximum length of the payload is 91 bytes in the above example, see for more information.
`` is a wrapped class of uint32_t
type, which uses MSB as a failure success flag and 31 bits as data. It is a return type of pkt.transmit()
, and success or failure of the request (cast to bool
type) and the ID are stored in the data part (.get_value()
).
Input from serial port -
Digital (button) input -
Analogue input -
connected to UART with products/TWE-Lite-DIP/index.html), etc.
Include <TWELITE>
in all ACTs. Here, the simple network should be included.
description sample. For details, please refer to for details.
is used to acquire sensor values.
Configuring settings via Interactive settings mode -
State transition control by state machine -
Board operation with board behavior
Environment sensor pal ) include board behavior.
First, the board support is registered. The sensors and DIOs are initialized when the board support is initialized. The reason for doing this first is that it is common to check the status of the board's DIP SW, etc., and then configure network settings, etc.
Starts the acquisition of a sensor on the board. See the description of .
The loop()
is controlled by the step
for control. This is to concisely represent the sequence of events from sleep recovery, sensor value acquisition, wireless packet transmission, waiting for transmission to complete, and sleep. In the combat of the loop, a brd
object is acquired.
ACT can operate with lower energy consumption by sleeping while waiting for sensor data acquisition.
Parent Node
Child Node
IO communication (basic function of standard application App_Twelite)
This is a sample using board support <BRD_APPTWELITE>
, which assumes the same wiring as required by App_TweLite.
This ACT includes:
Transmission and reception of wireless packets
This sample cannot communicate with App_TweLite.
Read M1 to determine the parent or child unit.
Reads the value of DI1-DI4; the Buttons class reduces the effect of chattering, so that changes are notified only when the values are the same consecutively. Communication is performed when there is a change.
Reads the value of AI1-AI4.
Sends the values of DI1-4, AI1-4, and VCC to a child unit if it is the parent unit, or to the parent unit if it is a child unit, every DI change or every 1 second.
Set to DO1-4 or PWM1-4 depending on the value of the received packet.
Parent Node
Child Node
It also includes <STG_STD>
to add Interactive settings mode.
sample-act common declaration
Its prototype declarations (send and receive), since the longer process is functionalized
Variables for holding data in the application
The general flow of the program is the initial setup of each section and the start of each section.
Register BEHAVIOR objects to determine system behavior. It can be an Interactive settings mode, board support, or a network description of wireless packets.
It will not work unless registered within setup()
.
Initialize the Interactive settings mode. First, a set
object is obtained. Then, the following process is performed.
Set the application name to "BRD_APPTWELITE"
(used in the menu)
Rewrite default Application ID and CHANNEL values
Delete unnecessary items
reads configuration values saved by set.reload()
.
Copy the values of OPT_BITS
and LID
into the variables
Below is an example screen. + + + and + three times with a pause of 0.2 to 1 second to bring up the Interactive settings mode screen.
This object behaves as the core of TWENET.
Register a board (this ACT registers <BRD_APPTWELITE>
). Specify the name of the board you want to register with <>
after use
as follows.
The return value obtained by universal reference (auto&&
) is a board object of reference type. This object contains board-specific operations and definitions. The following example uses the board object to check the status of the M1 pin: if the M1 pin is LO, set LID = 0, i.e., the Parent Node address.
Initial configuration is required to make the the_twelite work. Application ID and wireless CHANNEL settings are mandatory.
Use <<
to apply the setting to the_twelite
.
TWENET::rx_when_idle()
Specification to open the receive circuit.
Next, register the network.
The first line is written in the same way as for board registration, where <>
is <NWK_SIMPLE>
.
The second and third lines are settings for <NWK_SIMPLE>
. The first one reflects the value of the settings in Interactive settings mode. The items to be reflected are LID and the number of retransmissions. In this application, LID is set again in the third line because LID=0 may be set depending on the state of the M1 pin.
Class object that handles ADCs (analog-to-digital converters).
The Initialization is performed by Analogue.setup()
. The parameter true
specifies to wait in place until the ADC circuit is stable; the second parameter specifies to start the ADC synchronously with Timer0.
To start the ADC, call Analogue.begin()
. The parameter is a bitmap corresponding to the pin of the ADC target.
The pack_bits()
function is used to specify a bitmap. It is a function with variable number of arguments, each argument specifies a bit position to be set to 1. For example, pack_bits(1,3,5)
returns the binary value 101010
. This function has the constexpr
specification, so if the parameters are constants only, they are expanded to constants.
Parameters and BRD_APPTWELITE::
specified as PIN_AI1..4
are defined, corresponding to AI1..AI4 used in App_Twelite. AI1=ADC1, AI2=DIO0, AI3=ADC2, AI4=DIO2 and so on are assigned. PIN_ANALOGUE. PIN_ANALOGUE::
defines the list of pins available for ADC.
Detects changes in DIO (digital input) values; Buttons only detect a change in value after the same value has been detected a certain number of times in order to reduce the effects of chattering (sliding) on mechanical buttons.
Initialization is done with Buttons.setup()
. The parameter 5 is the number of detections required to determine the value, but it is the maximum value that can be set. Internally, the internal memory is allocated based on this number.
Start is done with Buttons.begin()
The first parameter is the DIO to be detected. The second parameter is the number of detections required to determine the state. Since 4
is specified, the HIGH and LOW states are determined when the same value is detected five times in a row every 4ms.
Since App_Twelite uses a timer origin to control the application, the same timer interrupt/event should be run in this ACT. Of course, you can also use the system's TickTimer, which runs every 1 ms.
The first parameter in the above example specifies a timer frequency of 32 Hz; setting the second parameter to `true' enables software interrupts.
After calling Timer0.begin()
, the timer starts running.
Execute the_twelite.begin()
at the end of the setup()
function.
Serial objects can be used without initialization or initiation procedures.
This sample displays some system configuration values as a startup message. The Serial
object is given a string of type const char*, a string of type int (no other integer types), a format()
that behaves almost identically to printf, and a crlf
that outputs newline characters to the << operator.
Loop functions are called as callback functions from the main loop of the TWENET library. The basic description here is to wait for the object to be used to become available and then process it. This section describes the use of some of the objects used in ACT.
The main loop of the TWENET library processes incoming packets and interrupt information stored in the FIFO queue in advance as events, after which loop()
is called. After exiting loop()
, the CPU enters DOZE mode and waits until a new interrupt occurs with low current consumption.
Therefore, code that assumes the CPU is always running will not work well.
It becomes available at the timing when a change in DIO (digital IO) input is detected, and is read by Buttons.read()
.
The first parameter is a bitmap of the HIGH/LOW of the current DIO, ordered from bit0 to DIO0,1,2,... in order from bit0. For example, for DIO12, HIGH / LOW can be determined by evaluating bp & (1UL << 12)
. The bit that is set to 1 is HIGH.
Next, the values are extracted from the bitmap and stored in u8DI_BM
. Here, the collect_bits()
function provided by the MWX library is used.
collect_bits()
takes an integer argument of bit positions similar to pack_bits()
above. It is a function with a variable number of arguments and arranges the parameters as many as necessary. In the above process, bit0 is stored in u8DI_BM as the value of DI1, bit1 as the value of DI2, bit2 as the value of DI3, and bit3 as the value of DI4.
In App_Twelite, since wireless transmission is performed when there is a change from DI1 to DI4, the transmission process is performed starting from Buttons.available()
. The details of the transmit()
process are described below.
It becomes available at loop()
immediately after the analog-to-digital conversion of the ADC is completed. Until the next ADC starts, data can be read back as if it was acquired immediately before.
Use the Analogue.read()
or Analogue.read_raw()
method to read ADC values. read()
is the value converted to mV, read_raw()
is the ADC value of 0..1023. The parameter is the pin number of ADC, which is defined in PIN_ANALOGUE::
or BRD_APPTWELITE::
.
ADC values that are executed cyclically may read more recent values prior to the AVAILABLE notification, depending on the timing.
Since this ACT processes at a relatively slow cycle of 32 Hz, it is not a problem if the processing is done immediately after the AVAILABLE decision, but if the conversion cycle is short, or if you are doing a relatively long process in loop()
, be careful.
In Analogue
, you can specify a callback function that will be called from within the interrupt handler after the conversion is complete. For example, this callback function will store the value in the FIFO queue, and the application loop will perform asynchronous processing such as sequential reading of the queue value.
The Timer0
runs at 32 Hz. It becomes available at loop()
immediately after a timer interrupt occurs. In other words, it processes 32 times per second. Here, the transmission is processed when it reaches exactly 1 second.
AppTwelite performs a periodic transmission about every second. When Timer0
becomes available, it increments u16ct
. Based on this counter value, transmit()
is called to send a radio packet when the count is completed 32 times.
The value judgment of u8DI_BM
and au16AI[]
is whether or not it is just after initialization. If the values of DI1..DI4 and AI1..AI4 are not yet stored, nothing is done.
This function requests TWENET to send a radio packet. At the end of this function, the wireless packet is not yet processed. The actual transmission will be completed in a few ms or later, depending on the transmission parameters. This section describes typical transmission request methods.
MWX_APIRET
is a class that handles return values with data members of type uint32_t
, where MSB (bit31) is success/failure and the rest are used as return values.
Get a network object with the_twelite.network.use<NWK_SIMPLE>()
. Use that object to get a pkt
object with .prepare_tx_packet()
.
Here it is declared in a conditional expression in an if statement. The declared pkt
object is valid until the end of the if clause. pkt object gives a response of type bool, which here is true
if the TWENET send request queue is free and the send request is accepted, or false
if there is no free queue.
Suppresses screen output when Interactive settings mode screen is displayed.
Packets are configured using the <<
operator as in the initialization settings of the_twelite
.
Specify the destination address in the tx_addr()
parameter. If it is 0x00
, it means that you are the Child Node and broadcast to the Parent Node, and if it is 0xFE
, it means that you are the Parent Node and broadcast to any Child Node.
The tx_retry()
parameter specifies the number of retransmissions. The 1
in the example sends one retry, i.e., a total of two packets. Sending only one wireless packet will fail a few percent of the time even under good conditions.
tx_packet_delay()
Sets the transmission delay. The first parameter is the minimum wait time to start sending, the second is the maximum wait time, in this case approximately from 0 to 50 ms after issuing a request to send. The third is the retransmission interval. The third is the retransmission interval, meaning that a retransmission is performed every 10 ms after the first packet is sent.
Payload means a loaded item, but in wireless packets it is often used to mean "the main body of data to be sent". In addition to the main body of data, the data in a wireless packet also contains some auxiliary information, such as address information.
Be aware of the data order of the data payload for correct transmission and reception. In this example, the data order is as follows. Construct the data payload according to this data order.
Let's actually construct the data structure of the above data payload. The data payload can be referenced as a container of type simplbuf<uint8_t>
by pkt.get_payload()
. In this container, we build the data based on the above specification.
It can be written as above, but the MWX library provides an auxiliary function pack_bytes()
for data payload construction.
The first parameter of pack_bytes
specifies the container. In this case, it is pkt.get_payload()
.
The parameter after that is a variable number of arguments, pack_bytes
, specifying the required number of values of the corresponding type. The pack_bytes
internally calls the .push_back()
method to append the specified value at the end.
The third line, make_pair()
, is a standard library function to generate std::pair
. This is to avoid confusion of string types (specifically, including or excluding null characters when storing payloads). The first parameter of make_pair()
is the string type (char*
, uint8_t*
, uint8_t[]
, etc.) The second parameter is the number of bytes to store in the payload.
The fourth line is of type uint8_t
and writes a bitmap of DI1..DI4.
Lines 7-9 write the values of the au16AI
array in sequence. The values are of type uint16_t
and are 2 bytes, but are written in big-endian order.
This completes the packet preparation. Now all that remains is to make a request for transmission.
Packets are sent using the pkt.transmit()
method of the pkt
object. It returns a return value of type MWX_APIRET
, which is not used in this ACT.
When a wireless packet is received, on_rx_packet()
is called as a receive event.
Here, the values of DI1...DI4 and AI1...AI4 communicated by the other party are set to its own DO1...DO4 and PWM1...PWM4.
First, the data rx
of the received packet is passed as a parameter. From rx
the address information and data payload of the wireless packet is accessed. The parameter handled
is not normally used.
Received packet data is referenced to the source address (32-bit long address and 8-bit logical address) and other information. Output is suppressed when the Interactive settings mode screen is displayed.
The MWX library provides a function expand_bytes()
as a counterpart to pack_bytes()
used in transmit()
.
The first line declares an array of type char
for data storage. The size of the array is 5 bytes because the null character is included at the end for convenience in character output, etc. The trailing {}
specifies initialization. The trailing {}
specifies initialization, and although it is sufficient to set the fifth byte to 0, here the entire array is initialized in the default way, that is, to 0
.
The 4-byte string is extracted by expand_bytes()
in the second line. The reason for not specifying the container type in the parameters is that we need to know the read position to read this continuation. the first parameter specifies the first iterator of the container (uint8_t*
pointer), which can be obtained by the .begin()
. The second parameter is an iterator that points to the next to the end of the container and can be retrieved with the .end()
method. The second parameter is used to avoid reading beyond the end of the container.
The third variable to read is specified, again by make_pair
, which is a pair of string array and size.
If the identifier in the 4-byte string read is different from the identifier specified in this ACT, this packet is not processed.
The next step is to get the data part: store the values of DI1..DI4 and AI1..AI4 in separate variables.
The first parameter is the return value np
of the previous expand_bytes()
. The second parameter is the same.
The third and subsequent parameters are variables of the matching type alongside the data payload, in the same order as the sender's data structure. Once this process is complete, the specified variable will contain the value read from the payload.
Output to serial port for confirmation. Output is suppressed when the Interactive settings mode screen is displayed.
We use format()
because we need to format output of numbers. helper class that allows the same syntax as printf()
for the >>
operator, but limits the number of arguments to four. (Serial.printfmt()
has no limit on the number of arguments.)
The first line "DI:%04b"
prints a 4-digit bitmap of DI1..DI4, like "DI:0010"
The third line "/%04d"
prints the values of Vcc/AI1..AI4 as integers, like "/3280/0010/0512/1023/"
The fifth line mwx::crlf
prints a newline string.
Now that the necessary data has been extracted, all that remains is to change the values of DO1..DO4 and PWM1..PWM4 on the board.
digitalWrite()
changes the value of the digital output, the first parameter is the pin number and the second is HIGH
(Vcc level) or LOW
(GND level).
Timer?.change_duty()
changes the duty ratio of the PWM output. Specify a duty ratio of 0..1024 for the parameter. Note that the maximum value is not 1023 (the maximum value is 1024, a power of 2, due to the high cost of the division performed in the library). Setting the value to 0
will result in an output equivalent to the GND level, and setting it to 1024
will result in an output equivalent to the Vcc level.
The begin()
function exits the setup()
function (after which TWENET is initialized) and is called just before the first loop()
.
The first sleep is performed after setup()
ends. Although sensor data acquisition is started during setup()
, this result is not evaluated and is not necessarily necessary, since it means that the sensor is run once in advance.
Procedures after waking up. The following process is performed.
If the sensor data acquisition start has not yet been performed, the sensor data acquisition is performed and a short sleep is entered.
Since the start of sensor data acquisition was performed immediately before, the data is checked and sent wirelessly.
The above branch is controlled by the global variable b_sensor_started
. If !b_sensor_started
, it starts acquiring a sensor (startSensorCapture()
) and goes into a short sleep by napNow()
. The time is 100ms.
After returning from sleep by napNow()
, the clause b_sensor_started==true
is executed. Here, the E_EVENT_START_UP
event is notified to the two sensors. This event means that enough time has passed for the sensors to finish acquiring. Based on this notification, sns_LTR308ALS
and sns_SHTC3
are made available. After this, it will go to loop()
and wireless packets will be sent.
The event that notifies the sensor is used to determine if the required time wait is over. Whether or not the actual time has elapsed depends on whether or not the correct time was set with napNow()
. If the wake-up time is short, it is expected that the time elapsed is not enough to meet the required time elapsed and that subsequent processing will result in errors such as sensor data not being available.
Perform a very short sleep.
If the second parameter of sleep is set to true, the next wake-up time is adjusted based on the previous sleep wake-up time. This is useful if you always want to wake up every 5 seconds.
If the third parameter is set to true, the sleep mode is set without memory retention. After waking up, wakup() will not be called and the process will be the same as power-on.
The fourth specifies the use of the second wake-up timer. Here, the first is used for normal sleep and the second is used for short sleep. There is no strong reason to use the second timer in this ACT, but if, for example, the user wants to wake up every 5 seconds as described above, using the first timer for a short sleep would reset the counter value, which would complicate the elapsed time correction calculation, so the second timer is used.
The pulse counter counts the number of rising or falling pulses of a signal without intervening microcontroller. It can be used to count irregular pulses and send wireless packets when the count reaches a certain number of times.
Counts the pulses connected to DIO8 on the Child Node side and transmits them wirelessly after a certain period of time or when a certain number of counts are detected.
The Child Node side operates while sleeping.
Initializes the pulse counter.
Starts the pulse counter operation and performs the first sleep. The first parameter of PulseCounter.begin()
is the count number 100
to generate a wake-up interrupt, and the second is the falling detection PIN_INT_MODE::FALLING
.
Checks PulseCounter.available()
when waking up. available, that is, true
, indicates that the count is greater than or equal to the specified count. If it is false
, it resleeps.
If the count is more than the specified number, loop()
processes the transmission and waits for the completion of the transmission.
Reads the pulse count value. The counter is reset after the readout.
This is an Act sample using an I2C sensor device to measure temperature and humidity and send them periodically.
This Act includes following features:
Sending / Receiving wireless packets
Send / Receive I2C commands
Sleep periodically for running with a button cell.
Header files are <NWK_SIMPLE>
(Simple network support), <STG_STD>
(Interactive settings mode) and <SM_SIMPLE>
(Simple state machines).
For your reference, this Act is using SHTC3 (TWELITE AMB PAL) or SHT40 (TWELITE ARIA). You can switch them by editing #ifdef
and they have same function interfaces for code portability. Both codes are similar because they were designed by the same manufacturer. (Please #define
either USE_SHTC3
or USE_SHT40
.)
Below using the SHTC3.
The struct (class) SHTC3
has I2C-sensor-realted procedures. The member variable I2C_ADDR
is the I2C address and the CONV_TIME
is the waiting time for getting values. The instance created from the SHTC3
is sensor_device
.
Methods
Each procedures are shown below.
Set the I2C address and the sensor value acquisition wait time (10 ms for the above) in the member variables.
In principle, these values are fixed values, so there is no need to set them. Valid examples of how to treat them as variables: Managing the conversion time required when operating a sensor with higher precision or selecting an I2C sub-address by setting them.
Write commands to operate the sensor.
Wire.get_writer(I2C_ADDR)
opens the I2C device and returns a object for reading / writing data. If the device is not available, the object wrt
is evaluated as false
.
The line wrt << 0x60;
is writing a data to the device with the stream operator <<
. The operator is often used for writing data in uint8_t
.
Return the value of CONV_TIME
.
Get values from the sensor device.
SHTC3 reads the sensor value after waiting a few ms after starting the sensor readout by begin()
. The sequence of sensor values is as follows.
In SHTC3, the order of data also changes depending on the parameters given at the start of sensor acquisition, but if you start with the 0x609C
command written by the above begin()
, temperature data comes first.
In begin()
, the data was written out, but here the data is read in. To read the data, similarly, Wire.get_reader()
creates a helper object rdr
. If there are no errors, rdr
returns true in the if clause. The second parameter 6
given to get_reader(I2C_ADDR, 6)
is the number of bytes to read. When this number of bytes have been read, the procedure terminates reading the I2C bus. (Some devices may work even if these procedures are omitted, but usually you should give an appropriate value.)
Finally, 100 times the temperature [°C] and 100 times the humidity [%] are calculated and stored in i16Temp
and i16Humd
.Refer to the I2C device datasheet for the calculation formula.
Called once when the TWELITE is started. This function performs various initializations.
The state machine (state transition machine) is used to simplify the description in the loop()
statement that is called each time. Of course, you do not have to use SM_SMPLE
in this example to describe your application.
BEHAVIOR is a collection of functions used in a program. It describes how to behave when various events occur.
In this example, we use two kinds of BEHAVIOR: interactive settings mode screen <STG_STD>
and simple relay network <NWK_SMPLE>
.
SETTINGS::appname
: application name (string). It will be displayed on the first line on the interactive settings mode screen. Use a minimum string since there is not enough room for the number of characters on the screen.
SETTINGS::appid_default
: Default application ID. Execute this if you want your own application to have its own defined application ID.
SETTINGS::ch_default
: channel default. Execute if you want your own application to have its own default channel.
Then set.hide_items()
removes unnecessary settings items on the screen in the default interactive settings mode. If you do not mind displaying them all, you do not need to make this call.
This description starts in interactive settings mode when the DIO12 pin is LOW (GND level) and the device is powered up or reset. Read the pin status with digitalRead()
and reflect SETTINGS::open_at_start()
.
Set the state of the state machine to STATE::INTERACTIVE
, since it would be inconvenient if normal application processing is performed during interactive settings mode. In this state, no input or other processing is performed and the application stays in the same state.
Finally, load the data in interactive settings mode. By calling set.reload()
, data written to the EEPROM is read. If no setting is made and there is no information in the EEPROM, it can be read as the specified value.
Here, option bits set.u32opt1()
and 8-bit logical ID set.u8devid()
are read. child machine).
Finally, the_twelite
and nwk
reflect (some of) the configuration information. This reflects information essential for wireless communication, such as application ID and channel. There is no explicit read code for these settings in the above, but set.reload()
reads the settings to the specified values if they are not set, or to the set values if they are.
Initialization for the I2C device.
the_twelite.begin()
is a procedure that declares the completion of the initialization of the MWX library. Without this procedure, the MWX library will not work properly.
Messages at startup, etc. are also displayed here.
The control structure of the above do while statement is described below. The state is determined by step.state()
. The conditional expression of while is step.b_more_loop()
. This is because there are cases where you want to process continuously without leaving loop()
when a state transitions from one state to another. In other words, if you transition to another state and leave the switch clause, the case clause of the next state will be called. Be careful with this behavior.
It is not convenient for the main loop to operate during interactive settings mode, so it is fixed in this state.
Starts acquiring sensor data. The set_timeout()
statement waits for the sensor to acquire data.
Get the sensor value by sensor_device.read()
and store the value in the sensor
structure.
As a wireless sensor, there is no need to output the result to the serial port of TWELITE on the sensor side, but the acquired value is displayed on the serial port to facilitate operation check. Here, Serial.flush()
is executed to wait for output, which is a description assuming that the serial port output does not finish before TWELITE goes to sleep. This process also causes battery drain, so either do not do Serial.flush()
or do not output to the serial port.
Describes a communication procedure. In this state, there is no wait processing, etc., and the program transitions to the next state immediately after processing is executed. The reason why step.next(STATE::GO_SLEEP)
is written in advance is to avoid writing the same description in all places, since errors are detected in multiple places.
In if (auto&& pkt = the_twelite.network.use<NWK_SIMPLE>().prepare_tx_packet())
, the process is to create an object for the packet to be sent and execute the if clause if the object is successfully created.
First, configure the transmission settings. The destination tx_addr(0x00)
is set to the parent machine 0x00
, the number of retransmissions tx_retry(0x1)
is set to 1, and the packet delay setting tx_packet_delay(0, 0, 2)
is set to 0 delay for the first transmission and 2 ms for the retransmission interval. The packet delay is set to 0 for the first transmission and 2ms for the retransmission interval.
The identifier FOURCHARS
and the sensor data are stored in the payload part of the packet. Of the values obtained, the temperature value is int16_t
, but is cast to uint16_t
because the data structure of the outgoing packet is to be stored unsigned.
Call pkt.transmit()
to request a transmission. The MWX library will process the request at the appropriate time.
If the send request is successful, ret
will be true. Initialize the flags to determine completion step.clear_flag()
, set a timeout step.set_timeout(100)
to handle unexpected errors such as failed transmissions, and set the next state to STATE::TX_WAIT_COMP
(see STATE::GO_SLEEP
is overridden).
Here, it waits for the completion of transmission. Judgment of timeout (in case of error) or transmission completion event is made.
Processes the sleepNow()
function. By calling this function, the TWELITE wireless microcontroller goes to sleep.
This is a system event called when transmission is completed. Here .set_flag()
is called to set the flag of step
.
Procedures for going to sleep.
Initialize the state of the state machine by .on_sleep(false)
before sleep. The parameter false' starts from
STATE::INIT(=0)` after returning from sleep.
Here, the time to wake up is set between 1750ms and 2250ms by a random number. This avoids continuous collisions with packets from other devices transmitting at a similar period.
Lines 8 and 9, this example goes to sleep waiting for output from the serial port. Usually, we want to minimize energy consumption, so we minimize (or eliminate) output from the serial port before sleep.
Line 12, to enter sleep, call the_twelite.sleep()
. In this call, the pre-sleep procedures of the hardware on the board are performed.
The sleep time is specified in ms as a parameter.
TWELITE PAL must always wake up once within 60 seconds and reset the watchdog timer. The sleep time must be specified not to exceed 60000
.
When the program wakes up from sleep, wakeup()
is called. After that, loop()
is called each time. Before wakeup()
, each peripheral such as UART and devices on the board are woken up. For example, it restarts the LED lighting control.
WirelessUART performs serial communication.
Communicate between two UART-connected TWELITEs in ASCII format.
Two of the following devices serially connected to a PC.
Interactive settings mode is initialized. This sample provides two or more devices that have different logical device IDs (LIDs) from each other.
When data input from the serial is received, one byte is input to the serial parser. When the ASCII format is accepted to the end, SerialParser.parse()
returns true
.
The SerialParser
can access the internal buffer with smplbuf
. In the example above, the first byte of the buffer is taken as the destination address, and the second byte to the end is passed to the transmit()
function.
When a packet is received, a buffer smplbuf_u8<128> buf
containing the payload followed by the source as the first byte is created and output serially from the serial parser serparser_attach pout
for output.
Test data must be entered into the terminal using the paste function. This is because there is a timeout for input.
Note: To paste in TWE Programmer or TeraTerm, use Alt+V.
Send 00112233
to any Child Node.
Send AABBCC00112233
to Child Node #3.
Sent to any Parent Node or Child Node (0xFF
) and to the Parent Node (0x00
).
This ACT includes
Sending and receiving wireless packets
Uses the OPEN-CLOSE SENSE PAL to interrupt and wake up when the magnetic sensor is detected and transmit wirelessly.
Uses the sleep function to operate on coin cell batteries.
First, register the board behavior <PAL_MAG>
. When the board behavior is initialized, the sensors and DIOs are initialized. The reason for doing this first is that it is common to check the status of the board's DIP SW, etc., and then configure the network settings and so on.
Here, 3 bits of the 4-bit DIP SW on the board are read and set as the ID of the Child Node; if 0, the Child Node without ID (0xFE
) is assumed.
Set the LED settings. Here the ON/OFF blinking is set to blink every 10ms (in an application that sleeps and has a short wake-up time, this is almost equivalent to setting the LED to turn on during wake-up).
The begin()
function exits the setup()
function (after which TWENET is initialized) and is called just before the first loop()
.
Call sleepNow()
after setup()
ends to perform the first sleep.
Before going to sleep, set the interrupt for the DIO pin of the magnetic sensor. Use pinMode()
, the second parameter is PIN_MODE::WAKE_FALLING
. This is a setting to wake up when the pin state changes from HIGH to LOW.
In line 7, the_twelite.sleep()
executes sleep. The parameter 60000 is the wake-up setting required to reset the watchdog on the TWELITE PAL board. If not reset, a hard reset will be performed after 60 seconds.
When the program wakes up from sleep, wakeup()
is called. After that, loop()
is called each time. Before wakeup()
, each peripheral such as UART and devices on the board are woken up (e.g. resetting the watchdog timer). For example, it restarts the LED lighting control.
Here, in the case of wake-up from the wake-up timer (the_twelite.is_wokeup_by_wktimer()
), sleep is performed again. This is a wake-up only for the purpose of resetting the watchdog timer described above.
In the case of wakeup upon detection of the magnetic sensor, the loop() process will continue.
Here, the DIO of the detected magnetic sensor is checked, packets are sent, and sleep is performed again after packet transmission is complete.
The b_transmit
variable controls the behavior within loop()
. After a successful transmission request, this value is set to 1, and the program waits for the packet transmission to complete.
Check the detection DIO pins of the magnetic sensor. There are two types of detection pins: N-pole detection and S-pole detection. If you simply want to know that a magnet is approaching, you need to know that one of the pins has been detected.
Use the_twelite.is_wokeup_by_dio()
to check the pin of the wakeup factor. The parameter is the pin number. The reason why the return value is stored in uint8_t is to store it in the payload of the packet.
After setting communication conditions and storing data in the payload, transmission is performed.
Then, if b_transmit
is true
during loop()
, a completion check is performed, and if completed, sleepNow()
puts the program to sleep.
The completion of transmission is confirmed by the_twelite.tx_status.is_complete(u8txid)
. The u8txid
is the ID value returned on transmission.
The ACT of the ACT includes the following.
Sending and receiving wireless packets
Uses the motion sensor PAL MOTION SENSE PAL to continuously measure the acceleration of the accelerometer and transmit it wirelessly.
Use the sleep function, to operate on coin cell batteries.
First, register the board BEHAVIOR <PAL_MOT>
. When the board BEHAVIOR is initialized, the sensors and DIOs are initialized. The reason for doing this first is that it is common to check the status of the board's DIP SW, etc., and then configure the network settings, etc.
Here, 3 bits of the 4-bit DIP SW on the board are read and set as the Child Node ID. 0 means no ID (0xFE
).
Set the LED settings. Here the ON/OFF blinking is set to blink every 10ms (in an application that sleeps and has a short wake-up time, this is almost equivalent to setting the LED to turn on during wake-up).
Starts accelerometer measurement. The accelerometer settings (SnsMC3630::Settings
) specify the measurement frequency and measurement range. Here we use 14HZ measurement (SnsMC3630::MODE_LP_14HZ
) with ±4G range (SnsMC3630::RANGE_PLUS_MINUS_4G
).
Once started, the accelerometer takes 14 measurements per second and the values are stored in a FIFO queue inside the sensor. The sensor is notified when 28 measurements have been taken.
The begin()
function exits the setup()
function (after which TWENET is initialized) and is called just before the first loop()
.
Call sleepNow()
after setup()
ends to perform the first sleep.
Before going to sleep, set up an interrupt for the accelerometer's DIO pin, which is generated when the FIFO queue reaches a certain number. The second parameter is PIN_MODE::WAKE_FALLING
. This is a setting to wake up when the pin state changes from HIGH to LOW.
The third line executes sleep with the_twelite.sleep()
. The parameter 60000 is the wake-up setting required to reset the watchdog on the TWELITE PAL board. If not reset, a hard reset will be performed after 60 seconds.
When the accelerometer wakes up from sleep due to a FIFO interrupt from the accelerometer, wakeup()
is called. After that, loop()
is called each time. Before wakeup()
, each peripheral such as UART and devices on the board are woken up (e.g. resetting the watchdog timer). For example, it restarts the LED lighting control.
Here we are initializing variables to be used in loop()
.
Here, the acceleration information stored in the FIFO queue in the accelerometer is retrieved and packet transmission is performed based on this information. After the packet transmission is completed, sleep is executed again.
The b_transmit
variable controls the behavior within loop()
. After a successful transmission request, this value is set to 1, and the program waits for the packet transmission to complete.
First check to see if the sensor is available. Since it is after an interrupted wake-up, it is not normal for it not to be AVAILABLE, and it will go straight to sleep.
It is not used in the wireless transmission packet, but we will check the information on the acceleration taken out.
The measurement results of the accelerometer are stored in a FIFO queue obtained by brd.sns_MC3630.get_que()
.
The structure axis_xyzt
that stores the measurement results of the accelerometer contains the information of three axes (x, y, z) and the sequence number t. The number of samples stored is the size of the queue.
The number of samples stored can be checked by reading the queue size (brd.sns_MC3630.get_que().size()
). Normally 28 samples, but it may go a little further due to processing delays, etc. The first sample is taken from front()'. The first sample can be obtained by
front(). Its sequential number is
front().t`.
Here we will take the average of the samples before taking them out of the queue. Each element of the queue can be accessed with a for statement (for (auto&& v: brd.sns_MC3630.get_que()) { ... }
), where v.x, v.y, v.z
in the for statement is each element. Here, the sum of each element is calculated, and after the for statement, the average is calculated by dividing by the number of elements.
Next, a packet is generated and a request for transmission is made, but because the data volume is large, the packet is divided into two and transmitted twice. Therefore, the sending process is performed twice in the for statement.
The number of samples to be included in the packet to be sent and the sample first sequence number are stored in the first part of the packet payload.
Finally, the acceleration data is stored. While earlier we only referenced each element of the queue to calculate the average value, here we read one sample at a time from the queue and store it in the payload of the packet.
Use .front()
to read the head of the data queue from the accelerometer. After reading, .pop()
is used to release the top of the queue.
The data acquired from the accelerometer is in milli-G units, where 1G is 1000. Since the range is set to ±4G, the data is divided by 2 to be stored within a 12-bit range. To save the number of data, the first 4 bytes store the upper 8 bits of the X, Y and Z axes, and the next 1 byte stores the lower 4 bits of the Z axis, generating a total of 5 bytes.
The transmission ID is stored in the txid[]
array to wait for two transmissions.
Then, if b_transmit
is true
during loop()
, a completion check is performed, and if completed, sleepNow()
puts the program to sleep.
The completion of transmission is confirmed by the_twelite.tx_status.is_complete()
The .txid[]
is the ID value returned on transmission.
This ACT acquires several samples of acceleration data after sleep recovery and sends that data.
The ACT of the "Act" includes the following
Sending and receiving wireless packets
Wake up → start acquiring accelerometer data → wait for accelerometer FIFO interrupt → retrieve accelerometer data → wireless transmission → sleep.
To support MOT PAL or TWELITE CUE, the include part is a macro. Define either USE_PAL_MOT
or USE_CUE
.
Data structure for storing sensor data.
Registers board, configuration, and network behavior objects.
Initialize the Interactive settings mode.
First, adjust the configuration items. Here, we set the title name SETTINGS::appname
to be displayed in menu items, the default value of Application ID SETTINGS::appid_default
, the default value of CHANNEL SETTINGS::ch_default
, the default value of logical device ID default SETTINGS::lid_default
, and .hide_items()
for hidden items.
This sample transitions to Interactive settings mode when the SET pin is LO at startup. If digitalRead(brd.PIN_SET)
confirms that the pin is LO, specify SETTINGS::open_at_start()
. This specification causes the Interactive settings mode screen to appear immediately after exiting SETUP()
. When the screen is displayed, begin()
and loop()
are executed. In this sample, the state STATE::INTERACTIVE
is set so that no action such as sleep is performed during loop()
and nothing is done.
Next, the configuration values are read. To read the configuration values, be sure to execute .reload()
. In this sample, the option bit setting .u32opt1()
is read.
the_twelite
is a class object that manages the basic behavior of the system. This object performs various initializations such as Application ID and CHANNEL within setup()
.
Settings are also made for the network behavior object. The logical device ID (LID) and retransmission settings in Interactive settings mode are reflected.
LED blink settings and other settings.
Called after setup()
is finished. Here, the first sleep is performed. However, if the screen in Interactive settings mode is displayed, it does not sleep.
After waking up, the state variable eState
is set to the initial state INIT. After this, loop()
is executed.
The basic structure of loop()
is <SM_STATE>
state machine state
with switch ... . case clause. The initial state is STATE::INIT
or STATE::INTERACTIVE
.
This is the state of the Interactive settings mode screen when it is displayed. Nothing is done. Interactive settings mode is used for Serial input and output in this screen.
INIT in its initial state.
In state INIT, initialization (clearing the queue for storing results) and initialization of the data structure for storing results is performed. transition to STATE::START_CAPTURE. After setting this transition, the while loop is executed again.
In the state START_CAPTURE, the FIFO acquisition of the MC3630 sensor is started. Here, the FIFO interrupt is set to occur when 4 samples are acquired at 400 Hz.
Set timeout for exception handling and transition to the next state STATE::WAIT_CAPTURE
.
In the state WAIT_CAPTURE, it waits for a FIFO interrupt. When an interrupt occurs and data is stored in the result storage queue, sns_MC3630.available()
becomes true
. Call sns_MC3630.end()
to terminate the process.
Get the number of samples and sample sequence numbers.
Reads and averages all sample data.
Here, the maximum and minimum are obtained for the acquired samples using the iterator corresponding to each axis.
Call .sns_MC3630.get_que().clear()
to clear the data in the queue. If this is not called, subsequent sample acquisitions will not be possible. After that, it transits to the STATE::REQUEST_TX
state.
. is_timeout()
checks for timeout. If timeout occurs, it transits to STATE::EXIT_FATAL
as an error.
In state REQUEST_TX
, the locally defined function TxReq()
is called to process the obtained sensor data and generate and send a transmission packet. The transmission request may fail due to the status of the transmission queue or other factors. If the transmission request succeeds, TxReq()returns as true, but no transmission is performed yet. The
on_tx_comp()` callback is called to complete the transmission.
Also, .clear_flag()
clears the flag to indicate that transmission is complete. At the same time, a timeout is set.
In state STATE::WAIT_TX
, it waits for the completion of wireless packet transmission. The flag is set by the on_tx_comp()
callback function, and .is_flag_ready()
becomes true after it is set.
When a series of operations is completed, it transits to the state STATE::EXIT_NORMAL
and calls the locally defined function sleepNow()
to execute sleep. When an error is detected, it transits to state STATE::EXIT_FATAL
and performs a system reset.
The last step is to generate a packet and request that it be sent. The packet should include the continuation number, the number of samples, the average value of XYZ, the minimum sample value of XYZ, and the maximum sample value of XYZ.
Sleep procedure.
Serial ports should call Serial.flush()
to output all before sleep.
The <SM_SIMPLE>
state machine must do on_sleep()
.
ACT in action.
+
Interactive settings mode -
Digital (button) input -
analog input -
At a minimum, wire M1=GND, DI1:button, DO1:LED.
At a minimum, wire M1=open, DI1:button, DO1:LED.
Include <TWELITE>
in all ACTs. Here is a simple network and board support <BRD_APPTWELITE>
should be included.
set
reflects some of the settings (Application ID, radio channel, etc.) read from Interactive settings mode. The items to be reflected are described in .
The sample in is slightly modified so that the waiting time (approx. 50 ms) during sensor data acquisition is replaced by a sleep wait.
Please see the explanation of the ACT in before the explanation of this ACT.
is an example of an ACT.
In this sample, using I2C sensors on our product ( or ). However you can also use generic I2C devices by editing receiving/sending procedures. Connect a generic I2C device according to the following diagram.
Configuration with "interactive settings mode" -
Building a state machine with "SM_SIMPLE" -
In the MWX Libary, you can choose two different ways to control the I2C line from the . This act is using .
Read is done with the stream operator >>
. There are several other read methods. For details, see for details. When using the stream operator, enter values into pre-declared variables of type uint8_t
, uint16_t
, or uint32_t
. rdr >> u16temp
reads from the 2-byte I2C bus for variables of type uint16_t
and stores them in big-endian format (first byte is upper byte).
is implemented in a very short code that allows simple management of transitions to states, timeouts, and flags. The states are defined in advance as enumerations. In the above example, it is enum class STATE
. The entity of the state machine is declared as SM_SMPLE<STATE> step
with the predefined enum STATE
as a parameter.
In order to make the interactive settings mode configuration items suitable for the application you are describing, initialize . /settings/stg_std.md) to set the initial settings.
The loop()
is controlled by the step
for control. The purpose is to concisely express the sequence of steps from sleep recovery, sensor value acquisition, wireless packet transmission, waiting for transmission to complete, and sleep.
For a sensor that waits for a very long time, it is possible to extend the battery life by writing a process to sleep once here, but this example is omitted because of the complexity of the structure. If necessary, you can use .
First, a timeout check is performed by step.is_timeout()
. The starting point of timeout is step.set_timeout()
. If the timeout does not occur, the if clause is not executed and loop()
is left as is. The next hardware event (often a system timer, , which generates an interrupt every 1 ms. /api-reference/predefined_objs/ticktimer.md), which generates an interrupt every 1ms), the TWELITE microcontroller will be in DOZE mode, where the CPU is in low-power standby.
The is a low-cost function to divide by 100.
connected to UART with products/TWE-Lite-DIP/index.html) etc.
Packets addressed to the Parent Node can also be received by .
Initialize .
is used to acquire sensor values.
Configuring settings via Interactive settings mode -
State transition control by state machine -
or board manipulation via board behavior
Board BEHAVIOR for open/close sensor pal is included.
is used to acquire sensor values.
Interactive settings mode configuration -
State transition control by state machine -
or Board operation with board BEHAVIOR
Board BEHAVIOR is included.
Configuration through Interactive settings mode -
State transition control by state machine -
or board operation with board BEHAVIOR
If USE_PAL_MOT
is defined, the board BEHAVIOR is included.
Define states for sequential processing during loop()
and also step
is declared.
Here is reflected.
setup()
Initialize the sensor instead of the constructor (uncalled due to the limitation of the MWX Library).
begin()
Begin getting values from the sensor. After starting, it is necessary to wait a certain amount of time until the appropriate sensor values are obtained.
get_convtime()
Get the waiting time to acquire the sensor value.
read(int&, int&)
Read the sensor value.
0
Temperature Value (MSB side)
1
Temperature Value (LSB side)
2
CRC8 value of the byte 0 and 1
3
Humidity Value (MSB side)
4
Humidity Value (LSB side)
5
CRC8 value of the byte 3 and 4
Rcv_Unvsl (Universal Reciever)
By running NWK_LAYERED
on twe_twelite.network
and NWK_SIMPLE
on twe_twelite.network2
in the MWX library, packets of different types, including packets of layered tree net (TWELITE PAL, ARIA, etc) You can receive and interpret various types of packets.
However, wireless packets must be on the same CHANNEL and have the same Application ID.
setup()
, loop()
, and the callback function on_rx_packet()
for incoming packets.
These objects are declared in pkt_handler.cpp
and initialized by pnew()
in setup()
. It mainly interprets the payload (data) of the packet.
Two network objects are created. Be sure to NWK_LAYERED
to the_twelite.network
.
In this sample, the important part of the loop()
process is the .refresh()
process, which is performed about every second. Only g_pkt_apptwelite().refresh()
does the duplicate checker timeout. Other objects do nothing.
This is the most important part of this sample code. auto type = rx.get_network_type();
determines the packet type.
mwx::NETWORK::LAYERED
: NWK_LAYERED
layer tree net ket
mwx::NETWORK::SIMPLE
: NWK_SIMPLE
packets
mwx::NETWORK::NONE
: no network (e.g. App_Twelite)
other : error or unsupported packets
In the case of mwx::NETWORK::NONE
, the MWX library does not handle duplicate checkers for the same packet that may be sent in multiple retransmissions. It is necessary to describe the handling of these. In this sample, dup_checker.hpp
and dup_checker.cpp
are provided.
Interpretation of the packet refers to the packet_rx&
object that wraps tsRxDataApp*
. The packet_rx
class itself has no special functionality, it only defines a means of accessing some information obtained from tsRxDataApp*
using get_psRxDataApp()
.
It is defined for the purpose of unifying the interface of the packet interpretation part.
analyze()
: interpret the payload of a packet.
display()
: Display packet information.
refresh()
: describe the process every second.
self()
: cast to the derived class D
.
In addition, the packet interpretation class (example pkt_handler_apptwelite
above) contains a member object pkt
. The actual packet interpretation part is done in pkt_???.cpp
.
A packet interpretation part analyze()
for each packet type and a data structure data
are defined. The member data
is a structure but inherits the common structure of PktDataCommon
. This common part is used to concisely describe the code for serial output of the packet's data.
Corresponds to PAL-related packets; the PAL packet structure has a complex data structure. The implementation here is based on the EASTL container.
_vect_pal_sensors
: pool of _pal_sensor
objects. This object is a dedicated class for use with instusive map.
_map_pal_sensors
: intrusive map structure for efficient retrieval of sensor data.
Allocate an entry in _vect_pal_sensors
for each of the multiple data in a packet as they are added and store the value. Once all the data in a packet has been interpreted, a _map_pal_sensors
is constructed with the sensor type as the key.
Implement a duplicate checker. The behavior of the checker can be customized by template arguments.
MODE
: If MODE_REJECT_SAME_SEQ
is specified, packets with the same sequence number are excluded. This is used when the packet order is reordering. The MODE_REJECT_OLDER_SEQ
adopts a more recent number.
TIMEOUT_ms
: The interval at which to initialize the duplicate database. If 1000
is specified, data after 1 second will be erased. Packets that were excluded immediately before will be adopted again when the duplicate database is initialized.
N_ENTRIES
: The maximum number of elements to be allocated in the data structure.
N_BUCKET_HASH
: Maximum number of hash values. Specifies a prime number. It is determined based on the type of wireless node being received.
_mmap_entries
: It is an intrusive hash multi-map structure. The search key is the serial number of the wireless node.
_vect_pool
: Allocates a fixed number (N_ENTRIES
) of elements used in the map structure.
_ring_vecant_idx
: Keeps track of _vect_pool
elements not used in _mmap_entries
by array index number. It is a ring buffer structure, which takes one value from the ring buffer when adding elements and returns a value to the ring buffer when removing elements.
To retrieve data from the multimap structure, call .equal_range()
. The resulting r
is an iterator that enumerates elements with the same serial number.
Each element (_dup_checker_entry
) has a timestamp and sequence number. Duplicates are checked according to these values.
Class Objects
Class objects are predefined objects in the MWX library, such as the_twelite
, which handles TWENET, and objects for using peripherals.
Each object must be initialized by calling the .setup()
or .begin()
method. (Only Serial objects that use UART0 do not require initialization.)
Digital input management class (mwx::periph_buttons)
Detects changes in digital inputs. This class detects changes when the same detected value is obtained multiple times. This is useful for reducing the effect of chattering on mechanical buttons.
The parameter max_history
is the maximum number of references that can be set with begin()
. Memory is allocated and initialized here.
Starts the Buttons
operation, the first parameter bmPortMask
specifies the bitmap of digital inputs to be monitored. bit 0 corresponds to DIO 0, ... , bit N corresponds to DIO N. More than one can be specified: the second u8HistoryCount
is the number of times required to confirm the value; the third tick_delta
specifies the interval in ms between value checks; the fourth u8HistoryCount
is the number of times to confirm the value.
It will take u8HistoryCount*tick_delta
[ms] to confirm the value. For example, if u8HistoryCount
=5 and tick_delta
=4, it will take at least 20ms to confirm the state.
The confirmation is done in the TickTimer
event handler. Since it is not an interrupt handler, it is affected by delays in processing, etc., but it is sufficient to suppress chattering of mechanical buttons, etc.
Terminates the Buttons
operation.
Returns true
when a change is detected. It is cleared when read()
is executed.
Called when u32port
becomes available. u32portis the bitmap of the current input DIO and
u32changed` is the bitmap of the DIO where the change was detected.
Returns false
if Buttons is not working.
When Buttons starts to operate, the input status of DIO is not yet determined. When the value is determined, it becomes available. At this time, the MSB (bit31) of the bitmap read by read()
is set to 1.
Since this function requires the operation to be determined, it cannot be used to monitor ports whose input values change constantly.
If Buttons is in operation before Sleep, it will resume after it is restored. After resumption, initial determination is performed.
Act/behavior Programming Interface
The MWX library API may be subject to specification changes in the future for the purpose of improvement.
Parent Node
Child Node
Parent
Children
Parent Node
Child Node
Parent Node
Child Node
An ACT (Act) beginning with _Unit_ is used to describe a very single function or to check the operation of a function.
The core class object of TWENET (mwx::twenet)
The the_twelite
object summarizes the procedures for using TWENET and includes procedures for operating the wireless microcontroller, such as basic wireless settings, sleep, and other procedures.
the_twelite
is set up in the setup()
function, which also calls the_twelite.begin()
to start the process. It is not possible to set up the_twelite
except in setup()
.
In the above example, the application ID is set, the communication channel is set, and the receiving circuit is set.
Various procedures are included.
In the above example, two types of behaviors are registered: environmental sensor pal behavior <PAL_AMB>
and simple relay network <NWK_SIMPLE>
. By registering these, hardware such as sensors on the environmental sensor pal can be easily handled. In addition, the complicated handling of wireless packets can be implicitly provided with functions such as relay processing and automatic discarding of duplicate packet arrivals.
The MWX library defines other methods in addition to those introduced here.
Act descriptions include those that are not directly related to the act description, those that are set up but do not function effectively, and those that are used internally.
<<operator
(setting)The <<
operator is used to initialize the object the_twelite
.
The following configuration class objects are used as input, and default values are applied if no configuration is made.
Set the parameter id
to the specified application ID. This is a required specification.
Reading the settings is done with uint32_t the_twelite.get_appid()
.
Set the specified channel number (11
..26
) in parameter ch
.
Reading the settings is done with uint8_t
the_twelite.get_channel()
.
Parameter pw
is set to (0
. 3
) for the radio output. The default is (3: no output attenuation).
Reading the settings is done with uint8_t the_twelite.get_tx_power()
.
If the parameter bEnable
is 1
, the receiver circuit is always activated and can receive radio packets from others. The default is 0
, which is dedicated to transmission only.
Reading the settings is done with uint8_t the_twelite.get_rx_when_idle()
.
Enables the channel manager. If multiple channels are specified, sending/receiving is performed on multiple channels; if 0 is specified for ch2 and ch3, the specification is disabled.
The items that will be reflected are as follows
app_id
channel
tx_power
Number of retransmissions when MAC ack is used
There are other settings in the MWX library code that are irrelevant to the library's functionality at this time or that may cause conflicts if set.
It is executed after preconfiguration (see <<
operator) and registration of the behavior. It is usually placed at the end of the setup()
function.
the_twelite
finished setting up
Behavior initialization
TWENET initialization is also performed after the setup()
function exits. Since many processes are to be executed after setup()
exits, do not do anything other than initialization here.
Changes channel settings. On failure, the channel is not changed and false
is returned.
Obtains the currently set channel number (11..26) from the MAC layer API.
Get the module serial number.
Put the module to sleep.
u32Periodms
Sleep duration[ms]
bPeriodic
Recalculate the next wake-up time based on the previous wake-up time. ※It may be from the current timing for reasons such as the next wake-up timing is imminent.
bRamoff
Set to true
to sleep without retaining RAM (must be reinitialized from setup()
instead of wakeup()
after wakeup)
u8Device
Designation of a wake-up timer to be used for sleep. Specify
TWENET::SLEEP_WAKETIMER_PRIMARY
orTWENET::SLEEP_WAKETIMER_SECONDARY
.
Returns true
if the return factor from sleep is the specified digital pin.
Returns true
if the wake-up timer is the wake-up factor for returning from sleep.
Resets the system. After reset, the process starts from setup()
.
Stops the watchdog timer. Stop the timer if you are going to wait for polling for a long time.
Restart the watchdog timer.
Three behaviors can be registered with the_twelite
, and the following class objects are defined to store them.
board
: Behaviors for board support. Procedures for using each device on the board are added.
app
: Behaviors describing user applications. Behaviors can be written in terms of interrupts, events, and state transitions using state machines. It is also easy to define multiple application descriptions and select an application with completely different behavior at startup.
After registration, the object is retrieved in the same way as at the time of registration.
If an incorrect behavior is specified, a panic operation (infinite loop) occurs and program operation stops.
the_twelite
defines the three class objects board
, network
, and app
mentioned above, but also the following
Notification of transmission completion status.
Returns true
when the packet with the specified ID has completed transmission.
Returns true
when the packet with the specified ID has completed transmission and has been successfully sent.
Retrieve incoming packets.
The received packet data obtained by the read()
method is designed to be overwritten when subsequent packets are received and processed. If you read the data immediately after available()
and do some short processing, this will not be a problem, but as a general rule, read the data, copy the data needed for application use, and finish loop()
promptly. For example, a long delay()
during loop()
will cause incoming packets to be dropped.
Returns true
if there is an incoming packet that has not yet been read.
Read packets.
format input parser for serial input (mwx::serial_parser)
This built-in class is defined as a built-in object, intended to be used for format input via serial port.
It is defined as a mwx::serial_parser<mwx::alloc_heap<uint8_t>>
type that allocates buffer space for internal use from the heap at initialization (begin()
).
PulseCounter (mwx::periph_pulse_counter)
The pulse counter is a circuit that reads and counts pulses even when the microcontroller CPU is not running. There are two systems of pulse counters: PC0 is assigned to PulseCounter0
and PC1 to PulseCounter1
.
PC0 is assigned to PulseCounter0
and PC1 is assigned to PulseCounter1
.
Initializes the object and starts counting. the first parameter refct
is the count number on which interrupts and available decisions are based. When this number is exceeded, it is reported to the application. You can also set refct
to 0. In this case, it is not a sleep wake-up factor.
The second parameter edge
specifies whether the interrupt is rising (PIN_INT_MODE::RISING
) or falling (PIN_INT_MODE::FALLING
).
The third debounce
takes the values 0, 1, 2, or 3. 1, 2, or 3 settings require the same consecutive value to detect a change in value to reduce the effect of noise.
Discontinue detection.
If the specified count (refct
in begin()
) is 0, true
is returned when the count exceeds 1.
If the specified count (refct
in begin()
) is 1 or more, true
is returned when the number of detections exceeds the specified count.
Reads the count value. Resets the count value to 0 after reading.
UART0 port of TWELITE (mwx::serial_jen)
The Serial
object is initialized at system startup with UART0, 115200 bps, and the initialization process is performed in the library. On the user code, it is available from setup()
.
The Serial1
object is provided in the library, but no initialization process is performed; to enable UART1, perform the necessary initialization procedures Serial1.setup(), Serial1.begin()
.
Output may become unstable during setup(), wakeup()
just after startup or flush
process just before sleep.
Initialize objects.
Allocate memory for FIFO buffers for TX/RX
Allocating memory for TWE_tsFILE structure
Serial
(UART0) will automatically call setup()
in the library. There is no need for a user call.
Also, the buffer size of Serial
(UART0) is determined at compile time. You can change it by the macro MWX_SER_TX_BUFF
(768 if not specified) or MWX_SER_RX_BUFF
(256 if not specified).
Initialize hardware.
The Serial
(UART0) has an automatic begin()
call in the library. No user call is required.
(Not implemented) Stop using hardware.
Get a structure in the TWE_tsFILE*
format used in the C library.
ADC (mwx::periph_analogue.hpp)
Analogue performs ADC and acquires values. It can continuously acquire multiple channels at a time, and can do so sequentially according to a timer or other cycle.
In the standard application (App_Twelite), the pin names ADC2/ADC3 in the semiconductor data sheet are AI3/AI2 to match the TWELITE DIP. Please note this.
*1 ADC2/ADC3 pins for both digital and analog are subject to usage procedures and restrictions.
Assume no pull-up on the pin to be used before the ADC starts. If this is not done, the pull-up voltage will always be observed in the ADC.
In a normal circuit configuration, current leakage occurs during sleep. It cannot be circumvented by software description alone.
To avoid current leakage during sleep, the GND of the analog circuit section should be disconnected with a FET switch or the like and left floating during sleep. Also, before sleep, set the pin to input and pull-up state.
Initializes ADCs. setup() starts the regulator inside the semiconductor, specifies the timer device for periodic execution, and specifies the callback function to be called when all ADCs on the specified channel have completed.
After calling begin()
, the first ADC processing starts immediately, and the processing of the next pin starts from its end interrupt. When all processing is finished (if specified), the callback function is called. It waits until the next timer interrupt occurs before starting a new ADC process.
The second parameter specifies the number of timer interrupts before AC starts. For example, TickTimer
is called every 1 ms, but if you specify 16 for the parameter, it will be processed every 16 ms.
Starts ADC processing with default ADC pins (PIN_ANALOGUE::A1
, PIN_ANALOGUE::A2
). end()
resumes the interrupted ADC processing.
ADC processing is terminated and the regulator inside the semiconductor is stopped.
The value of the ADC is set to true
after the acquisition. After confirmation by this function, it is false
until the next ADC completion.
Reads ADC values. The parameter specifies the ADC pin to be read. read()
returns the read value converted to mV and read_raw() returns the ADC value (0..1023).
After ADC completion (available), if the value is read after a delay until near the timing when the next ADC process is executed, the next ADC value may be returned, because the ADC process is executed by an interrupt handler and the value is updated even during the loop()
process.
The ADC interrupt handler is set to periph_analogue::ADC_handler()
when setup()
is called.
If a different handler is specified by the semiconductor peripheral library, it will not work properly.
If the ADC is in cyclic execution state by begin()
, ADC processing is resumed after returning from sleep.
EEPROM
TWELITE Reads and writes to the built-in EEPROM of the wireless microcontroller.
The built-in EEPROM has 3480 bytes available from address 0x000 to 0xEFF.
Read the data corresponding to address
from EEPROM.
No error detection.
Write value
from EEPROM to address
.
No error detection.
This function is used when you want to reduce the number of rewrites in consideration of the rewrite life of EEPROM.
Obtain a helper object to read and write using mwx::stream
described below.
Interfaces defined in mwx::stream
, such as the <<
operator, can be used on this object.
.seek()
is used to move the EEPROM address to 1024.
The above writes an 8-byte string (00bc614e
), a 4-byte integer (0x12ab34cd
), a 16-byte byte string (HELLO WORLD!....
), and a 1-byte terminating character.
Move the EEPROM address to 1024 using .seek()
.
Reads the data sequence written out earlier. In order, 8-byte characters, 4-byte integers, and 16-byte strings are read out using the >>
operator.
ACT in action.。
1. 2. +
Run the Act on the
- + -
Act in action.
+
Act to work.
+
Unit_ADC
This sample runs the ADC, reading and displaying about every second while continuously running the ADC every 100ms. Sleep by [s]
key.
Unit_I2Cprobe
Scan the I2C bus for responding device numbers (some devices may not respond to this procedure).
Unit_delayMicoroseconds
, and compare with the count of 16Mhz TickTimer.
Unit_brd_CUE
Check the operation of the acceleration sensor, magnetic sensor, and LED of . Input [a]
,[s]
,[l]
keys from the terminal.
Unit_brd_PAL_NOTICE
. All lights flash at startup, then operate via serial input.
- r
,g
,b
,w
: Toggle each color lighting mode
- R
,G
,B
,W
: change the brightness of each color (disabled when lights off or all lights on)
- c
: change the cycle (when blinking)
- C
: change the duty when blinking (when blinking)
Unit_div100
Find the divisor and quotient of 10,100,1000 . -9999999 to 999999 and compare the elapsed time with the usual division by /
,%
.
Unit_div_format
results in string output.
Unit_UART1
UART1 () is a sample of using UART0 () outputs input from UART1 to UART1 and input from UART1 to UART0.
Unit_Pkt_Parser
Sample use of the packet information parser <a href="../It can interpret the output of App_Wings. * You may want to connect two TWELITE radio modules serially and use one as App_Wings and the other to interpret its output. If you want to connect the other to a non-TWELITE radio module, you can use "".
Uint_EEPROM
EEPROM read/write test code.
Unit_Cue_MagBuz
This program uses the TWELITE CUE magnet sensor and SET pin (connected to a piezoelectric buzzer) to sound a buzzer when the magnet is released.
Unit_doint-bhv
Example of BEHAVIOR description for handling IO interrupts.
Unit_EASTL
A collection of small codes using the EASTL library.
It is also possible to register classes that handle wireless networks, classes that summarize board support, and classes that perform event-driven processing described by the user. By registering these classes, dedicated functions can be easily used. These classes are referred to as "" in this explanation.
Reflects the setting .
network
: A behavior that implements a network. Normally, is registered.
network2
: BEHAVIOR that implements networking. Use this behavior if you want another network BEHAVIOR to process packets which were not accepted by the first network
due to the data structure of the payload or other reasons. ()
settings
: Behavior for performing configuration (interactive mode). Registers .
Register behavior . Registration is done within . The return value is a reference to the object corresponding to the registered behavior.
For details, see class for details.
Implement and input/output with UART0 of TWELITE.
The first parameter specifies the port where the ADC is to be made. The port specification is a bitmap with the bits set corresponding to the port numbers mentioned in the pin definition.For example, if you want to get the values of two pins PIN_ANALOGUE::A2
and PIN_ANALOGUE::VCC
, specify (1 <<PIN_ANALOGUE::A1 | 1<<PIN_ANALOGUE::VCC )
. You can also use and write pack_bits(PIN_ANALOGUE::A1,PIN_ANALOGUE::VCC)
.
The first part is , so it is recommended to use the second half of the address when used together. The amount of space consumed by the settings (Interactive settings mode) depends on its implementation. Even with minimal settings, up to 256 bytes are used from the beginning, so use of the later addresses is recommended.
operators and methods. Using mwx::stream
, you can read and write integer types such as uint16_t
and uint32_t
types, read and write fixed-length array types such as uint8_t
, and format them using format()
objects.
0
-
100Khz
1
2
3.7Khz
2
4
2.2Khz
3
8
1.2Khz
buf_tx
FIFO buffer size for TX
buf_rx
FIFO buffer size for RX
speed
Specifies the baud rate of the UART.
config
When the serial_jen::E_CONF::PORT_ALT
bit is specified, UART1 is initialized with DIO14,15. If not specified, initializes UART1 with DIO11(TxD),9(RxD).
uint8_t PIN_ANALOGUE::A1 = 0
ADC1 pin
AI1
uint8_t PIN_ANALOGUE::A2 = 1
ADC2 pin
AI3
uint8_t PIN_ANALOGUE::A3 = 2
uint8_t PIN_ANALOGUE::D0 = 2
ADC3 pin (DIO0) *1
AI2
uint8_t PIN_ANALOGUE::A4 = 3
uint8_t PIN_ANALOGUE::D1 = 3
ADC4 pin (DIO1) *1
AI4
uint8_t PIN_ANALOGUE::VCC = 4
Vcc Supply voltage
bWaitInit
If true is specified, waits for initialization of the regulator inside the semiconductor.
kick_ev
Specify the timer device to be specified for cyclic execution. The following five types of devices can be specified, and AD is started in the interrupt handler except for the first time.
E_AHI_DEVICE_TICK_TIMER (TickTimer)
E_AHI_DEVICE_TIMER0 .. 4 (Timer0 .. 4)
fp_on_finish
This callback function is called from within the interrupt handler after all ADCs on the specified ports have finished, and is used when ADC measurement values are to be stored separately in a FIFO queue, etc.
SPI access (using member functions)
After initializing the hardware by the begin()
method, the bus can be read and written by beginTransaction()
. Executing beginTransaction()
selects the SPI select pin. Read/write is done with the transfer()
function; SPI reads and writes at the same time.
Starts the use of the bus; sets the SPI select pin.
If called with the settings
parameter given, the bus is set.
Terminates the use of the bus; releases the SPI select pin.
Reads and writes the bus. transfer()
transfers 8 bits, transfer16()
transfers 16 bits, and transfer32()
transfers 32 bits.
system timer (mwx::periph_ticktimer)
TickTimer is used for internal control of TWENET and is implicitly executed. The period of the timer is 1ms. Only the available()
method is defined in loop()
for the purpose of describing processing every 1ms by the TickTimer event.
Note that it is not always available in 1ms increments.
There are cases in which events are skipped due to a large delay caused by factors such as the contents of the user program description or the system's internal interrupt processing.
available()
It is set after the TickTimer interrupt occurs and becomes true
in the loop()
immediately following it. It is cleared after loop()
ends.
SPI access (using helper class)
The helper class version is a more abstract implementation. Creating a transceiver
object for reading and writing is the start of using the bus, and destroying the object is the end of using the bus.
By creating the object in the decision expression of the if statement, the validity period of the object is limited to the scope of the if clause, and the object is destroyed when it exits the if clause, at which point the bus usage termination procedure is performed.
In addition, read/write objects implement the mwx::stream
interface, allowing the use of the <<
operator, etc.
The start and end of bus usage is aligned with the validity period of the object to improve the visibility of the source code and to prevent omissions in the termination procedure, etc.
Unify read/write procedures with the mwx::stream
interface
Reading process and its termination procedure in scope if() { ... }
to do the reading with a helper class.
In the above, the x
object created by the get_rwer()
method is used to read and write one byte at a time. 1.
create x
object in if(...)
Generate the x
object in the get_rwer()
method. At the same time, set the select pin of the SPI bus. (The type is resolved with the universal reference auto&&
by type inference. 2.)
the generated x
object has a operator bool ()
defined, which is used to evaluate the decision expression, which is always true
for the SPI bus.
x
object defines uint8_t transfer(uint8_t)
method, which is called to perform 8bit read/write transfer to SPI. 4. 4.if() { ... }
Destructor of x
is called at the end of the scope to release the select pin of the SPI bus.
Obtains the worker object used to read/write the SPI bus.
Each transfers 8-bit, 16-bit, and 32-bit data, and returns the read result with the same data width as the written data width.
The int
and uint8_t
types perform 8-bit transfer.
Types uint16_t
and uint32_t
perform 16-bit and 32-bit transfers, respectively.
The transfer result is stored in an internal FIFO queue of up to 16 bytes and read by the >>
operator. Since the buffer is not large, it is assumed to be read out after each transfer.
Specify a variable with the same data width as the previous transfer.
If the result of the read is not needed, use the null_stream() object; it skips reading by the data byte specified by i. If the result of the read is not needed, use the null_stream() object.
Timers, PWM (mwx::periph_timer)
The timer has two functions: to generate a software interrupt at a specified period, and to output a PWM at a specified period. 5 timers in total are available in the TWELITE radio module, from 0...4.
The name of the built-in object is Timer0..4
, but will be referred to as TimerX
on this page.
Initializes the timer. This call allocates the necessary memory space.
Starts a timer; the first parameter is the period of the timer in Hz; setting the second parameter to true
enables software interrupts; setting the third parameter to true
true` enables PWM output.
Stops the timer operation.
It becomes true
at loop()
immediately after a timer interrupt occurs, and becomes false
when loop()
ends.
Set the duty ratio. the first parameter specifies the duty ratio (a small value makes the average of the waveform closer to the GND level, a large value makes it closer to the Vcc level). the second parameter specifies the maximum duty ratio value of the duty ratio.
It is recommended that duty_max
be one of 1024,4096,16384
.
The internal calculation of the count value involves division, but only for these three values is bit shifting used, while for other values, a computationally expensive division process is performed.
Sets the frequency of the timer; the second parameter is an integer with three decimal places for the frequency. For example, to set the frequency to 10.4 Hz, specify hz=10, mil=400
.
Read/write two-wire serial (I2C) master (mwx::periph_wire)
Reads and writes two-wire serial (I2C) master.
The mwx::periph_wire<MWX_TWOWIRE_RCVBUFF>
can be referred as TwoWire
.
The following definition types describe the types of arguments and return values.
Some APIs make calls where the STOP bit is not strictly handled.
Instance creation and necessary initialization is done in the library. In user code, it is made available by calling Wire.begin()
.
When using the requestFrom()
method, you can specify the size of the FIFO queue for temporary data storage. At compile time, compile the macro MWX_TWOWIRE_BUFF
with the required number of bytes. The default is 32 bytes.
Example:
-DMWX_TWOWIRE_BUFF=16
Initialize hardware.
Performing any Wire operation without initialization will cause the TWELITE radio module to hang.
When waking from sleep, if the module was operating just before sleep, it will return to the previous state.
u8mode
Specifies the bus frequency. Default is 100Khz (WIRE_CONF::WIRE_100KHZ
)
The frequency is specified by WIRE_CONF::WIRE_? KHZ
and ? KHZ
can be 50
,66
,80
,100
,133
,160
,200
,266
,320
,400
.
b_portalt
Change hardware pin assignments.
There are two types of read/write procedures. Select and use one of them.
Checks if the device specified by address
is responding. If the device exists, true
is returned.
The procedure is originally intended to change the bus frequency, but no action is taken.
Reads and writes the SPI bus (as Controller).
Reads and writes the SPI bus (as Controller).
const uint8_t
SPI_CONF::MSBFIRST
MSB as the first bit
const uint8_t
SPI_CONF::LSBFIRST
make LSB the first bit
const uint8_t
SPI_CONF::SPI_MODE0
set SPI MODE 0
const uint8_t
SPI_CONF::SPI_MODE1
set to SPI MODE 1
const uint8_t
SPI_CONF::SPI_MODE2
set to SPI MODE 2
const uint8_t
SPI_CONF::SPI_MODE3
set to SPI MODE 3
The procedure for using the SPI bus depends on the begin()
method.
Reads and writes the SPI bus (MASTER).
Initialize hardware.
This process is also required after sleep recovery.
slave_select
Specify the SPI slave select pin to be used. 0 : DIO19
, 1 : DIO0
(DIO 19 is reserved), 2 : DIO1
(DIO 0,19 is reserved)
settings
Specifies the SPI bus setting.
The clock
[hz] specifies the SPI bus frequency. A divisor closer to the specified frequency will be selected: 16Mhz or 16Mhz divided by an even number.
bitOrder
specifies SPI_CONF::MSBFIRST
or SPI_CONF::LSBFIRST
.
dataMode
specifies SPI_CONF::SPIMODE0..3
.
Terminate the use of SPI hardware.
There are two types of read/write procedures. Select and use one of them.
To describe the processing of the interrupt handler, specify it in the definition.
requestFrom(), beginTransmission(), endTransmission(), write()
reader, writer
beginTransaction(), endTransaction(), transfer(), transfer16(), transfer32()
transceiver
API return value class that wraps 32-bit type. MSB (bit31) is a flag for failure or success. bit0..30 is used to store the return value.
The default constructor is constructed with a combination of false
and 0
.
It can also be explicitly constructed with bool
and uint32_t
types as parameters.
Since the constructor of type bool
is implemented, you can use true
/false
as follows.
Return true
if 1
is set in MSB.
Return true
if MSB is 0
.
Obtain the value part of bit0..30.
This class is a wrapper class for TWENET's tsRxDataApp
structure.
In packet_rx
, in particular, the data payload of the packet can be handled by the smplbuf
container, and utility functions such as expand_bytes()
simplify the payload interpretation description.
Get the data payload of the packet.
Obtain the receiving structure of the TWENET C library.
Returns the data length of the payload. The value is the same as .get_payload().size()
.
Obtain the LQI value (Link Quality Indicator).
Get the address of the sender.
get_addr_src_long()
is the serial number of the sender and MSB(bit31) is always 1.
get_addr_src_lid()
is the logical ID of the sender and takes values from 0x00
-0xFE
(the logical ID specified by <NWK_SIMPLE>
).
Gets the destination address.
The destination address is specified by the source, and the range of values varies depending on the type of destination.
MSB (bit31) is set.
0x00
-0xFF
Logical ID (8bit) is specified as the destination.
0x00
-0xFF
Returns true
for encrypted packets and false
for plaintext.
Returns network type of the packet identified by Network BEHAVIOR.
mwx::NETWORK::LAYERED
packets from <NWK_LAYERED>
mwx::NETWORK::SIMPLE
packets from <NWK_SIMPLE>
mwx::NETWORK::NONE
neworkless packets
others
error or unknow packet type
accessing Wire (using helper class)
The helper class version is a more abstract implementation. Creating objects reader, writer
for reading and writing is the start of using the bus, and destroying the objects is the end of using the bus.
By creating the object in the decision expression of the if statement, the validity period of the object is limited to the scope of the if clause, and the object is destroyed when it exits the if clause, at which point the bus usage termination procedure is performed.
Also, read/write objects implement the mwx::stream
interface, so you can use the <<
operator, etc.
Aligning the start and end of bus usage with the validity period of objects improves the visibility of source code and prevents omissions of termination procedures.
Unification of read/write procedures by mwx::stream
interface
Loading process and its termination procedure in scope if() { ... }
in scope.
The above reads one byte at a time using the rdr
object generated by the get_readr()
method. The parameter of the method is the two-line serial ID to be read. 1.
in if(...) Create a
rdrobject in
. (The type is resolved with the universal reference auto&&
by type inference. 2.) The generated rdr
object defines an operator bool ()
, which is used to evaluate the decision expression. If communication is possible with the specified ID, true
is returned. 3. The rdr
object defines an int operator () (void)
operator, which is called to read one byte of data from the 2-wire serial bus. If the read fails, -1
is returned, and if it succeeds, the read byte value is returned. 4.
if() { ... }
The destructor of rdr
is called at the end of the scope to STOP
the two-wire serial bus.
I2C 読み出しに用いるワーカーオブジェクトを取得します。
Reading with a helper class to perform the writing process and its termination procedure in scope if() { ... }
in scope.
In the above, the wrt
object generated by the get_writer()
method is used to write one byte at a time. The method parameter is the two-line serial ID to be read.
if(...) Generate a
wrtobject in
. (The type name is resolved with auto' because it is long. 2.) The generated
wrtobject defines an
operator bool (), which is used to evaluate the decision expression. If communication is possible with the specified ID,
trueis returned. 3. The
wrtobject defines an
int operator () (void)operator, which is called to write one byte of data to the 2-wire serial bus. If it fails,
-1` is returned, and if it succeeds, the written byte value is returned. 4.
if() { ... }
The destructor of wrt
is called at the end of the scope
to STOP
the two-wire serial bus.
Obtains the worker object used for I2C export.
The int
and uint8_t
types transfer 8 bits.
Write out 1 byte.
Reads only the size of each data type.
Reads 1 byte. Returns -1 if there is an error, returns the byte value read if normal.
If b_stop
is set to true
, the STOP bit is issued on that read.
The following example shows a measurement example of the temperature/humidity sensor SHTC3 of the environmental sensor PAL.
This class object is a wrapper class for callback function or by .
addr
I2C address for reading
read_count
Number of bytes to read (specifying this value will issue a STOP bit on the last transfer); specifying 0 will result in no STOP bit (some devices may work)
addr
I2C address for writing out
Wire (using member function)
The method using member functions has a relatively low level of abstraction and follows the general API system as provided by the C library. The procedures for operating the two-wire serial bus are more intuitive.
However, it is necessary to be explicitly aware of the start and end of bus usage.
Reads the specified number of bytes at once. Since the result of reading is stored in a queue, call the .read()
method immediately afterward until the queue is empty.
u8address
I2C address to be read
length
Number of bytes read
b_send_stop=true
When true
, the STOP
bit is set at the end of reading.
return type size_type
Number of bytes read. 0
means read failure.
Writing is performed by the write()
method after executing beginTransmission()
. Call endTranmission()
after a series of writing is finished.
Initialize the export transfer. Call endTransmission()
as soon as the writing process is finished.
u8address
I2C address to be written out
Writes one byte.
Return value value
Bytes to be written.
Return value size_type
Number of bytes written. A value of 0
is an error.
Writes a byte sequence.
*value
the byte sequence to be written
Return value size_type
Number of bytes to be written.
Return value size_type
Number of bytes written. 0 is an error.
Processes the end of the export.
sendStop = true
Issue the STOP bit.
Return value uint8_t
0: Success 4: Failure
Specified as a template argument of a container class (smplbuf
, smplque
) to allocate or specify an area of memory to be used internally.
This class is not called directly from user code, but is used internally to declare containers.
alloc_attach<T>
specify an already existing buffer
alloc_local<T, int N>
statically allocate a buffer of N bytes internally
alloc_heap<T>
allocate a buffer of the specified size in the heap
In alloc_attach
and alloc_heap
, initialization methods (init_???()
) must be executed according to the memory allocation class.
Initialize with buffer p
and size n
.
Returns the size of the buffer.
This method is used to generate a compile error, like static_assert
, for a method call description that is different from the expected alloc class.
This structure is used to store the values of 3-axis accelerometers, but procedures have been added to increase convenience when stored in a container class.
Generates an iterator that accesses an element on the X, Y, or Z axis using the iterator of the container class containing axis_xyzt
as a parameter.
In the example below, buf.begin()
and buf.end()
are used as iterators for the X axis in the algorithm std::minmax_element
.
This function generates a virtual container class from which one of the XYZ axes of the container class containing axis_xyzt
is taken. Only the begin()
and end()
methods are implemented in this generated class. The iterator that can be obtained by these begin()
and end()
methods is the same as the iterator in the previous section .
This is done by the network behavior .prepare_tx_packet()
.
In the above example, the_twelite.network.use<NWK_SIMPLE>()
retrieves an object of the network behavior. The object pkt
is created by .prepare_tx_packet()
of this object. It is a derived class of packet_tx
, although the type name is inferred by auto&&.
This pkt
object first returns true
or false
in the if()
. A false
return occurs when the queue for sending is full and no more requests can be added.
Various settings are configured in the radio packet to deliver the packet to the destination, including destination information. To configure the settings, an object containing the settings is given as the right-hand side value of the << operator.
The following is a description of the objects used for configuration.
The availability and implications of each setting depend on the network behavior specification.
Specify the destination address addr
. See the Network Behavior specification for the destination address value.
<NWK_SIMPLE>
An address set with an MSB (bit31=0x80000000
) means that it is addressed to the serial number of the radio module. 0x00
. 0xFE means that the address is addressed to the child module (0x01
...0xEF
). 0xFE means broadcast to the child (
0x01...
0xEF), and
0xFF` means broadcast to both the parent and the child.
Specifies the number of retransmissions. The number of retransmissions is specified by u8count. force_retry is a setting to retransmit a specified number of times regardless of whether the transmission is successful or not.
<NWK_SIMPLE>
ネットワークビヘイビア<NWK_SIMPLE>
では、同じ内容のパケットをu8count+1
回送信します。 force_retry
の設定は無視されます。
Sets the delay between sending packets and the retransmission interval. Specify two values, u16DelayMin
and u16DelayMax
, in milliseconds [ms]. Transmission will be started at some point during this interval after a transmission request is made. Specify the retransmission interval as the value of u16RetryDur
in milliseconds[ms]. The retransmission interval is constant.
The transmission process may not start at the specified timing due to internal processing. In addition, IEEE802.15.4 processing also causes time blurring before packets are created. These timing blurs are an effective means of avoiding packet collisions in many systems.
Strict timing of packet transmission should be considered an exception due to the nature of the IEEE802.15.4 standard.
<NWK_SIMPLE>
This specification is valid.
If the same packet is retransmitted and arrives more than 1 second after the first transmission, it is not deduplicated because a new packet has arrived. The same packet may be received after 1 second due to a longer retransmission interval or packet arrival delay in relay. You can configure the handling of duplicate packets by initializing the <NWK_SIMPLE>
behavior.
This setting requests that packet transmission be performed "as quickly as possible." Packet transmission processing in TWENET is performed from the TickTimer, which runs every 1 ms. By setting this parameter, the packet transmission request is processed as soon as possible after the request is made. Of course, any setting other than tx_packet_delay(0,0,0)
is a meaningless specification.
If other packet transmission processing is in progress, it will be processed as usual.
<NWK_SIMPLE>
This designation is valid.
In wireless packet communications, there is a transmission method in which a short radio message called ACK (ACK) is obtained from the destination after the transmission is completed to indicate that the transmission was successful. By setting this option, transmission with ACK is performed.
<NWK_SIMPLE>
This specification is VALID for <NWK_SIMPLE>
. It will result in a compile error. <NWK_SIMPLE>
is intended to implement a simple working relay network and does not communicate with ACK.
Specify broadcast.
<NWK_SIMPLE>
This specification is VALID for <NWK_SIMPLE>
. It will result in a compile error.
Specify the destination address tx_addr(0xFF)
(broadcast) or tx_addr(0xFE)
(broadcast to the child) instead.
Specify the type ID of the TWENET radio packet that can be 0..7.
<NWK_SIMPLE>
This specification is VALID for <NWK_SIMPLE>
. It will result in a compile error.
`` uses type ID internally. Users cannot use it.
This class is a wrapper class for the tsTxDataApp
structure of the TWENET C library, and objects of derived classes based on this class are obtained from network behaviors or .
Parses a sequence of bytes.
The T
specifies the packet type to be parsed. For example, TwePacketTwelite
is specified for 0x81 messages in standard applications.
p
and e
specify the next to the beginning and the end of the byte sequence.
The return value is of type E_PKT
. In case of an error, E_PKT::PKT_ERROR
is returned.
Returns a reference to an object corresponding to the packet type of the interpreted byte sequence. It can be called if parse<T>
was executed beforehand and there were no errors.
The T
can be the same type as the one executed with parse<T>
, or a TwePacket
from which only basic information can be obtained.
pktparser(parser_packet) performs content interpretation on the byte sequence converted by performs content interpretation on the byte sequence converted by .
The above example interprets . parser_ser object converts the message input from Serial into a byte string. This byte string is first identified by to determine the type of the message . Once the message type is determined, the message is parsed by .parse<TwePacketTwelite>()
. The parsed result will be of type TwePacketTwelite
, and .use<TwePacketTwelite>()
is the procedure to extract this object. The TwePacketTwelite
type is a class, but it refers to its member variables directly as a structure.
If the packet cannot be interpreted as a specific packet, E_PKT::PKT_ERROR
is returned.
Determines the type of packet using the packet data byte sequence as input. The return value is .
packet type definition
Corresponds to the following packet
PKT_ERROR
TwePacket does not contain meaningful data such as before packet interpretation or packet type cannot be identified.
PKT_TWELITE
of the standard application App_Twelite is interpreted.
PKT_PAL
Interpret serial format of
PKT_APPIO
of remote control application products/ TWE-APPS/App_IO/uart.html) interpreted by the remote control application .
PKT_APPUART
of the serial communication application products/ TWE-APPS/App_Uart/mode_format.html) is interpreted.
PKT_APPTAG
The UART message of the wireless tag application App_Tag is interpreted. The sensor specific part is not interpreted and the byte string is reported as payload.
PKT_ACT_STD
Output format used in sample, etc.
The TwePacketTwelite
class is a standard application , which is an interpretation of the TwePacketTwelite
class.
The TwePacketAppUart
class is the format in which the extended format of the App_UART is received by the parent and repeater application App_Wings.
Various information in the packet data is stored in DataAppUART
after parse<TwePacketUART>()
execution.
The payload
is the data part, but the method of data storage changes depending on the macro definition.
If MWX_PARSER_PKT_APPUART_FIXED_BUF
is compiled with the value 0
, payload
refers directly to the byte sequence for packet analysis. If the value of the original byte sequence is changed, the data in payload
will be destroyed.
If you define the value of MWX_PARSER_PKT_APPUART_FIXED_BUF
to be greater than 0
, the payload
will allocate a buffer of that value (bytes). However, if the data of the serial message exceeds the buffer size, parse<TwePacketAppUART>()
fails and returns E_PKT::PKT_ERROR
.
It is a base class of packet type, but the member structure common
contains address information and other common information.
The TwePacketPal
class interprets TWELITE PAL packet data. This class handles TWELITE PAL (sensor data and other upstream data) commonly.
PAL common data is defined in DataPal
.
Generator functions are provided to retrieve data specific to each PAL sensor board.
Although the packet data structure of PAL differs depending on the connected sensors, etc., DataPal
holds the data structure of the common part.
The PAL packet data structure consists of two blocks: the common part for all PALs and the individual data part. The individual data part does not interpret the packet, but stores it as is. To simplify handling, data exceeding 32 bytes is stored in dynamically allocated uptr_snsdata
.
The individual data parts are stored in a structure whose base class is PalBase
. This structure is generated by the generator function defined in TwePacketPal
.
If the size fits in MWX_PARSER_PKT_APPPAL_FIXED_BUF
when parse<TwePacketPAL>()
is executed, a separate sensor object is generated.
If it does not fit, a reference to the byte sequence used for analysis is stored in au8snsdata
. In this case, if the data in the byte string used for analysis is rewritten, sensor-specific objects cannot be generated.
All data structures for each sensor in PAL inherit from PalBase
. The sensor data storage status u32StoredMask
is included.
PAL events are information that is sent when certain conditions are met by processing sensor information, rather than directly from sensors. For example, when an acceleration sensor detects a certain level of acceleration from a stationary state.
If event data exists, it can be determined by .is_PalEvent()
of TwePacketPal
being true
, and .get_PalEvent()
will yield a PalEvent
data structure.
Generator functions are used to extract various types of data from sensor PAL.
To use the generator function, first determine if pkt
is an event (.is_PalEvent()
). If it is an event, it has get_PalEvent()
. Otherwise, it creates an object according to u8palpcb
.
If .u8palpcb==E_PAL_PCB::MAG
, the data PalMag
of the open/close sensor pal is taken.
If .u8palpcb==E_PAL_PCB::AMB
, the data PalAmb
of the environmental sensor pal is taken.
If .u8palpcb==E_PAL_PCB::MOT
, the data PalMot
of the operating sensor pal is taken.
Extracts a PalEvent
(PAL event) if .is_PalEvent()
is true
.
This is a container class with an array structure inside. The maximum size of the buffer is determined at initialization, and it behaves as a variable-length array up to that maximum size.
Here is an example of object declaration. Immediately after the declaration, a method call is made for initialization. The maximum size of each object is 128 bytes immediately after initialization, and the size is 0. Use while extending the size as needed.
Alias classes are defined for uint8_t
type.
Elements can be accessed like normal arrays, using the [] operator, etc., and iterators can also be used to access elements.
The class also has push_back()
method, which enables some type of algorithm from standard C++ library.
Declares a container of type T
and size N
. After the declaration, call the initialization method.
smplbuf_local
allocates an area by a fixed array inside. It can also be initialized by the constructor.
In smplbuf_attach
, specify the first pointer T* buf
of the buffer to be used, the initial size size
and maximum size N
of the array. Initialization by the constructor is also possible.
The smplbuf_heap
allocates memory in the HEAP area (a memory area that cannot be released but can be allocated at any time). Once allocated, this area cannot be released, so it is usually defined as a global area. Allocation is done by init_heap()
. Memory allocation by the constructor is not allowed. Please call init_heap()
to use this function.
When creating a global object, initialization by the constructor is not possible. Please call initialization functions init_local()
,attach()
,init_heap()
at the beginning of execution (setup()
is recommended).
Initializer list { ... }
can be used to initialize members. Except for use in the constructor in a local declaration of smplbuf_local
, it is valid after calling the initialization method.
right side value of assignment operator (smplbuf_local
, smplbuf_attach
, smplbuf_heap
)
constructor (local declaration of smplbuf_local
, global declaration is not allowed)
Add a member c
at the end. The return value of append()
is bool
, which returns false
if the buffer is full and cannot be added.
The pop_back()
deletes the trailing entry. However, it does not clear the entry.
empty()
returns true
when no elements are stored in the array. is_end()
returns true
when the array is full.
size()
returns the number of array elements.
capacity()
returns the maximum number of elements in the array.
reserve()
extends the size of the array. The area where the array is not stored is initialized by default.
reserve_hear()
allocates an area of the specified size at the top of the array. This area cannot be referenced by the container object. For example, it can be used as a container to access a sub-array of a packet payload whose header part is skipped. To return the container to the allocated area so that all of it can be accessed again, give it the same negative value as when it was allocated.
redim()
resizes the allocated area. Unlike reserve()
, it does not initialize the unused area.
element.
A negative value for i
makes the element from the end of the buffer. For -1
, the element is the last element, and for -2
, it is one before the end.
Array objects of type uint8_t
(smplbuf<uint8_t, *>
) can be output as is to derived objects of mwx::stream
.
Outputs a sequence of bytes for mwx::stream
derived objects such as Serial
.
Used for output to stream purposes. Used to implement the << operator.
mwx::streamdefines functions and operators to output bytes to a stream, such as
<<operator and
printfmt()method. You can use the stream output procedure with a smplbuf array of type
uint8_t` as the output destination.
There are two methods.
The TwePacketAppIO
class interprets the standard app The TwePacketAppIO
class is the one that interprets the
smplbuf
is a container class that provides array operations on memory areas specified by element type T
and . Since the specification of alloc
is complicated, alias definitions using using
are used.
Using helper object generated by .
Use the smplbuf class that .
Enter printf format in mwx::stream
Helper class for writing format format to the << operator of mwx::stream
. In the library, it is alias defined as Using format=mwx::mwx_format;
.
The maximum number of arguments that can be registered in the variable number argument list is 8. 64-bit parameters such as double and uint64_t type are limited in number. If the limit is exceeded, a compile error will result due to static↵_assert.
Store the argument list received in the constructor in a variable internal to the class using the expand function of the parameter pack.
Call fctprintf()
at the point operator <<
is called to write data to the stream
The constructor stores the format pointer and parameters. The subsequent call with the <<
operator interprets the format and processes the output.
fmt
Format format. See TWESDK/TWENET/current/src/printf/README.md
...
Parameters according to the format format. * The maximum number of parameters is 4. 5 or more parameters will result in a compile error. * Since consistency with the format is not checked, it is not safe for inconsistent input.
The fmt
must remain accessible until this object is destroyed.
Helper object type names are resolved by auto&&
because they are long. The interfaces defined in mwx::stream
, such as <<
operator, can be used for this object.
The generated helper object bs
starts reading/writing from the beginning of the main array b
when it is created. If it is at the end of the array, data is added by append()
. Each time a read/write operation is performed, the position is moved to the next one.
Helper functions can use the >>
operator for reading.
serial format input-output (mwx::serial_parser)
Used for serial format input/output. It has an internal buffer that holds the interpreted binary series. On input, the series is stored in the internal buffer one byte at a time according to the read format, and becomes complete when the interpretation of the series is complete. Conversely, on output, the buffer is output from the internal buffer according to the specified output format.
The type of format to pass in the initialization parameter of begin()
. There are two types here: ASCII format and binary format.
uint8_t PARSER::ASCII = 1
ASCII format
uint8_t PARSER::BINARY = 2
binary format
Binary format is generally more complicated to handle than ASCII format, including the necessary tools and confirmation methods. Normally, ASCII format should be used.
The ASCII format is a way to represent a sequence of data in binary as a string.
For example, 00A01301FF123456
in ASCII format is expressed as follows. The beginning is :
, B1
is the checksum, and the end is [CR:0x0d][LF:0x0a]
.
:00A01301FF123456B1[CR][LF]
You can omit the terminating checksum. Replace the CRLF series from the checksum with X
. This is useful when you want to send data for experiments, etc., although it is less vulnerable to wrong data series due to garbled characters.
:00A01301FF123456X
header
1
Data part
N
2N
Each byte of the original data is represented by two ASCII characters (A-F in upper case).
For example, 0x1F is represented as 1
(0x31) F
(0x46).
checksum
2
The sum of each byte of the data part is calculated in 8-bit width and taken as 2's complement. In other words, the sum of each byte of the data part plus the checksum byte is calculated to be 0 in 8-bit width.
The checksum byte is represented by two ASCII characters.
For example, in 00A01301FF123456
, 0x00 + 0xA0 + ... + 0x56 = 0x4F, the two's complement of which is 0xB1. (i.e. 0x4F + 0xB1 = 0x00)
footer
2
[CR] (0x0D) [LF] (0x0A)
Normally, ASCII format should be used.
However, to check the transmission and reception in experiments, it is necessary to prepare a special terminal that supports binary communication, and checksum calculation is also required. It is more difficult to use than the ASCII format.
Binary format is a method of sending a sequence of data consisting of binary data with a header and a checksum.
For example, 00A01301FF123456
is expressed in binary format as follows.
0xA5 0x5A 0x80 0x08 0x00 0xA0 0x13 0x01 0xFF 0x12 0x34 0x56 0x3D
header
2
Data Length
2
The data length is two bytes in big-endian format, with MSB (0x8000) set and the length of the data part specified.
For example, if the length of the data part is 8 bytes, specify 0x80 0x08
.
Data part
N
N
Specifies the original data.
Checksum
1
Calculates the XOR of each byte of the data part.
For example, if the data part is 00A01301FF123456
, 0x00 xor 0xA0 xor ... 0x56 = 0x3D.
footer
(1)
The checksum is effectively the end of the line. The output from the radio module will be appended with 0x04
(EOT).
The declaration specifies the memory allocation class. Since this specification is complicated, an alias definition is used as described above.
Class name (alias definition) Memory Allocation
Contents
serparser_attach
specify an already existing buffer with begin()
serparser_local<N>
allocate a buffer of N bytes internally
serparser_heap
allocate a heap of the size specified by the parameters of the begin()
method
Calls the begin()
method according to the memory allocation class.
The buffer specified by p
is used in the [format] (ser_parser.md#nitsuite) specified by ty
. The maximum length of the buffer is specified by max_siz
and the effective data length of the buffer by siz
.
This definition is used especially when you want to format output data columns (see >>
operator).
Once allocated, heap space cannot be released.
Returns an internal buffer. The buffer will be of type smplbuf<uint8_t, alloc>
.
Processes input characters. Receives a single byte of input string of formatted input and interprets it according to the format. For example, in ASCII format, it receives a series like :00112233X
as input. X, one byte at a time, and when the last
X` is entered, the interpretation of the format is completed and reported as done.
The parameter to parse()
is the input byte, and the return value is true
if the interpretation is complete.
If true
, reading is completed by parse()
; if false
, interpretation is in progress.
Outputs the internal buffer to the stream (Serial) in a formatted format.
Container class with FIFO queue structure.
Container class with FIFO queue structure.
As a rule, element types are assumed to be structures that store numbers, numerical values, etc. It is not assumed to store objects that need to be destroyed by the destructor (since there is no process to delete the object when the element is deleted from the queue).
A class Intr
can be registered to set interrupt disabling at declaration time. If this class is not specified, normal operation without interrupt disable control is performed.
Example of object declaration. Immediately after the declaration, a method call is made for initialization. The maximum size of any of these objects is 128 bytes immediately after initialization, and the initial size is 0 and nothing is stored. The maximum size cannot be changed.
Since it is a FIFO queue, it is operated using methods such as push()
,pop()
,front()
.
Access by iterator is also possible.
Declares a container of type T
and size N
. After the declaration, call the initialization methods.
The smplque_local
allocates the area by a fixed array inside. Initialization by the constructor is also possible.
In smplque_attach
, specify the first pointer T* buf
of the buffer to be used, the initial size size
and the maximum size N
of the array. Initialization by the constructor is also possible.
smplque_heapallocates memory in the HEAP area (an area of memory that cannot be released but can be allocated at any time). Once allocated, this area cannot be released, so it is usually defined in the global area. Allocation is done by
init_heap(). Memory allocation by the constructor is not allowed. Please call
init_heap()` to use this function.
When creating a global object, initialization by the constructor is not possible (due to compiler limitation). Please call initialization functions init_local()
,attach()
,init_heap()
at the beginning of execution (setup()
is recommended).
push()` adds an entry to the queue.
pop()` removes an entry from the queue.
front()` refers to the first entry (the first one added).
back()` refers to the last entry (the last one added).
pop_front()` refers to the first entry as a return value and deletes it from the queue.
empty()
returns true
if the array contains no elements. is_full()
returns true
when the array is full.
size()
returns the number of elements stored in the queue.
capacity()
returns the maximum number of elements stored in the queue.
Erase all elements of the queue.
element. 0
is the first element added.
You can get an iterator by begin()
and end()
. The beginning of the iterator is the first registered element of the queue. By using the iterator, range for statements and algorithms can be used.
Operators and methods by via a that references a smplbuf array of type uint8_t
.
Three class names are defined according to the memory buffer handling method ().
Initialize with the specified by ty
.
Initialize the heap by allocating the size specified by siz
to the heap in the specified by ty
.
smplque
is a container class that provides FIFO queue operations for memory areas specified by element type T
and . Since the specification of alloc
is complicated, an alias definition using using
is used.
One application is .
Example
The uint8_t
type smplbuf_strm_u8??
is also available in the interface, so several methods for streams can be used.
Flush buffered output to twe::stream.
Flush the output buffer of mwx::stream
. Instance to a helper class that calls the flush()
method.
For serial ports, wait polling until output completes.
For mwx::simpbuf
buffers, output 0x00
at the end (size is not changed)
input-output stream
Upper-level class that handles input/output streams.
Interfaces to several classes (Serial, Wire, SPI, smplbuf
) by polymorphism using CRTP (Curiously Recurring Template Pattern) method.
In CRTP, lower classes are defined as template class Derived : public stream<Derived>;
and methods of lower classes are referenced from upper classes.
This class defines common processing such as print
method, <<
operator, etc., and calls write()
method, etc. implemented in lower classes, which is similar to using virtual functions.
Lower classes implement the functions listed below.
Returns 1 if the input exists, 0 if it does not.
Return value int
0: no data 1: data present
The return value of this implementation is not the buffer length.
Flush output (wait for output to complete).
Inputs 1-byte data from the stream. If the data does not exist, return -1
.
Outputs 1 byte to the stream.
n
The character you want to output.
Return value size_t
1 if output succeeds, 0 if it fails.
This is a static function that produces a single byte output. Since this is not a class method, member variables and other information are not available. Instead, a pointer to the class instance is passed to vp, which is passed as a parameter.
This static function is used internally and the function pointer is passed as a one-byte output function of fctprintf()
. This is used to implement the print
method, etc.
out
the character to output
vp
pointer to a class instance Usually, cast to the original class and call the write() method
Output a single byte.
Performs various types of formatted output.
val
Integer powerups
base
power form BIN binary / OCT 8 math / DEC 10 math / HEX 16 math
place
Number of decimals below the decimal point
back size_t
the number of booklets
Prints output in printf format.
TWESDK/TWENET/current/src/printf/README.md 参照
char
1-byte output (not formatted as a number)
1-byte output (not formatted as a number)
double
numeric output (printf's "%.2f")
uint8_t
output 1 byte (same as char type)
uint16_t
output 2 bytes (big-endian order)
uint32_t
output 4 bytes (big-endian order)
const char*
uint8_t*
const char[S]
Output up to the terminating character. Output does not include the terminating character. >(S
specifies the size of the fixed array)
uint8_t[S]
Output the array size S
bytes as is. (S
is the fixed array size specification)
format()
output in printf format
mwx::crlf
output of newline CRLF
mwx::flush
flush output
bigendian()
output numeric types in big-endian order. (right-hand side value)
std::pair<T*, T*>
A pair containing begin(), end()
pointers of byte type. Can be created by make_pair
. Tis assumed to be of type
uint8_t`. (right-hand side value)
output byte string using bytelist()
std::initializer_list
.
smplbuf<uint8_t,AL>&
smplbuf<uint8_t, AL>::to_stream()
Manages input timeouts and errors using the >>
operator.
The timeout period is specified by set_timeout()
, and input is processed using the >>
operator. If no input is received within the given timeout period, the error value can be read out with get_error_status()
. The error status is cleared by clear_error_status()
.
centisec
Sets the timeout period in units of 1/10 second. If 0xff
is specified, timeout is disabled.
0
No Error
1
Error Status
Input processing.
Cannot be executed within setup()
.
Because it waits for polling, depending on the timeout time setting (e.g. no timeout), the watchdog timer may be triggered and reset.
The following is a list of types that can be read and stored.
1-byte input (big-endian order)
2-byte input (big-endian order)
uint32_t
4-byte input (big-endian order)
uint8_t[S]
input for S
bytes (S
specifies fixed array size)
null_stream(int n)
read n
bytes away
Output the contents of an array class of type uint8_t
. ALC
is .
Outputs data of smplbuf<T>
T
is of type uint8_t
, AL
is a .
stream_helper is a helper object that grants the mwx::stream
interface. It creates a helper object that references a data class and performs data input/output via the helper object.
The following creates a helper object bs
from an array b
of smplbufs and inputs data using the mwx::stream::operator <<()
operator.
stream_helper behaves as if the data array were a stream.
Internally, it holds read/write positions in the data array. It behaves as follows.
After the last data is read or written, it moves to the next read/write position.
After the last data is read or data is appended to the end, the read/write position becomes the end.
If the read/write position is the end of the line, then
available()
becomes false
.
Read cannot be performed.
Writing is appended if it is within the writable range.
Moves the read/write position to the beginning.
Set the read/write position.
MWX_SEEK_SET
set from the leading position. If offset
is 0
, it means the same as rewind()
.
MWX_SEEK_CUR
Move by offset
with respect to the current position.
MWX_SEEK_END
end position. Setting offset
to 0
sets the end position. If -1
is set, it moves to the last character.
Returns the read/write position. Returns -1
for the end position.
Returns 0
if the read/write position is the end. If it is not the end, it returns any other value.
stream_helper is a data class (, ) member functions.
Call back functions describing an application.
This is a callback function that describes the application. Callback means called by the system (library). The user defines several callback functions to describe the behavior of the system.
The following callback functions are mandatory definitions.
setup()
loop()
If no other functions are defined, an empty function that does nothing is linked instead.
Please refer to the source code mwx_appcore.cpp
if you want to see the exact behavior.
Please refer to the source code mwx_appcore.cpp
if you want to see the exact behavior.
the main loop function.
This is the main loop of the application. After the loop ends, the CPU transitions to DOZE mode and waits for the next interrupt with low current consumption.
In the Act description, most of the processing is described in this loop.
SM_SIMPLE is provided to wait for processing such as state transitions, waiting for timeouts, and completion of transmission in the sample code.
The following is a basic code excerpt from SM_SIMPLE.
To use SM_SIMPLE, you need to define an enum class
as a list of states. In the above, it is defined as STATE
. The class object is generated as SM_SIMPLE<STATE> step;
with this stage as a parameter. The generated class object should be initialized by .setup()
.
The initial state of SM_SIMPLE has the value 0, corresponding to STATE::INIT
in the above example. To get the current state, use .state()
and use it as a judgment expression in the switch clause of the do while statement as in the above example.
Call .next()
for state transitions. If the state changes, b_more_loop()
is set to true
and the loop in the do while clause is executed again. In the example, calling .next(STATE::TX)
from the STATE::SENSOR
state will cause the loop to be executed again and the case STATE::TX:
clause will also be executed. If the state is not changed, the do while loop is escaped and the loop()
is terminated once. Wait until the next call to loop()
.
If you want to wait for processing such as completion of transmission, call .clear_flag()
, and then signal the completion of processing by .set_flag(uint32_t)
in another callback function or the like. Parameters of type uint32_t
specified here can be read from .get_flag_value()
.
If you want to process a timeout, you can record the time when you call .set_timeout(uint32_t)
and check if the timeout time has elapsed with .is_timeout()
.
SM_SIMPLE will be used again when returning from sleep, but it should always be called .on_sleep(bool)
before sleep. If you put false
in the parameter, it will start from the 0
state after recovery, and if you put true
, it will resume from the state just before sleep.
The following is the source code for SM_SIMPLE.
Contents may change depending on the version.
The main body will be stored in SM_SIMPLE.hpp in the mwx library resource folder.
It is called only once before the first call to the loop()
function; since TWENET is already initialized, there is no need to consider constraints such as setup()
.
The main usage is to
Displaying startup messages
Writing code for testing
Sleep transition immediately after startup
Processing that is inconvenient for setup()
(radio packet processing, timer operation, etc.)
twe::stream に改行コードを出力する
Instance of a helper class to output a newline code (CR LF) for the <<
operator of mwx::stream
.
the function called when waking up from sleep.
Called before loop()
when waking up from sleep, and includes procedures for initialization after returning from sleep and for branching processing depending on the state of return.
Usually not used.
Called at the re-initialization of code execution, with no peripheral API or initialization.
the setup function to initialize an application.
Called at the beginning of code execution to write initialization code.
TWENET initialization is also executed after the setup()
function exits. Do not do anything other than initialization here, since most of the processing is done after TWENET exits.
Items to be noted are listed below.
Sleep the_twenet.sleep()
cannot be executed. If you want to sleep immediately after initialization, write the first sleep process in the begin()
function.
The delay()
function is replaced by the processing described below. In this case, the parameter ms
does not specify milliseconds. \frz
* Alternative processing for delay()
.
Obtain the system time, [ms].
Obtain the system time, [ms].
The system tick time is updated by TickTimer interrupt.
[BEHAVIOR] (./) is used to describe the Parent Node Child Node.
Child Nodes are described by a state machine.
Uses the environmental sensor PAL AMBIENT SENSE PAL to acquire sensor values.
Use the sleep function to operate with coin cell batteries.
Parent Node
Child Node
When using PAL as the Parent Node, coin cell batteries cannot be used. As a rule of thumb, prepare a power supply environment that can provide a stable current of 50 mA or more.
PAL_AMB-behavior.hpp : Only setup()
is defined. read DIP-SW and if D1..D3 is upper position, it works as Parent Node, otherwise it sets ID corresponding to DIP SW as Child Node.
Parent/myAppBhvParent.hpp : behavior class definition for Parent Node
Parent/myAppBhvParent.cpp : implementation
Parent/myAppBhvParent-handlers.cpp : implementation of handlers
Parent/myAppBhvParent.hpp : behavior class definition for Child Nodes
Parent/myAppBhvParent.cpp : implementation
Parent/myAppBhvParent-handlers.cpp : implementation of handlers
The Parent Node's BEHAVIOR name is <MY_APP_PARENT>
and the Child Node is <MY_APP_CHILD>
.
If the DIP SW reading is 0, register the behavior <MY_APP_PARENT>
for the Parent Node, otherwise register the behavior <MY_APP_CHILD>
for the Child Node.
If the Parent Node is MONOSTICK, the DIP SW for PAL reads 0 and behaves as the Parent Node. However, this behavior is not defined in the MONOSTICK specifications.
The Parent Node behaves as a non-sleeping receiver and outputs packet information to the serial port when it receives a packet from a Child Node.
When a packet is received for the Parent Node, if the first four characters of the packet can be matched (FOURCHARS
), the contents of the packet are displayed.
The Parent Node's interrupt handler blinks the LED.
When the button (5) on the PAL is pressed, the E_ORDER_KICK
event is issued to the state machine.
The state machine is described as a reference for state transitions and is not meaningful for the operation of the application. It executes state transitions by the E_ORDER_KICK event sent from the button, timeouts, and so on.
The behavior flow of the Child Node is the same as that of the PAL_AMB-usenap. From the initial sleep, "wake up → start sensor operation → short sleep → wake up → acquire sensor value → wireless transmission → wait for wireless transmission completion → sleep" is repeated.
The _begin()
function, called from on_begin()
, executes the first sleep.
(*It is acceptable to describe this process directly in on_begin()
without describing it in _begin()
function.)
This is a description of the process of waking up from sleep.
The first time Wire.begin()
is executed here, which is redundant for the second and later times when the device wakes from sleep. This process can be moved to on_begin()
.
Processes E_ORDER_KICK
messages to the state machine upon completion of transmission.
Defines the state name.
This is an example of sensor acquisition implementation for SHTC3. For details on sending commands, etc., refer to the SHTC3 datasheet.
This is an example of LTR308ALS sensor acquisition implementation. Please refer to the LTR308ALS datasheet for details on sending commands, etc.
WireWriteAndGet()
sends 1 byte of cmd
to the device of addr
, then receives 1 byte and returns the value.
State 0 has a special meaning. It is the state immediately after startup or after returning from sleep.
Immediately after startup PEV_is_coldboot(ev,evarg)
judgment becomes true
and is called. Since it goes straight to sleep from on_begin()
, it does not contain any code that transitions the state. **At this point, the major initialization has not yet been completed, so complex processing such as sending wireless packets cannot be performed. **In order to perform the first state transition for such processing, send an event from on_begin()
and perform the state transition according to that event.
After returning from sleep, there is a first call to PEV_is_warmboot(ev,evarg)
which will be true
. A call to PEV_SetState()
will transition to the STATE_SENSOR
state.
When the transition is made from STATE_IDLE
after returning from sleep, the state handler for STATE_SENSOR
is called continuously. The event ev
at this time is E_EVENT_NEW_STATE
.
Here, the operation of two sensors, SHTC3 and LTR308ALS, is started. After a certain period of time, the sensors will be ready to acquire data. This time wait is done with the sleep setting of 66
ms. Note that PEV_KeepStateOnWakeup()
is called before sleep. After this call, the state after returning from sleep is not STATE_IDLE
, but the state it was in when it went to sleep, i.e. STATE_SENSOR
.
When returning from a short sleep, a call is first made with the PEV_is_warmboot(ev,evarg)
decision set to true
. At the time of this call, wireless packets can be sent, etc. Transition to STATE_TX
.
Here, at the time of the E_EVENT_NEW_STATE
event, the sensor data reading and wireless packet transmission procedures are started. Please refer to other act sample examples for details of the transmission procedure.
Unlike the ACT description in the loop, the process of waiting for the message by PEV_Process()
from transmit_complete()
is used as a confirmation of completion. Sleep is performed upon receipt of the message. Sleep processing is done by transitioning to STATE_SLEEP
.
Finally, timeout processing is performed. This is in case the completion message of the sent packet has not been returned. PEV_u32Elaspsed_ms()
returns the elapsed time since the transition to that state in milliseconds [ms]. If the time has elapsed, the above will perform a system reset the_twelite.reset_system()
(assuming this timeout is too much).
Perfom sleep. Describe it in E_EVENT_NEW_STATE
immediately after the transition from the previous state. Since other events may be called just before sleep, be sure to execute the_twelite.sleep()
in a decision expression that is executed only once.
is used to acquire sensor values.
The sensor is described directly using instead of using the function to obtain values.
See , , and before the explanation of this ACT. Also see .
This sample shows how to write . BEHAVIORS are used to describe more complex applications.
+
Build files can be added by /... /install_n_build/makefile.md).
on_tx_comp()
Called when transmission is complete.
This function is called from within the MWX library with data stored in ev
as packet_ev_tx
when the wireless packet transmission is finished. If this function is not defined in the application, a weak function that does nothing is linked.
ev.u8CbId
is the ID at the time of transmission and ev.bStatus
is a flag indicating success (1) or failure (0) of the transmission.
Setting b_handled
to true in this function tells the MWX library that the incoming packet has been processed in the application. If set to processed, it suppresses unnecessary processing. (Do not call event callback functions for the_twelite.app
, .board
, or .settings
)
BEHAVIOR definition requires a class definition as shown below.
The above example defines a BEHAVIOR class with the name MY_APP_CLASS. MY_APP_CLASS must be described in several places.
Define the class name and the base (parent) class. MWX_APPDEFS_CRTP()
is a macro.
Here, the necessary definitions are imported by #include
.
Constructor definition.
This is the main loop and has the same role as loop()
in the global definition.
on_create()
is called at object creation time (use<>()
method). The val
is a parameter for future extension.
on_begin()is called after
setup()ends.
val` is a parameter for future extension.
Called before sleep. val
is a parameter for future extension.
Called in the initial stage when returning from sleep. The val
is a parameter for future expansion.
At this point, the peripherals have not yet been initialized. The sleep wake-up factor can be checked.
Called when waking up from sleep. The val
is a parameter for future extension.
When a packet is received, it is called with the received packet information as rx
.
The transmission information is called as evTx
when packet transmission is completed. The evTx.u8CbId
is the ID at the time of transmission and evTx.bStatus
is the flag indicating success (1
) or failure (0
) of the transmission.
BEHAVIOR handlers (interrupt, event, and state definitions) are defined in a cpp file. The file cannot be split and all handler definitions must be in one file.
Even in the case of BEHAVIORs that do not define handlers, be sure to create the following cpp file.
The required definitions of the MWX library (#include "_mwx_cbs_cpphead.hpp"
) must be included at the beginning and end of the cpp file.
At the beginning of the file, include the .hpp file of the BEHAVIOR definition as shown above. Specify the class name of the behavior in __MWX_APP_CLASS_NAME
. In the above, it is MY_APP_CLASS
.
At the end of the file, include the necessary definitions (#include "_mwx_cbs_cpptail.cpp"
).
The handler definition is written as shown in the following example. Types of definitions are described later. The definition of the handler to be used is described by using the macro for definition. Do not write handlers that are not used.
The MWX_????? _INT()
is the definition of an interrupt handler, and MWX_? _EVENT()
is the definition of an event handler, and MWX_STATE()
is the state definition of a state machine.
Interrupt handlers are executed when a microcontroller interrupt occurs, interrupting the code currently being executed. For this reason, it is desirable to write as short a process as possible, and great care must also be taken with regard to manipulation of variables and the like.
The interrupt handler has a parameter uint8_t& handled
, and setting this value to true
will prevent subsequent event calls from being made.
If the interrupt handler exits with handled
set to false
, the event handler will be called in the application loop (normal code). The event handler has no handled
parameter. Since the event handler is normal code, it can perform relatively large processing. However, the event handler also incurs overhead, so it may not be able to handle the processing that is called at each frequent interrupt. In addition, since events are processed by the system's internal FIFO queue, events may be lost if they cannot be processed within a certain period of time.
The following is an explanation of macros for defining handler functions.
DIO (digital IO) interrupt event. N
specifies the target DIO number. The arg
is a definition for future extension.
TickTimer interrupt and event. The arg
is a definition for future extension.
The handled
flag of the TickTimer must not be set to true
, otherwise TWENET will not work.
Timer interrupt event. TheN
specifies the number of the target timer. The arg
is a definition for future extension.
Definition of other interrupts and events that are not defined standardly in the MWX library and require an understanding of the AHI Peripherals Manual.
Other interrupt events can be received by the following handler functions. These will not be available in the future when dedicated handlers are defined.
Peripheral (AHI) interrupt handler u32DeviceId
corresponds to arg
and u32ItemBitmap
corresponds to arg2
.
A state machine (state machine) is a method of describing an application that receives messages and operates by transitioning its state in response to those messages.
The events to be received are as follows.
E_EVENT_START_UP
It is called at system startup. Immediately after power-on, it is called with 0
parameters. Because it is in the initial stage of execution, PEV_Process()
is called once from the begin() method to start the operation when transitioning to the normal processing state.
It is still called after returning from sleep, but with parameters other than 0
. Normal processing can be performed from this state.
E_EVENT_NEW_STATE
It is called in a new state immediately after a state transition. Describes the process that is first executed when a transition is made to a certain state.
E_EVENT_TICK_TIMER
Called by TickTimer every 1ms
E_EVENT_TICK_SECOND
It is called every second.
The state is set to s
.
Exiting the state handler causes a transition to the next state, followed by a state handler being called with the E_EVENTS_NEW_STATE
event.
Returns the elapsed time ≪ ms] since the state transition. It is used for purposes such as managing timeouts.
In the above example, a system reset is performed after 100 ms.
Called from outside the state handler. Execute the state handler with the event ev
parameter u32evarg
.
The transmission completion event is communicated to the state machine. In other words, call the state handler.
Do not call the state handler directly. It will cause problems such as E_EVENT_NEW_STATE
not being executed.
Set just before sleep. After returning from sleep, the previous state is maintained. That is, the state handler is called with E_EVENT_START_UP
with sleep started.
Determine if the event is E_EVENT_START_UP
on wake-up.
Judges whether the event is E_EVENT_START_UP
when returning from sleep.
BEHAVIOR can be defined by defining the class in the specified way, so that class object. The registered BEHAVIOR will be embedded in TWENET, allowing the user code to describe the application's behavior. It is possible to define callback functions for interrupts and events from TWENET, which is not possible in a loop description. Although it requires more definitions than a loop description, it is suitable for describing more complex applications.
See sample BEHAVIOR .
To generate an interrupt, use , .
In order to generate an interrupt, the is started with software interrupts enabled.
The sample describes the flow of the application's operation, including the start of sensor operation, acquisition of sensor values, wireless packet transmission to completion of transmission, and sleep transition. Please refer to it as an actual example.
DESC
Usually not used.
Called in re-initialization when the peripheral API is not initialized after returning from sleep.
Wait for time by polling (specified in μsec).
It is not included in MWSDK2020_05. Supported packages will be MWSDK_2020_07_UNOFFICIAL or later.
Wait for time by polling (specified in μsec).
Wait for a given period of time in microsec
.
The time is measured by the TickTimer count. When waiting for a long time, the CPU clock is reduced and polling is performed.
In the setup(), wakeup()
function, TickTimer is not yet running, so it waits for a while in a while loop. In this case, the error with the specified value will be large. This loop counter is adjusted to 32Mhz. If the CPU clock is changed in these functions, the error will be proportional to the clock.
If you specify a short time, such as less than 10 for a parameter, the error may be large.
Generates an random number.
Generates an random number.
The first line returns the value of 0.. (maxval-1)
value is returned. Note that the value of maxval is not the maximum value.
The second line returns the value of minval..maxval-1
.
on_rx_packet()
Receives incoming packets.
Setting b_handled
to true in this function tells the MWX library that the incoming packet has been processed in the application. If set to processed', it suppresses unnecessary processing. (Do not process
the_twelite.receiver`)
The the_twelite.receiver
is not recommended.
The receiver was previously processed by the_twelite.receiver
with the intention of describing it in loop()
. However, on_rx_packet()
was added because it is a delayed processing by a queue, which in principle causes overflow, and also because it tends to be complicated to describe.
API for DIO (General-purpose digital IO)
The following functions are used for general-purpose digital IO (DIO) operations.
pinMode()
digitalWrite()
digitalRead()
attachIntDio()
detachIntDio()
The following enumeration values are handled with the type name E_PIN_MODE
.
The following enumeration values are handled with the type name E_PIN_MODE
.
The following enumeration values are handled with the type name E_PIN_STATE
.
The following enumeration values are handled with the type name E_PIN_INT_MODE
.
Waiting for time by polling.
Waiting for time by polling.
The program waits for a given period of time in ms
.
The time is measured by the TickTimer count. When waiting for a long period of time, the CPU clock is decreased and polling is performed.
In the setup(), wakeup()
function, the TickTimer is not yet running, so it waits for a time by a while loop. In this case, the error with the specified value will be large. This loop counter is adjusted to 32Mhz. If the CPU clock is changed in these functions, the error will be proportional to the clock.
If you specify a short time, such as 1 or 2 as a parameter, the error may be large.
When a wireless packet is received, this function is called from within the MWX library with the data stored in pkt
as . If this function is not defined in the application, a weak function that does nothing is linked.
When using , use the callback function in BEHAVIOR.
const uint8_t PIN_DIGITAL::DIO0 .. 19
DIO pins 0 to 19
const uint8_t PIN_DIGITAL::DO0 .. 1
DO pin 0,1
PIN_MODE::INPUT
None
Input
PIN_MODE::OUTPUT
None
Output
PIN_MODE::INPUT_PULLUP
Yes
Input
PIN_MODE::OUTPUT_INIT_HIGH
None
Output(init HIGH)
PIN_MODE::OUTPUT_INIT_LOW
None
Output(init LOW)
PIN_MODE::WAKE_FALLING
None
Input, raised pin, falling
PIN_MODE::WAKE_RISING
None
Input, rising pin, rising
PIN_MODE::WAKE_FALLING_PULLUP
Yes
Input, raised pin, falling
PIN_MODE::WAKE_RISING_PULLUP
Yes
Input, rising pin, rising
PIN_MODE::DISABLE_OUTPUT
Yes
return to the input state
PIN_MODE::OUTPUT
Contribute
PIN_MODE::OUTPUT_INIT_HIGH
Output (initial state HIGH)
PIN_MODE::OUTPUT_INIT_LOW
Output (initial state LOW)
PIN_MODE::DISABLE_OUTPUT
Stop setting output
PIN_STATE::HIGH
1
HIGH(=Vcc) level
PIN_STATE::LOW
0
LOW(=GND) level
PIN_INT_MODE::FALLING
falling edge
PIN_INT_MODE::RISING
rising edge
Change the setting of the digital output pins.
Change the setting of the digital output pins.
The first parameter specifies the pin number to be set, and the second parameter specifies either HIGH
or LOW
.
Sets the DIO (general-purpose digital IO) pin.
Sets the DIO (general-purpose digital IO) pin.
This function allows you to change the state of DIO0..19 and the pins DO0,1. The setting contents are described in the enumeration value of E_PIN_MODE
, description of DIO and Description of DO.
DO0,1 are special pins, which in principle are used for other purposes, but can also be configured as outputs. However, these pins have hardware restrictions, so care must be taken when using them.
Both pins must be guaranteed to be at a HIGH level when power is applied. If the circuit is configured to take unstable voltages, the module will not start up.
Reads the value of the port of the input configuration.
Reads the value of the port of the input configuration.
Get the input value of a pin that has been previously set as an input, either LOW
or HIGH
.
to unregister the interrupt handler.
Unregisters the interrupt handler.
to enable DIO interrupt.
Enables DIO interrupts.
For a preconfigured pin, the first parameter is the pin number for which you want to enable interrupts, the second is the interrupt direction (rising, falling.
Set up an interrupt to be generated when the DIO5 pin changes from HIGH to LOW.
Basic definition of the application behavior myAppClass
. Details are omitted.
Description of the interrupt handler of the application behavior myAppClass
, which inverts the output setting of DIO12 when an interrupt of DIO5 is generated and displays *
on the serial port Serial
for events occurring after the interrupt handler is finished.
Interrupt handlers and event handlers are written in .
Reads the values of all ports in the input settings at once.
Included in mwx library 0.1.4 or later
Reads the values of all ports in the input settings at once.
The values are stored in the order of DIO0 ... DIO19 from the LSB side. DIO19 are stored in this order.
The pins on the HIGH side are set to 1 and the pins on the LOW side are set to 0.